android 和java平台通用的AES加密解密

转载 2016年08月30日 10:22:05
import java.io.UnsupportedEncodingException;
import java.security.InvalidKeyException;
import java.security.Key;
import java.security.NoSuchAlgorithmException;
import javax.crypto.BadPaddingException;
import javax.crypto.Cipher;
import javax.crypto.IllegalBlockSizeException;
import javax.crypto.KeyGenerator;
import javax.crypto.NoSuchPaddingException;
import javax.crypto.spec.SecretKeySpec;
 
public class AES {
 
    static final String algorithmStr = "AES/ECB/PKCS5Padding";
 
    private static final Object TAG = "AES";
 
    static private KeyGenerator keyGen;
 
    static private Cipher cipher;
 
    static boolean isInited = false;
       
      private static  void init() {
        try { 
                /**为指定算法生成一个 KeyGenerator 对象。
                *此类提供(对称)密钥生成器的功能。
                *密钥生成器是使用此类的某个 getInstance 类方法构造的。
                *KeyGenerator 对象可重复使用,也就是说,在生成密钥后,
                *可以重复使用同一 KeyGenerator 对象来生成进一步的密钥。
                *生成密钥的方式有两种:与算法无关的方式,以及特定于算法的方式。
                *两者之间的惟一不同是对象的初始化:
                *与算法无关的初始化
                *所有密钥生成器都具有密钥长度 和随机源 的概念。
                *此 KeyGenerator 类中有一个 init 方法,它可采用这两个通用概念的参数。
                *还有一个只带 keysize 参数的 init 方法,
                *它使用具有最高优先级的提供程序的 SecureRandom 实现作为随机源
                *(如果安装的提供程序都不提供 SecureRandom 实现,则使用系统提供的随机源)。
                *此 KeyGenerator 类还提供一个只带随机源参数的 inti 方法。
                *因为调用上述与算法无关的 init 方法时未指定其他参数,
                *所以由提供程序决定如何处理将与每个密钥相关的特定于算法的参数(如果有)。
                *特定于算法的初始化
                *在已经存在特定于算法的参数集的情况下,
                *有两个具有 AlgorithmParameterSpec 参数的 init 方法。
                *其中一个方法还有一个 SecureRandom 参数,
                *而另一个方法将已安装的高优先级提供程序的 SecureRandom 实现用作随机源
                *(或者作为系统提供的随机源,如果安装的提供程序都不提供 SecureRandom 实现)。
                *如果客户端没有显式地初始化 KeyGenerator(通过调用 init 方法),
                *每个提供程序必须提供(和记录)默认初始化。
                */
            keyGen = KeyGenerator.getInstance("AES");
        } catch (NoSuchAlgorithmException e) {
            e.printStackTrace();
        }
        // 初始化此密钥生成器,使其具有确定的密钥长度。
        keyGen.init(128); //128位的AES加密
        try {    
                // 生成一个实现指定转换的 Cipher 对象。
            cipher = Cipher.getInstance(algorithmStr);
        } catch (NoSuchAlgorithmException e) {
            e.printStackTrace();
        } catch (NoSuchPaddingException e) {
            e.printStackTrace();
        }
        //标识已经初始化过了的字段
        isInited = true;
    }
 
    private static byte[] genKey() {
        if (!isInited) {
            init();  
        }
        //首先 生成一个密钥(SecretKey),
        //然后,通过这个秘钥,返回基本编码格式的密钥,如果此密钥不支持编码,则返回 null。 
        return keyGen.generateKey().getEncoded();
    }
 
