java加密

来源:互联网 发布:32u网络机柜尺寸 编辑:程序博客网 时间:2024/05/22 17:24

加密,是以某种特殊的算法改变原有的信息数据,使得未授权的用户即使获得了已加密的信息,但因不知解密的方法,仍然无法了解信息的内容。大体上分为双向加密和单向加密,而双向加密又分为对称加密和非对称加密(有些资料将加密直接分为对称加密和非对称加密)。

双向加密大体意思就是明文加密后形成密文,可以通过算法还原成明文。而单向加密只是对信息进行了摘要计算,不能通过算法生成明文,单向加密从严格意思上说不能算是加密的一种,应该算是摘要算法吧。


**一、双向加密
(一)、对称加密
采用单钥密码系统的加密方法,同一个密钥可以同时用作信息的加密和解密,这种加密方法称为对称加密,也称为单密钥加密。
需要对加密和解密使用相同密钥的加密算法。由于其速度,对称性加密通常在消息发送方需要加密大量数据时使用。对称性加密也称为密钥加密。
所谓对称,就是采用这种加密方法的双方使用方式用同样的密钥进行加密和解密。密钥是控制加密及解密过程的指令。

算法是一组规则,规定如何进行加密和解密。因此对称式加密本身不是安全的。   
常用的对称加密有:DES、IDEA、RC2、RC4、SKIPJACK、RC5、AES算法等

