出自: http://blog.csdn.net/hongtashan11/article/details/6599645

    http://www.cnblogs.com/liunanjava/p/4297854.html

  随着对称密码的发展,DES数据加密标准算法由于密钥长度较小(56位),已经不适应当今分布式开放网络对数据加密安全性的要求,因此1997年NIST公开征集新的数据加密标准,即AES[1]。经过三轮的筛选,比利时Joan Daeman和Vincent Rijmen提交的Rijndael算法被提议为AES的最终算法。此算法将成为美国新的数据加密标准而被广泛应用在各个领域中。尽管人们对AES还有不同的看法,但总体来说,AES作为新一代的数据加密标准汇聚了强安全性、高性能、高效率、易用和灵活等优点。AES设计有三个密钥长度:128,192,256位,相对而言,AES的128密钥比DES的56密钥强1021倍[2]。AES算法主要包括三个方面:轮变化、圈数和密钥扩展。本文以128为例,介绍算法的基本原理;结合AVR汇编语言,实现高级数据加密算法AES。  AES是分组密钥,算法输入128位数据,密钥长度也是128位。用Nr表示对一个数据分组加密的轮数(加密轮数与密钥长度的关系如表1所列)。每一轮都需要一个与输入分组具有相同长度的扩展密钥Expandedkey(i)的参与。由于外部输入的加密密钥K长度有限,所以在算法中要用一个密钥扩展程序(Keyexpansion)把外部密钥K扩展成更长的比特串,以生成各轮的加密和解密密钥。
  1.1圈变化
  AES每一个圈变换由以下三个层组成:
  非线性层——进行Subbyte变换;
  线行混合层——进行ShiftRow和MixColumn运算;
  密钥加层——进行AddRoundKey运算。
  ① Subbyte变换是作用在状态中每个字节上的一种非线性字节转换,可以通过计算出来的S盒进行映射。
  ② ShiftRow是一个字节换位。它将状态中的行按照不同的偏移量进行循环移位,而这个偏移量也是根据Nb的不同而选择的[3]。
  ③ 在MixColumn变换中,把状态中的每一列看作GF(28)上的多项式a(x)与固定多项式c(x)相乘的结果。 b(x)=c(x)*a(x)的系数这样计算:*运算不是普通的乘法运算,而是特殊的运算,即 b(x)=c(x)·a(x)(mod x4+1) 对于这个运算 b0=02。a0+03。a1+a2+a3 令xtime(a0)=02。a0其中,符号“。”表示模一个八次不可约多项式的同余乘法[3]。
  对于逆变化,其矩阵C要改变成相应的D,即b(x)=d(x)*a(x)。
  ④ 密钥加层运算(addround)是将圈密钥状态中的对应字节按位“异或”。
  ⑤ 根据线性变化的性质[1],解密运算是加密变化的逆变化。这里不再详细叙述。
  1.2轮变化
  对不同的分组长度,其对应的轮变化次数是不同的,如表1所列。
  1.3密钥扩展

  AES算法利用外部输入密钥K(密钥串的字数为Nk),通过密钥的扩展程序得到共计4(Nr+1)字的扩展密钥。它涉及如下三个模块:① 位置变换(rotword)——把一个4字节的序列[A,B,C,D]变化成[B,C,D,A];② S盒变换(subword)——对一个4字节进行S盒代替;③ 变换Rcon——Rcon表示32位比特字[xi-1,00,00,00]。这里的x是(02),如 Rcon[1]=[01000000];Rcon[2]=[02000000];Rcon[3]=[04000000]…… 扩展密钥的生成:扩展密钥的前Nk个字就是外部密钥K;以后的字W[]等于它前一个字W[[i-1]]与前第Nk个字W[[i-Nk]]的“异或”,即W[]=W[[i-1]]W[[i- Nk]]。但是若i为Nk的倍数,则W=W[i-Nk]Subword(Rotword(W[[i-1]]))Rcon[i/Nk]。