    private static byte[] encrypt(byte[] content, byte[] keyBytes) {
        byte[] encryptedText = null;
        if (!isInited) { 
            init();
        }
        /**
        *类 SecretKeySpec
        *可以使用此类来根据一个字节数组构造一个 SecretKey,
        *而无须通过一个(基于 provider 的)SecretKeyFactory。
        *此类仅对能表示为一个字节数组并且没有任何与之相关联的钥参数的原始密钥有用 
        *构造方法根据给定的字节数组构造一个密钥。
        *此构造方法不检查给定的字节数组是否指定了一个算法的密钥。
        */
        Key key = new SecretKeySpec(keyBytes, "AES");
        try {
                // 用密钥初始化此 cipher。
            cipher.init(Cipher.ENCRYPT_MODE, key);
        } catch (InvalidKeyException e) {
            e.printStackTrace();
        }
        try {
                //按单部分操作加密或解密数据,或者结束一个多部分操作。(不知道神马意思)
            encryptedText = cipher.doFinal(content);
        } catch (IllegalBlockSizeException e) {
            e.printStackTrace();
        } catch (BadPaddingException e) {
            e.printStackTrace();
        }
        return encryptedText;
    }
 
    private static byte[] encrypt(String content, String password) {
        try {
            byte[] keyStr = getKey(password);
            SecretKeySpec key = new SecretKeySpec(keyStr, "AES");
            Cipher cipher = Cipher.getInstance(algorithmStr);//algorithmStr          
            byte[] byteContent = content.getBytes("utf-8");
            cipher.init(Cipher.ENCRYPT_MODE, key);//   ʼ  
            byte[] result = cipher.doFinal(byteContent);
            return result; //     
        } catch (NoSuchAlgorithmException e) {
            e.printStackTrace();
        } catch (NoSuchPaddingException e) {
            e.printStackTrace();
        } catch (InvalidKeyException e) {
            e.printStackTrace();
        } catch (UnsupportedEncodingException e) {
            e.printStackTrace();
        } catch (IllegalBlockSizeException e) {
            e.printStackTrace();
        } catch (BadPaddingException e) {
            e.printStackTrace();
        }
        return null;
    }
  
    private static byte[] decrypt(byte[] content, String password) {
        try {
            byte[] keyStr = getKey(password);
            SecretKeySpec key = new SecretKeySpec(keyStr, "AES");
            Cipher cipher = Cipher.getInstance(algorithmStr);//algorithmStr           
            cipher.init(Cipher.DECRYPT_MODE, key);//   ʼ  
            byte[] result = cipher.doFinal(content);
            return result; //     
        } catch (NoSuchAlgorithmException e) {
            e.printStackTrace();
        } catch (NoSuchPaddingException e) {
            e.printStackTrace();
        } catch (InvalidKeyException e) {
            e.printStackTrace();
        } catch (IllegalBlockSizeException e) {
            e.printStackTrace();
        } catch (BadPaddingException e) {
            e.printStackTrace();
        }
        return null;
    }
     
    private static byte[] getKey(String password) {
        byte[] rByte = null;
        if (password!=null) {
            rByte = password.getBytes();
        }else{
            rByte = new byte[24];
        }
        return rByte;
    }
 
    /**
     * 将二进制转换成16进制
     * @param buf
     * @return
     */
    public static String parseByte2HexStr(byte buf[]) {
        StringBuffer sb = new StringBuffer();
        for (int i = 0; i < buf.length; i++) {
            String hex = Integer.toHexString(buf[i] & 0xFF);
            if (hex.length() == 1) {
                hex = '0' + hex;
            }
            sb.append(hex.toUpperCase());
        }
        return sb.toString();
    }
 
    /**
     * 将16进制转换为二进制
     * @param hexStr
     * @return
     */
    public static byte[] parseHexStr2Byte(String hexStr) {
        if (hexStr.length() < 1)
            return null; 
        byte[] result = new byte[hexStr.length() / 2];
        for (int i = 0; i < hexStr.length() / 2; i++) {
            int high = Integer.parseInt(hexStr.substring(i * 2, i * 2 + 1), 16);
            int low = Integer.parseInt(hexStr.substring(i * 2 + 1, i * 2 + 2),
                    16);
            result[i] = (byte) (high * 16 + low);
        }
        return result;
    }
     