对称加密一般java类中中定义成员

 import java.security.InvalidKeyException;  import java.security.NoSuchAlgorithmException;   import java.security.Security;  import javax.crypto.BadPaddingException;  import javax.crypto.Cipher;  import javax.crypto.IllegalBlockSizeException;  import javax.crypto.KeyGenerator;  import javax.crypto.NoSuchPaddingException;  import javax.crypto.SecretKey;  public class EncrypDES3 {      // KeyGenerator 提供对称密钥生成器的功能,支持各种算法      private KeyGenerator keygen;      // SecretKey 负责保存对称密钥      private SecretKey deskey;      // Cipher负责完成加密或解密工作      private Cipher c;      // 该字节数组负责保存加密的结果      private byte[] cipherByte;      public EncrypDES3() throws NoSuchAlgorithmException, NoSuchPaddingException {          Security.addProvider(new com.sun.crypto.provider.SunJCE());          // 实例化支持DES算法的密钥生成器(算法名称命名需按规定,否则抛出异常)          keygen = KeyGenerator.getInstance("DESede");          // 生成密钥          deskey = keygen.generateKey();          // 生成Cipher对象,指定其支持的DES算法          c = Cipher.getInstance("DESede");      }      /**      * 对字符串加密      *       * @param str      * @return      * @throws InvalidKeyException      * @throws IllegalBlockSizeException      * @throws BadPaddingException      */      public byte[] Encrytor(String str) throws InvalidKeyException,              IllegalBlockSizeException, BadPaddingException {          // 根据密钥,对Cipher对象进行初始化,ENCRYPT_MODE表示加密模式          c.init(Cipher.ENCRYPT_MODE, deskey);          byte[] src = str.getBytes();          // 加密,结果保存进cipherByte          cipherByte = c.doFinal(src);          return cipherByte;      }      /**      * 对字符串解密      *       * @param buff      * @return      * @throws InvalidKeyException      * @throws IllegalBlockSizeException      * @throws BadPaddingException      */      public byte[] Decryptor(byte[] buff) throws InvalidKeyException,              IllegalBlockSizeException, BadPaddingException {          // 根据密钥,对Cipher对象进行初始化,DECRYPT_MODE表示加密模式          c.init(Cipher.DECRYPT_MODE, deskey);          cipherByte = c.doFinal(buff);          return cipherByte;      }      /**      * @param args      * @throws NoSuchPaddingException       * @throws NoSuchAlgorithmException       * @throws BadPaddingException       * @throws IllegalBlockSizeException       * @throws InvalidKeyException       */      public static void main(String[] args) throws Exception {          EncrypDES3 des = new EncrypDES3();          String msg ="郭XX-搞笑相声全集";          byte[] encontent = des.Encrytor(msg);          byte[] decontent = des.Decryptor(encontent);          System.out.println("明文是:" + msg);          System.out.println("加密后:" + new String(encontent));          System.out.println("解密后:" + new String(decontent));      }  }  

  1. AES密码学中的高级加密标准(Advanced Encryption Standard,AES),又称 高级加密标准
    Rijndael加密法,是美国联邦政府采用的一种区块加密标准。这个标准用来替代原先的DES,已经被多方分析且广为全世界所使用。经过五年的甄选流程,高级加密标准由美国国家标准与技术研究院(NIST)于2001年11月26日发布于FIPS PUB 197,并在2002年5月26日成为有效的标准。2006年,高级加密标准已然成为对称密钥加密中最流行的算法之一。   该算法为比利时密码学家Joan Daemen和Vincent Rijmen所设计,结合两位作者的名字,以Rijndael之命名之,投稿高级加密标准的甄选流程。(Rijdael的发音近于 “Rhinedoll”。)
    import java.security.InvalidKeyException;      import java.security.NoSuchAlgorithmException;      import java.security.Security;      import javax.crypto.BadPaddingException;      import javax.crypto.Cipher;      import javax.crypto.IllegalBlockSizeException;      import javax.crypto.KeyGenerator;      import javax.crypto.NoSuchPaddingException;      import javax.crypto.SecretKey;      public class EncrypAES {          //KeyGenerator 提供对称密钥生成器的功能,支持各种算法          private KeyGenerator keygen;          //SecretKey 负责保存对称密钥          private SecretKey deskey;          //Cipher负责完成加密或解密工作          private Cipher c;          //该字节数组负责保存加密的结果          private byte[] cipherByte;          public EncrypAES() throws NoSuchAlgorithmException, NoSuchPaddingException{              Security.addProvider(new com.sun.crypto.provider.SunJCE());              //实例化支持DES算法的密钥生成器(算法名称命名需按规定,否则抛出异常)              keygen = KeyGenerator.getInstance("AES");              //生成密钥              deskey = keygen.generateKey();              //生成Cipher对象,指定其支持的DES算法              c = Cipher.getInstance("AES");          }          /**          * 对字符串加密          *           * @param str          * @return          * @throws InvalidKeyException          * @throws IllegalBlockSizeException          * @throws BadPaddingException          */          public byte[] Encrytor(String str) throws InvalidKeyException,                  IllegalBlockSizeException, BadPaddingException {              // 根据密钥,对Cipher对象进行初始化,ENCRYPT_MODE表示加密模式              c.init(Cipher.ENCRYPT_MODE, deskey);              byte[] src = str.getBytes();              // 加密,结果保存进cipherByte              cipherByte = c.doFinal(src);              return cipherByte;          }          /**          * 对字符串解密          *           * @param buff          * @return          * @throws InvalidKeyException          * @throws IllegalBlockSizeException          * @throws BadPaddingException          */          public byte[] Decryptor(byte[] buff) throws InvalidKeyException,                  IllegalBlockSizeException, BadPaddingException {              // 根据密钥,对Cipher对象进行初始化,DECRYPT_MODE表示加密模式              c.init(Cipher.DECRYPT_MODE, deskey);              cipherByte = c.doFinal(buff);              return cipherByte;          }          /**          * @param args          * @throws NoSuchPaddingException           * @throws NoSuchAlgorithmException           * @throws BadPaddingException           * @throws IllegalBlockSizeException           * @throws InvalidKeyException           */          public static void main(String[] args) throws Exception {              EncrypAES de1 = new EncrypAES();              String msg ="郭XX-搞笑相声全集";              byte[] encontent = de1.Encrytor(msg);              byte[] decontent = de1.Decryptor(encontent);              System.out.println("明文是:" + msg);              System.out.println("加密后:" + new String(encontent));              System.out.println("解密后:" + new String(decontent));          }      }  

非对称加密
1976年,美国学者Dime和Henman为解决信息公开传送和密钥管理问题,提出一种新的密钥交换协议,允许在不安全的媒体上的通讯双方交换信息,安全地达成一致的密钥,这就是“公开密钥系统”。相对于“对称加密算法”这种方法也叫做“非对称加密算法”。 与对称加密算法不同,非对称加密算法需要两个密钥:公开密钥(publickey)和私有密钥
(privatekey)。公开密钥与私有密钥是一对如果用公开密钥对数据进行加密,只有用对应的私有密钥才能解密;如果用私有密钥对数据进行加密,那么只有用对应的公开密钥才能解密。因为加密和解密使用的是两个不同的密钥,所以这种算法叫作非对称加密算法。
RSA 公钥加密算法是1977年由Ron Rivest、Adi Shamirh和LenAdleman在(美国麻省理工学院)开发的。RSA取名来自开发他们三者的名字。RSA是目前最有影响力的公钥加密算法,它能够抵抗到目前为止已知的所有密码攻击,已被ISO推荐为公钥数据加密标准。RSA算法基于一个十分简单的数论事实:将两个大素数相乘十分容易,但那时想要对其乘积进行因式分解却极其困难,因此可以将乘积公开作为加密密钥。

import java.security.InvalidKeyException;  import java.security.KeyPair;  import java.security.KeyPairGenerator;  import java.security.NoSuchAlgorithmException;  import java.security.interfaces.RSAPrivateKey;  import java.security.interfaces.RSAPublicKey;  import javax.crypto.BadPaddingException;  import javax.crypto.Cipher;  import javax.crypto.IllegalBlockSizeException;  import javax.crypto.NoSuchPaddingException;  public class EncrypRSA {      /**      * 加密      * @param publicKey      * @param srcBytes      * @return      * @throws NoSuchAlgorithmException      * @throws NoSuchPaddingException      * @throws InvalidKeyException      * @throws IllegalBlockSizeException      * @throws BadPaddingException      */      protected byte[] encrypt(RSAPublicKey publicKey,byte[] srcBytes) throws NoSuchAlgorithmException, NoSuchPaddingException, InvalidKeyException, IllegalBlockSizeException, BadPaddingException{          if(publicKey!=null){              //Cipher负责完成加密或解密工作,基于RSA              Cipher cipher = Cipher.getInstance("RSA");              //根据公钥,对Cipher对象进行初始化              cipher.init(Cipher.ENCRYPT_MODE, publicKey);              byte[] resultBytes = cipher.doFinal(srcBytes);              return resultBytes;          }          return null;      }      /**      * 解密       * @param privateKey      * @param srcBytes      * @return      * @throws NoSuchAlgorithmException      * @throws NoSuchPaddingException      * @throws InvalidKeyException      * @throws IllegalBlockSizeException      * @throws BadPaddingException      */      protected byte[] decrypt(RSAPrivateKey privateKey,byte[] srcBytes) throws NoSuchAlgorithmException, NoSuchPaddingException, InvalidKeyException, IllegalBlockSizeException, BadPaddingException{          if(privateKey!=null){              //Cipher负责完成加密或解密工作,基于RSA              Cipher cipher = Cipher.getInstance("RSA");              //根据公钥,对Cipher对象进行初始化              cipher.init(Cipher.DECRYPT_MODE, privateKey);              byte[] resultBytes = cipher.doFinal(srcBytes);              return resultBytes;          }          return null;      }      /**      * @param args      * @throws NoSuchAlgorithmException       * @throws BadPaddingException       * @throws IllegalBlockSizeException       * @throws NoSuchPaddingException       * @throws InvalidKeyException       */      public static void main(String[] args) throws NoSuchAlgorithmException, InvalidKeyException, NoSuchPaddingException, IllegalBlockSizeException, BadPaddingException {          EncrypRSA rsa = new EncrypRSA();          String msg = "郭XX-精品相声";          //KeyPairGenerator类用于生成公钥和私钥对,基于RSA算法生成对象          KeyPairGenerator keyPairGen = KeyPairGenerator.getInstance("RSA");          //初始化密钥对生成器,密钥大小为1024位          keyPairGen.initialize(1024);          //生成一个密钥对,保存在keyPair中          KeyPair keyPair = keyPairGen.generateKeyPair();          //得到私钥          RSAPrivateKey privateKey = (RSAPrivateKey)keyPair.getPrivate();                       //得到公钥          RSAPublicKey publicKey = (RSAPublicKey)keyPair.getPublic();          //用公钥加密          byte[] srcBytes = msg.getBytes();          byte[] resultBytes = rsa.encrypt(publicKey, srcBytes);          //用私钥解密          byte[] decBytes = rsa.decrypt(privateKey, resultBytes);          System.out.println("明文是:" + msg);          System.out.println("加密后是:" + new String(resultBytes));          System.out.println("解密后是:" + new String(decBytes));      }  } 

**单向加密(信息摘要)
Java一般需要获取对象MessageDigest来实现单项加密(信息摘要)。
MD5 即Message-Digest Algorithm 5(信息-摘要算法 5),用于确保信息传输完整一致。是计算机广泛使用的杂凑算法之一(又译摘要算法、哈希算法),主流编程语言普遍已有MD5实现。将数据(如汉字)运算为另一固定长度值,是杂凑算法的基础原理,MD5的前身有MD2、MD3和MD4。MD5的作用是让大容量信息在用数字签名软件签署私人密钥前被”压缩”成一种保密的格式(就是把一个任意长度的字节串变换成一定长的十六进制数字串)。
除了MD5以外,其中比较有名的还有sha-1、RIPEMD以及Haval等

    import java.security.MessageDigest;      import java.security.NoSuchAlgorithmException;      public class EncrypMD5 {          public byte[] eccrypt(String info) throws NoSuchAlgorithmException{              //根据MD5算法生成MessageDigest对象              MessageDigest md5 = MessageDigest.getInstance("MD5");              byte[] srcBytes = info.getBytes();              //使用srcBytes更新摘要              md5.update(srcBytes);              //完成哈希计算,得到result              byte[] resultBytes = md5.digest();              return resultBytes;          }          public static void main(String args[]) throws NoSuchAlgorithmException{              String msg = "郭XX-精品相声技术";              EncrypMD5 md5 = new EncrypMD5();              byte[] resultBytes = md5.eccrypt(msg);              System.out.println("密文是:" + new String(resultBytes));              System.out.println("明文是:" + msg);          }      }  ----------**SHA 是一种数据加密算法,该算法经过加密专家多年来的发展和改进已日益完善,现在已成为公认的最安全的散列算法之一,并被广泛使用。该算法的思想是接收一段明文,然后以一种不可逆的方式将它转换成一段(通常更小)密文,也可以简单的理解为取一串输入码(称为预映射或信息),并把它们转化为长度较短、位数固定的输出序列即散列值(也称为信息摘要或信息认证代码)的过程。散列函数值可以说时对明文的一种“指纹”或是“摘要”所以对散列值的数字签名就可以视为对此明文的数字签名。**
import java.io.FileInputStream;  import java.io.InputStream;  import java.security.MessageDigest;  public class FileHashUtil {      public static final char[] hexChar = {               '0', '1', '2', '3', '4', '5', '6', '7', '8', '9', 'a', 'b', 'c', 'd', 'e', 'f' };      public static final String[] hashTypes = new String[] { "MD2", "MD5", "SHA1", "SHA-256", "SHA-384", "SHA-512" };      public void MD5File(String fileName) throws Exception{          //String fileName = args[0];          System.out.println("需要获取hash的文件为: " + fileName);          java.util.List<MessageDigest> mds = new java.util.ArrayList<MessageDigest>();          for (String hashType : hashTypes) {              MessageDigest md = MessageDigest.getInstance(hashType);              mds.add(md);          }          InputStream fis = null;          try {              fis = new FileInputStream(fileName);              byte[] buffer = new byte[1024];              int numRead = 0;              while ((numRead = fis.read(buffer)) > 0) {                  for (MessageDigest md : mds) {                      md.update(buffer, 0, numRead);                  }              }          } catch (Exception ex) {              ex.printStackTrace();          } finally {              if (fis != null) {                  fis.close();              }          }          for (MessageDigest md : mds) {              System.out.println(md.getAlgorithm() + " == " + toHexString(md.digest()));          }      }      public static void main(String[] args) throws Exception {          String[] fileName = new String[] {"D:/hapfish/ShellFolder.java","D:/hapfish/ShellFolder - 副本.java",                    "E:/ShellFolder - 副本.java","E:/ShellFolder.txt","D:/hapfish/ShellFolder.jpg",                    "E:/ShellFolder增加字符.txt","D:/hapfish/birosoft.jar"};          FileHashUtil files  = new FileHashUtil();          for(int i=0;i<fileName.length;i++){              files.MD5File(fileName[i]);          }       }      public static String toHexString(byte[] b) {          StringBuilder sb = new StringBuilder(b.length * 2);          for (int i = 0; i < b.length; i++) {              sb.append(hexChar[(b[i] & 0xf0) >>> 4]);              sb.append(hexChar[b[i] & 0x0f]);          }          return sb.toString();      }  }  "D:/hapfish/ShellFolder.java",  "D:/hapfish/ShellFolder - 副本.java",  "E:/ShellFolder - 副本.java",  "E:/ShellFolder.txt",  "D:/hapfish/ShellFolder.jpg",  以上五个文件是同一文件经过复制、改扩展名的,最后计算哈希结果是一致的。  "E:/ShellFolder增加字符.txt" 增加了几个字符串,就不一样了  "D:/hapfish/birosoft.jar" 完全不相关的另外一个文件  
需要获取hash的文件为: D:/hapfish/ShellFolder.java  MD2 == 3a755a99c5e407005cd45ebd856b4649  MD5 == 5d08d440fa911d1e418c69a90b83cd86  SHA1 == 522c8c4f4ff1dd669e251c2ab854c3033a51ca63  SHA-256 == d1feb0c73c10a759e88bd240cb9d56d0598b4ff83a0704c6679f7ba12f6c4d99  SHA-384 == 8f8c9da4cd7241c58af3c52b49199033f2dcf3d67f421753999f87511618d9ea2d738e8c16b9b68a7572d06108ff10f6  SHA-512 == 4711579daee3ddacbaea189310348956cb43bcaaf0099f3be047b06f16c1a20a6b71ee3a4ee018128d647e9f2ef0d644747672238e49a8da3d0cd26dfe597458  需要获取hash的文件为: D:/hapfish/ShellFolder - 副本.java  MD2 == 3a755a99c5e407005cd45ebd856b4649  MD5 == 5d08d440fa911d1e418c69a90b83cd86  SHA1 == 522c8c4f4ff1dd669e251c2ab854c3033a51ca63  SHA-256 == d1feb0c73c10a759e88bd240cb9d56d0598b4ff83a0704c6679f7ba12f6c4d99  SHA-384 == 8f8c9da4cd7241c58af3c52b49199033f2dcf3d67f421753999f87511618d9ea2d738e8c16b9b68a7572d06108ff10f6  SHA-512 == 4711579daee3ddacbaea189310348956cb43bcaaf0099f3be047b06f16c1a20a6b71ee3a4ee018128d647e9f2ef0d644747672238e49a8da3d0cd26dfe597458  需要获取hash的文件为: E:/ShellFolder - 副本.java  MD2 == 3a755a99c5e407005cd45ebd856b4649  MD5 == 5d08d440fa911d1e418c69a90b83cd86  SHA1 == 522c8c4f4ff1dd669e251c2ab854c3033a51ca63  SHA-256 == d1feb0c73c10a759e88bd240cb9d56d0598b4ff83a0704c6679f7ba12f6c4d99  SHA-384 == 8f8c9da4cd7241c58af3c52b49199033f2dcf3d67f421753999f87511618d9ea2d738e8c16b9b68a7572d06108ff10f6  SHA-512 == 4711579daee3ddacbaea189310348956cb43bcaaf0099f3be047b06f16c1a20a6b71ee3a4ee018128d647e9f2ef0d644747672238e49a8da3d0cd26dfe597458  需要获取hash的文件为: E:/ShellFolder.txt  MD2 == 3a755a99c5e407005cd45ebd856b4649  MD5 == 5d08d440fa911d1e418c69a90b83cd86  SHA1 == 522c8c4f4ff1dd669e251c2ab854c3033a51ca63  SHA-256 == d1feb0c73c10a759e88bd240cb9d56d0598b4ff83a0704c6679f7ba12f6c4d99  SHA-384 == 8f8c9da4cd7241c58af3c52b49199033f2dcf3d67f421753999f87511618d9ea2d738e8c16b9b68a7572d06108ff10f6  SHA-512 == 4711579daee3ddacbaea189310348956cb43bcaaf0099f3be047b06f16c1a20a6b71ee3a4ee018128d647e9f2ef0d644747672238e49a8da3d0cd26dfe597458  需要获取hash的文件为: D:/hapfish/ShellFolder.jpg  MD2 == 3a755a99c5e407005cd45ebd856b4649  MD5 == 5d08d440fa911d1e418c69a90b83cd86  SHA1 == 522c8c4f4ff1dd669e251c2ab854c3033a51ca63  SHA-256 == d1feb0c73c10a759e88bd240cb9d56d0598b4ff83a0704c6679f7ba12f6c4d99  SHA-384 == 8f8c9da4cd7241c58af3c52b49199033f2dcf3d67f421753999f87511618d9ea2d738e8c16b9b68a7572d06108ff10f6  SHA-512 == 4711579daee3ddacbaea189310348956cb43bcaaf0099f3be047b06f16c1a20a6b71ee3a4ee018128d647e9f2ef0d644747672238e49a8da3d0cd26dfe597458  需要获取hash的文件为: E:/ShellFolder增加字符.txt  MD2 == f2717c24c6c0e110457bd17221c9ca6c  MD5 == c49e353a7c4c26bd7ccb5e90917c230f  SHA1 == 477c8a9e465bfaa4be42d35c032a17f7e6b42b97  SHA-256 == 9fa18adaf242ebcdc6563922d84c2a163c82e1a24db2eb2b73978ed1f354a8a3  SHA-384 == 4eee8f8e6d64d21c15dc01fa049f4d12a3b8e1d94d87763fe0bea75ab5ea8432fa8251289ece45ee39fe3d36b3c3020c  SHA-512 == e852ec0ff77250be497389d2f5a1818c18bb66106b9905c4ee26fe0d256eb3b77e0ce9a28a84e4b67e4332ba37ec3aa7518148e3a682318c0fc34c391f45c201  需要获取hash的文件为: D:/hapfish/birosoft.jar  MD2 == 38c5e1404718916dec59c33cafc909b3  MD5 == dc3e2cc4fb3949cf3660e0f5f8c3fba3  SHA1 == cde3dc25498afc5a563af0bb0eb54dc45f71bb28  SHA-256 == adf6a961c70c6ea677dff066fc5d896fb0beb4dd442ca0eb619ae1d1b04291e5  SHA-384 == fe7c6b754893c53ebd82bb53703fb5cc32115c9a38f98072f73def90729b271ee3c5c78e258bd9ff5ee5476193c2178b  SHA-512 == a15376f327256a6e049dfbdc5c2ad3a98bffccc6fa92ee01ff53db6b04471ca0f45ca28f76ff4a6911b57825afa046671299141f2499d71f1dac618c92385491  

“`