AES加密 – iOS與Java的同步實現

Welkin發表於2016-08-23

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實現

先定義一個初始向量的值。

確定金鑰長度,這裡選擇 AES-128。

加密方法

解密方法

Java實現

同理先在類中定義一個初始向量,需要與iOS端的統一。

另 Java 不需手動設定金鑰大小,系統會自動根據傳入的 Key 進行判斷。

加密方法

解密方法

至此,AES 在 iOS 與 Java 同步實現完成。

注意以上實現的是 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~

相關文章