        //注意: 这里的password(秘钥必须是16位的)
    private static final String keyBytes = "abcdefgabcdefg12"; 
     
    /**
    *加密
    */
    public static String encode(String content){
            //加密之后的字节数组,转成16进制的字符串形式输出
        return parseByte2HexStr(encrypt(content, keyBytes));
    }
     
    /**
    *解密
    */
    public static String decode(String content){
            //解密之前,先将输入的字符串按照16进制转成二进制的字节数组,作为待解密的内容输入
        byte[] b = decrypt(parseHexStr2Byte(content), keyBytes);
        return new String(b);
    }
     
    //测试用例
    public static void test1(){
        String content = "hello abcdefggsdfasdfasdf";
        String pStr = encode(content );
        System.out.println("加密前:"+content);
        System.out.println("加密后:" + pStr);
         
        String postStr = decode(pStr);
        System.out.println("解密后:"+ postStr );
    }
     
    public static void main(String[] args) {
        test1();
    }
}

相关文章推荐

android客户端和java服务端用aes加密结果不一样的解决方法。

最近在公司做一个项目,老大让我们实现加解密的方法,我把工作直接推给了java服务端,他们也是直接在网上copy的代码,说我直接放到我的android代码中就可以了,不需要太多的更改。我就照做了,但是在...

Java,Android使用RSA 算法加解密

RSA算法:RSA公钥加密算法是1977年由罗纳德·李维斯特(Ron Rivest)、阿迪·萨莫尔(Adi Shamir)和伦纳德·阿德曼(Leonard Adleman)一起提出的。当时他们三人都在...
  • mytfapp
  • mytfapp
  • 2015年08月05日 10:46
  • 1070

java Base64加密实现

Base64是网络上最常见的用于传输的8bit字节代码的编码方式,也常将把需要的二进制数据编码放在url中的形式上传,由于可读性差,所编码的数据不会被直接看到信息详情! Base64 是采用字符替换的...
  • axuanqq
  • axuanqq
  • 2016年05月15日 10:58
  • 1042

AES加密解密,iOS,Android,Java,.Net通用

  • 2015年08月28日 23:16
  • 246KB
  • 下载

Android和java两平台AES的互相加密解密

Android和java两平台AES的互相加密解密
  • xwqqq
  • xwqqq
  • 2016年04月08日 15:03
  • 681

php与java通用AES加密解密算法

AES指高级加密标准(Advanced Encryption Standard),是当前最流行的一种密码算法,在web应用开发,特别是对外提供接口时经常会用到,下面是我整理的一套php与java通用的...
  • niucsd
  • niucsd
  • 2015年12月08日 14:05
  • 169

[转]php与java通用AES加密解密算法

AES指高级加密标准(Advanced Encryption Standard),是当前最流行的一种密码算法,在web应用开发,特别是对外提供接口时经常会用到,下面是我整理的一套php与java通用的...

OC与JAVA通用的AES加密解密

最近项目中用到AES加密,但在网上找了很多的库都是OC与JAVA加密后不能项目解密,因为我们的服务器是用java写的,所以不能通用对于做iOS的就是个大麻烦,Android就比较悠哉用java写所以没...
  • qqMCY
  • qqMCY
  • 2014年09月22日 22:27
  • 3411

php与java通用AES加密解密算法

AES指高级加密标准(Advanced Encryption Standard),是当前最流行的一种密码算法,在web应用开发,特别是对外提供接口时经常会用到,下面是我整理的一套php与java通用的...

9、Android与IOS通用AES加密解密方法

1、Android (1)加密类,Base64Encoder.java package com.example.aes256; import java.io.*; public class B...
内容举报
返回顶部
收藏助手
不良信息举报
您举报文章:android 和java平台通用的AES加密解密
举报原因:
原因补充:

(最多只允许输入30个字)