AES对称加密和解密

 

  1. package demo.security;
  2.  
  3. import java.io.IOException;
  4. import java.io.UnsupportedEncodingException;
  5. import java.security.InvalidKeyException;
  6. import java.security.NoSuchAlgorithmException;
  7. import java.security.SecureRandom;
  8. import java.util.Base64;
  9. import java.util.Scanner;
  10.  
  11. import javax.crypto.BadPaddingException;
  12. import javax.crypto.Cipher;
  13. import javax.crypto.IllegalBlockSizeException;
  14. import javax.crypto.KeyGenerator;
  15. import javax.crypto.NoSuchPaddingException;
  16. import javax.crypto.SecretKey;
  17. import javax.crypto.spec.SecretKeySpec;
  18.  
  19. import sun.misc.BASE64Decoder;
  20. import sun.misc.BASE64Encoder;
  21.  
  22. /*
  23. * AES对称加密和解密
  24. */
  25. public class SymmetricEncoder {
  26. /*
  27. * 加密
  28. * 1.构造密钥生成器
  29. * 2.根据ecnodeRules规则初始化密钥生成器
  30. * 3.产生密钥
  31. * 4.创建和初始化密码器
  32. * 5.内容加密
  33. * 6.返回字符串
  34. */
  35. public static String AESEncode(String encodeRules,String content){
  36. try {
  37. //1.构造密钥生成器,指定为AES算法,不区分大小写
  38. KeyGenerator keygen=KeyGenerator.getInstance("AES");
  39. //2.根据ecnodeRules规则初始化密钥生成器
  40. //生成一个128位的随机源,根据传入的字节数组
  41. keygen.init(128, new SecureRandom(encodeRules.getBytes()));
  42. //3.产生原始对称密钥
  43. SecretKey original_key=keygen.generateKey();
  44. //4.获得原始对称密钥的字节数组
  45. byte [] raw=original_key.getEncoded();
  46. //5.根据字节数组生成AES密钥
  47. SecretKey key=new SecretKeySpec(raw, "AES");
  48. //6.根据指定算法AES自成密码器
  49. Cipher cipher=Cipher.getInstance("AES");
  50. //7.初始化密码器,第一个参数为加密(Encrypt_mode)或者解密解密(Decrypt_mode)操作,第二个参数为使用的KEY
  51. cipher.init(Cipher.ENCRYPT_MODE, key);
  52. //8.获取加密内容的字节数组(这里要设置为utf-8)不然内容中如果有中文和英文混合中文就会解密为乱码
  53. byte [] byte_encode=content.getBytes("utf-8");
  54. //9.根据密码器的初始化方式--加密:将数据加密
  55. byte [] byte_AES=cipher.doFinal(byte_encode);
  56. //10.将加密后的数据转换为字符串
  57. //这里用Base64Encoder中会找不到包
  58. //解决办法:
  59. //在项目的Build path中先移除JRE System Library,再添加库JRE System Library,重新编译后就一切正常了。
  60. String AES_encode=new String(new BASE64Encoder().encode(byte_AES));
  61. //11.将字符串返回
  62. return AES_encode;
  63. } catch (NoSuchAlgorithmException e) {
  64. e.printStackTrace();
  65. } catch (NoSuchPaddingException e) {
  66. e.printStackTrace();
  67. } catch (InvalidKeyException e) {
  68. e.printStackTrace();
  69. } catch (IllegalBlockSizeException e) {
  70. e.printStackTrace();
  71. } catch (BadPaddingException e) {
  72. e.printStackTrace();
  73. } catch (UnsupportedEncodingException e) {
  74. e.printStackTrace();
  75. }
  76.  
  77. //如果有错就返加nulll
  78. return null;
  79. }
  80. /*
  81. * 解密
  82. * 解密过程:
  83. * 1.同加密1-4步
  84. * 2.将加密后的字符串反纺成byte[]数组
  85. * 3.将加密内容解密
  86. */
  87. public static String AESDncode(String encodeRules,String content){
  88. try {
  89. //1.构造密钥生成器,指定为AES算法,不区分大小写
  90. KeyGenerator keygen=KeyGenerator.getInstance("AES");
  91. //2.根据ecnodeRules规则初始化密钥生成器
  92. //生成一个128位的随机源,根据传入的字节数组
  93. keygen.init(128, new SecureRandom(encodeRules.getBytes()));
  94. //3.产生原始对称密钥
  95. SecretKey original_key=keygen.generateKey();
  96. //4.获得原始对称密钥的字节数组
  97. byte [] raw=original_key.getEncoded();
  98. //5.根据字节数组生成AES密钥
  99. SecretKey key=new SecretKeySpec(raw, "AES");
  100. //6.根据指定算法AES自成密码器
  101. Cipher cipher=Cipher.getInstance("AES");
  102. //7.初始化密码器,第一个参数为加密(Encrypt_mode)或者解密(Decrypt_mode)操作,第二个参数为使用的KEY
  103. cipher.init(Cipher.DECRYPT_MODE, key);
  104. //8.将加密并编码后的内容解码成字节数组
  105. byte [] byte_content= new BASE64Decoder().decodeBuffer(content);
  106. /*
  107. * 解密
  108. */
  109. byte [] byte_decode=cipher.doFinal(byte_content);
  110. String AES_decode=new String(byte_decode,"utf-8");
  111. return AES_decode;
  112. } catch (NoSuchAlgorithmException e) {
  113. e.printStackTrace();
  114. } catch (NoSuchPaddingException e) {
  115. e.printStackTrace();
  116. } catch (InvalidKeyException e) {
  117. e.printStackTrace();
  118. } catch (IOException e) {
  119. e.printStackTrace();
  120. } catch (IllegalBlockSizeException e) {
  121. e.printStackTrace();
  122. } catch (BadPaddingException e) {
  123. e.printStackTrace();
  124. }
  125.  
  126. //如果有错就返加nulll
  127. return null;
  128. }
  129.  
  130. public static void main(String[] args) {
  131. SymmetricEncoder se=new SymmetricEncoder();
  132. Scanner scanner=new Scanner(System.in);
  133. /*
  134. * 加密
  135. */
  136. System.out.println("使用AES对称加密,请输入加密的规则");
  137. String encodeRules=scanner.next();
  138. System.out.println("请输入要加密的内容:");
  139. String content = scanner.next();
  140. System.out.println("根据输入的规则"+encodeRules+"加密后的密文是:"+se.AESEncode(encodeRules, content));
  141.  
  142. /*
  143. * 解密
  144. */
  145. System.out.println("使用AES对称解密,请输入加密的规则:(须与加密相同)");
  146. encodeRules=scanner.next();
  147. System.out.println("请输入要解密的内容(密文):");
  148. content = scanner.next();
  149. System.out.println("根据输入的规则"+encodeRules+"解密后的明文是:"+se.AESDncode(encodeRules, content));
  150. }
  151.  
  152. }

