美文网首页ios进阶iOS传道第三方工具类
AES加密 - iOS与Java的同步实现

AES加密 - iOS与Java的同步实现

作者: WelkinXie | 来源:发表于2016-08-13 01:57 被阅读12007次

    简书文章不再更新、管理。
    评论留言欢迎移步 https://welkinx.com/2016/07/30/10/ :)

    AES是开发中常用的加密算法之一。然而由于前后端开发使用的语言不统一,导致经常出现前端加密而后端不能解密的情况出现。然而无论什么语言系统,AES的算法总是相同的, 因此导致结果不一致的原因在于 加密设置的参数不一致 。于是先来看看在两个平台使用AES加密时需要统一的几个参数。

    • 密钥长度(Key Size)
    • 加密模式(Cipher Mode)
    • 填充方式(Padding)
    • 初始向量(Initialization Vector)

    密钥长度

    AES算法下,key的长度有三种:128、192和256 bits。由于历史原因,JDK默认只支持不大于128 bits的密钥,而128 bits的key已能够满足商用安全需求。因此本例先使用AES-128。(Java使用大于128 bits的key方法在文末提及)

    加密模式

    AES属于块加密(Block Cipher),块加密中有CBC、ECB、CTR、OFB、CFB等几种工作模式。本例统一使用CBC模式。

    填充方式

    由于块加密只能对特定长度的数据块进行加密,因此CBC、ECB模式需要在最后一数据块加密前进行数据填充。(CFB,OFB和CTR模式由于与key进行加密操作的是上一块加密后的密文,因此不需要对最后一段明文进行填充)

    在iOS SDK中提供了PKCS7Padding,而JDK则提供了PKCS5Padding。原则上PKCS5Padding限制了填充的Block Size为8 bytes,而Java实际上当块大于该值时,其PKCS5Padding与PKCS7Padding是相等的:每需要填充χ个字节,填充的值就是χ。

    初始向量

    使用除ECB以外的其他加密模式均需要传入一个初始向量,其大小与Block Size相等(AES的Block Size为128 bits),而两个平台的API文档均指明当不传入初始向量时,系统将默认使用一个全0的初始向量。

    有了上述的基础之后,可以开始分别在两个平台进行实现了。

    具体实现

    iOS实现

    先定义一个初始向量的值。

    NSString *const kInitVector = @"16-Bytes--String";
    

    确定密钥长度,这里选择 AES-128。

    size_t const kKeySize = kCCKeySizeAES128;
    

    加密操作:

    + (NSString *)encryptAES:(NSString *)content key:(NSString *)key {
    
        NSData *contentData = [content dataUsingEncoding:NSUTF8StringEncoding];
        NSUInteger dataLength = contentData.length;
        
        // 为结束符'\0' +1
        char keyPtr[kKeySize + 1];
        memset(keyPtr, 0, sizeof(keyPtr));
        [key getCString:keyPtr maxLength:sizeof(keyPtr) encoding:NSUTF8StringEncoding];
        
        // 密文长度 <= 明文长度 + BlockSize
        size_t encryptSize = dataLength + kCCBlockSizeAES128;
        void *encryptedBytes = malloc(encryptSize);
        size_t actualOutSize = 0;
        
        NSData *initVector = [kInitVector dataUsingEncoding:NSUTF8StringEncoding];
        
        CCCryptorStatus cryptStatus = CCCrypt(kCCEncrypt,
                                              kCCAlgorithmAES,
                                              kCCOptionPKCS7Padding,  // 系统默认使用 CBC,然后指明使用 PKCS7Padding
                                              keyPtr,
                                              kKeySize,
                                              initVector.bytes,
                                              contentData.bytes,
                                              dataLength,
                                              encryptedBytes,
                                              encryptSize,
                                              &actualOutSize);
        
        if (cryptStatus == kCCSuccess) {
            // 对加密后的数据进行 base64 编码
            return [[NSData dataWithBytesNoCopy:encryptedBytes length:actualOutSize] base64EncodedStringWithOptions:NSDataBase64EncodingEndLineWithLineFeed];
        }
        free(encryptedBytes);
        return nil;
    }
    

    Java实现

    同理先在类中定义一个初始向量,需要与iOS端的统一。

    private static final String IV_STRING = "16-Bytes--String";
    

    另 Java 不需手动设置密钥大小,系统会自动根据传入的 Key 进行判断。

    加密操作:

    public static String encryptAES(String content, String key) 
                throws InvalidKeyException, NoSuchAlgorithmException, 
                NoSuchPaddingException, UnsupportedEncodingException, 
                InvalidAlgorithmParameterException, IllegalBlockSizeException, BadPaddingException {
    
        byte[] byteContent = content.getBytes("UTF-8");
    
        // 注意,为了能与 iOS 统一
        // 这里的 key 不可以使用 KeyGenerator、SecureRandom、SecretKey 生成
        byte[] enCodeFormat = key.getBytes();
        SecretKeySpec secretKeySpec = new SecretKeySpec(enCodeFormat, "AES");
            
        byte[] initParam = IV_STRING.getBytes();
        IvParameterSpec ivParameterSpec = new IvParameterSpec(initParam);
            
        // 指定加密的算法、工作模式和填充方式
        Cipher cipher = Cipher.getInstance("AES/CBC/PKCS5Padding");
        cipher.init(Cipher.ENCRYPT_MODE, secretKeySpec, ivParameterSpec);
        
        byte[] encryptedBytes = cipher.doFinal(byteContent);
        
        // 同样对加密后数据进行 base64 编码
        Encoder encoder = Base64.getEncoder();
        return encoder.encodeToString(encryptedBytes);
    }
    

    注意以上实现的是 AES-128,因此方法传入的 key 需为长度为 16 的字符串。

    关于解密

    有了上述加密的基础之后,解密的实现就很简单了,直接写出对应的逆操作即可。因此代码就不铺张了,如果有需要的可以直接到文末下载。

    关于Java使用大于128 bits的key

    到Oracle官网下载对应Java版本的 JCE ,解压后放到 JAVA_HOME/jre/lib/security/ ,然后修改 iOS 端的 kKeySize 和两端对应的 key 即可。

    以上。


    实现代码:

    AESCipher-iOS

    AESCipher-Java

    可直接使用,欢迎各种star和fork~

    相关文章

      网友评论

      • 不辣先生:java后台 AES ECB PKCS5Padding 我用了你的传过去后台解不了?大佬
      • 妙算小神仙:兄弟,专门登录上来给100🌟评论。
      • 命中注定IU:解不出来兄弟 我这边加密 后台解不出来,后台加密我解不出来 我们是AEX+HEX
      • 8293ecf1ceff:可以直接对文件加密 解密 通用吗?
      • Ross_:你好,如果是AES-256,传入的 key 的长度应该是多少个字符串????
      • 就_这样:你的java代码里是不是用了自己封装的包,能把自己封装的包也一并发出来吗,要不后台还是用不了
      • watermelon_lp:首先谢谢楼主。我是用 128cbc,无填充的方式加密。
        当key位16位,iv为16位。一切正常。
        但是我们的key用的是32位的,iv是16位的。我这样加密就会不一样。和后台无法交互。
        我是用这个网址进行测试的 ,我弄了好久了,不知道当key为32位是为什么会不一样 http://tool.chacuo.net/cryptaes
        希望楼主能给点建议
      • Yokihr:你好,你说java的key不可以用KeyGenerator等生成,那用什么方式呢?
      • e7250c2aec45:为啥在return之后才free buffer啊
      • Xanthuim:有传入Byte数组的吗?我在CSDN上找了一份改成传入Byte数组,解密出来只有8位,实际应该有16位。
        +(NSData *)AES256ParmEncryptWithKey:(Byte *)key Encrypttext:(NSData *)text //加密
        {
        NSUInteger dataLength = [text length];
        size_t bufferSize = dataLength + kCCBlockSizeAES128;
        void *buffer = malloc(bufferSize);
        size_t numBytesEncrypted = 0;
        CCCryptorStatus cryptStatus = CCCrypt(kCCEncrypt, kCCAlgorithmAES128,
        kCCOptionPKCS7Padding,
        key, kCCBlockSizeAES128,
        NULL,
        [text bytes], dataLength,
        buffer, bufferSize,
        &numBytesEncrypted);
        if (cryptStatus == kCCSuccess) {
        return [NSData dataWithBytesNoCopy:buffer length:numBytesEncrypted];
        }
        free(buffer);
        return nil;
        }

        + (NSData *)AES256ParmDecryptWithKey:(Byte *)key Decrypttext:(NSData *)text //解密
        {
        NSUInteger dataLength = [text length];
        size_t bufferSize = dataLength + kCCBlockSizeAES128;
        void *buffer = malloc(bufferSize);
        size_t numBytesDecrypted = 0;
        CCCryptorStatus cryptStatus = CCCrypt(kCCDecrypt, kCCAlgorithmAES128,
        kCCOptionPKCS7Padding,
        key, kCCBlockSizeAES128,
        NULL,
        [text bytes], dataLength,
        buffer, bufferSize,
        &numBytesDecrypted);
        if (cryptStatus == kCCSuccess) {
        return [NSData dataWithBytesNoCopy:buffer length:numBytesDecrypted];
        }
        free(buffer);
        return nil;
        }
        Xanthuim:由于key固定的是Byte数组,我没办法改变。字节转字符串也是不行的(有些字节无法正常转换)。
      • 一个不太努力的代码搬运工:特地感谢作者,找了很多框架,也找了很多文章,终于解决我的问题了,多谢多谢!
        WelkinXie:@losedMemory :blush:喜欢的欢迎在github上点个星哦~
      • iloverain:16-Bytes--String 是?
        iloverain:@WelkinXie 新手勿喷 😒
        iloverain:@WelkinXie 比如16个英文字符, 或者 8个中文字符吗?
        WelkinXie:@rainloveyou 自定义的16个字节长度的字符串…
      • brownfeng:学习下!!
      • leoRR:感谢作者
      • 7339fce83143:你好,我在使用iOS的时候编译不通过,是还需要进行什么配置么?
        ShenYj:没导头文件?
        WelkinXie:@next_day 不需要的,提示什么错误了:flushed:
      • Pusswzy:我们公司的加密是AES+Base64的加解密, 之前一直弄不出来,是因为Java后台使用的base64是websafe模式,iOS官方无法对此加解密
      • 爱上别的吧:我们项目里调加密 两边加密结果不一样 但是都能互相解密对:joy: 也是醉了
      • 千里马的驴:写得不错,我也在弄加密这块
        WelkinXie:@千里马的驴 :grin:

      本文标题:AES加密 - iOS与Java的同步实现

      本文链接:https://www.haomeiwen.com/subject/lzmesttx.html