测试结果:

  1. 使用AES对称加密,请输入加密的规则
  2. 使用AES对称加密
  3. 请输入要加密的内容:
  4. 使用AES对称加密
  5. 根据输入的规则使用AES对称加密加密后的密文是:Z0NwrNPHghgXHN0CqjLS58YCjhMcBfeR33RWs7Lw+AY=
  6. 使用AES对称解密,请输入加密的规则:(须与加密相同)
  7. 使用AES对称加密
  8. 请输入要解密的内容(密文):
  9. Z0NwrNPHghgXHN0CqjLS58YCjhMcBfeR33RWs7Lw+AY=
  10. 根据输入的规则使用AES对称加密解密后的明文是:使用AES对称加密

JAVA的对称加密算法AES——加密和解密的更多相关文章

  1. JAVA的非对称加密算法RSA——加密和解密

    原文转载至:https://www.cnblogs.com/OnlyCT/p/6586856.html 第一部分:RSA算法原理与加密解密 一.RSA加密过程简述 A和B进行加密通信时,B首先要生成一 ...

  2. JavaScript前端和Java后端的AES加密和解密(转)

    在实际开发项目中,有些数据在前后端的传输过程中需要进行加密,那就需要保证前端和后端的加解密需要统一.这里给大家简单演示AES在JavaScript前端和Java后端是如何实现加密和解密的. java端 ...

  3. java独立小程序实现AES加密和解密

    一.需求: web项目中配置文件配置的密码是明文的, 现在需要修改成密文, 加密方式采用AES, 于是写了个工具类用于加密和解密. 又因为这个密码是由客户来最终确定, 所以为了部署时方便起见, 写了个 ...

  4. java与C#、.NET AES加密、解密 解决方案

      1.情景展示 Java提供的密钥,C#无法解密. 2.原因分析 在Java中,AES的实际密钥需要用到KeyGenerator 和 SecureRandom,但是C#和.NET 里面没有这2个类, ...

  5. java AES加密、解密(兼容windows和linux)

      java AES加密.解密 CreationTime--2018年7月14日10点06分 Author:Marydon 1.准备工作 updateTime--2018年8月10日15点28分 up ...

  6. Php AES加密、解密与Java互操作的问题

    国内私募机构九鼎控股打造APP,来就送 20元现金领取地址:http://jdb.jiudingcapital.com/phone.html 内部邀请码:C8E245J (不写邀请码,没有现金送) 国 ...

  7. JavaScript前端和Java后端的AES加密和解密

    在实际开发项目中,有些数据在前后端的传输过程中需要进行加密,那就需要保证前端和后端的加解密需要统一.这里给大家简单演示AES在JavaScript前端和Java后端是如何实现加密和解密的. 直接上代码 ...

  8. Java 环境下使用 AES 加密的特殊问题处理

    在 Java 环境下使用 AES 加密,在密钥长度和字节填充方面有一些比较特殊的处理. 1. 密钥长度问题 默认 Java 中仅支持 128 位密钥,当使用 256 位密钥的时候,会报告密钥长度错误 ...

  9. java sm4国密算法加密、解密

      java sm4国密算法加密.解密 CreationTime--2018年7月5日09点20分 Author:Marydon 1.准备工作 所需jar包: bcprov-jdk15on-1.59. ...

随机推荐

  1. [eShopOnContainers 学习系列] - 03 - 在远程 Ubuntu 16.04 上配置开发环境

    直接把 md 粘出来了,博客园的富文本编辑器换成 markdown,没啥效果呀 ,先凑合吧.实在不行换地方   # 在远程 Ubuntu 16.04 上配置开发环境 ## 零.因 为什么要用这么麻烦的 ...

  2. listView使用小技巧P66--P76

    1.设置分割线高度和颜色 android:divider="@android:color/darker_gray" android:dividerHeight="10dp ...

  3. Struts01---入门小案例

    创建web项目    实现的效果! 用户点击页面不同的链接,后台调用不同的代码! 创建两个类实现共同的接口! public interface Action { String execute(); } ...

  4. 几种常见排序算法的C++描述

    基本的排序算法有如下特点: 1.几种容易的算法都是以O(N2)排序的 2.Shell排序编程简单,其也是以O(N2)排序的,在实践中用的很多 3.复杂的排序算法往往都是按照O(NlogN)尽心排序的 ...

  5. C++复制构造函数的实现

    复制构造函数是一种特殊的构造函数,有一般构造函数的特性.它的功能是用一个已知的对象来初始化一个被创建的同类对象.复制构造函数的参数传递方式必须按引用来进行传递,请看实例: #include <i ...

  6. [分享]Google 全球 IP 地址库[Google Global Cache IPs]

    Google 全球 IP 地址库 IP 地址来源:http://www.kookle.co.nr,共计4351个. Bulgaria 93.123.23.1 93.123.23.2 93.123.23 ...

  7. Python之numpy库

    NumPy库知识结构 更多详细内容参考:http://www.cnblogs.com/zhanglin-0/p/8504635.html

  8. struts1和struts2原理解析

    1.struts1和struts2 是2个完全不同的框架 其实struts2核心就是 webwork框架 struts1以ActionServlet作为核心控制器,由ActionServlet负责拦截 ...

  9. python加密之hashlib

    1.强大的hashlib,提供了用于加密相关的操作,代替了md5模块和sha模块,主要提供 SHA1, SHA224, SHA256, SHA384, SHA512 ,MD5 算法 2.hmac模块实 ...

  10. VS 2015 开发Android底部导航条----[实例代码,多图]

      1.废话背景介绍  在Build 2016开发者大会上,微软宣布,Xamarin将被整合进所有版本的Visual Studio之中. 这也就是说,Xamarin将免费提供给所有购买了Visual ...