Golang實現(xiàn)AES對稱加密的過程詳解
AES加密
AES對稱加密簡介
AES是一個對稱密碼,旨在取代DES成為廣泛使用的標準。是美國聯(lián)邦政府采用的一種區(qū)塊加密標準。
AES對稱加密過程
加密解密算法的輸入是一個128位分組。這些分組被描述成4×4的字節(jié)方陣,這個分組被復制到數(shù)組中,并在加密和解密的每一階段都被修改。在字節(jié)方陣中,每一格都是一個字,包含了4字節(jié)。在矩陣中字是按列排序的。
加密由N輪構(gòu)成,輪數(shù)依賴于密鑰長度:16字節(jié)密鑰對應10輪,24字節(jié)密鑰對應12輪,32字節(jié)對應14輪。
AES加密模式
1.電碼本模式(Electronic Codebook Book (ECB)
ECB模式是最早采用和最簡單的模式,它將加密的數(shù)據(jù)分成若干組,每組的大小跟加密密鑰長度相同,然后每組都用相同的密鑰進行加密。
2.密碼分組鏈接模式(Cipher Block Chaining (CBC))
這種模式是先將明文切分成若干小段,然后每一小段與初始塊或者上一段的密文段進行異或運算后,再與密鑰進行加密。
3.密碼反饋模式(Cipher FeedBack (CFB))
隱藏了明文模式,分組密碼轉(zhuǎn)化為流模式,可以及時加密傳送小于分組的數(shù)據(jù)
4.OFB(Output FeedBack,輸出反饋)模式
隱藏了明文模式;,分組密碼轉(zhuǎn)化為流模式,可以及時加密傳送小于分組的數(shù)據(jù)
AES填充方式
AES支持支持幾種填充:NoPadding,PKCS5Padding,ISO10126Padding,PaddingMode.Zeros,PaddingMode.PKCS7。對于AES來說PKCS5Padding和PKCS7Padding是完全一樣的,不同在于PKCS5限定了塊大小為8bytes而PKCS7沒有限定。因此對于AES來說兩者完全相同
Golang實現(xiàn)AES加密解密
下面附上Golang實現(xiàn)AES加密ECB模式的源碼:
package main import ( "bytes" "crypto/aes" "fmt" "testing" ) //ECB模式解密 func ECBDecrypt(crypted, key []byte) ([]byte, error) { if !validKey(key) { return nil, fmt.Errorf("秘鑰長度錯誤,當前傳入長度為 %d",len(key)) } if len(crypted) < 1 { return nil, fmt.Errorf("源數(shù)據(jù)長度不能為0") } block, err := aes.NewCipher(key) if err != nil { return nil, err } if len(crypted)%block.BlockSize() != 0 { return nil, fmt.Errorf("源數(shù)據(jù)長度必須是 %d 的整數(shù)倍,當前長度為:%d",block.BlockSize(), len(crypted)) } var dst []byte tmpData := make([]byte, block.BlockSize()) for index := 0; index < len(crypted); index += block.BlockSize() { block.Decrypt(tmpData, crypted[index:index+block.BlockSize()]) dst = append(dst, tmpData...) } dst, err = PKCS5UnPadding(dst) if err != nil { return nil, err } return dst, nil } //ECB模式加密 func ECBEncrypt(src, key []byte) ([]byte, error) { if !validKey(key) { return nil, fmt.Errorf("秘鑰長度錯誤, 當前傳入長度為 %d",len(key)) } block, err := aes.NewCipher(key) if err != nil { return nil, err } if len(src) < 1 { return nil, fmt.Errorf("源數(shù)據(jù)長度不能為0") } src = PKCS5Padding(src, block.BlockSize()) if len(src)%block.BlockSize() != 0 { return nil, fmt.Errorf("源數(shù)據(jù)長度必須是 %d 的整數(shù)倍,當前長度為:%d",block.BlockSize(), len(src)) } var dst []byte tmpData := make([]byte, block.BlockSize()) for index := 0; index < len(src); index += block.BlockSize() { block.Encrypt(tmpData, src[index:index+block.BlockSize()]) dst = append(dst, tmpData...) } return dst, nil } // PKCS5填充 func PKCS5Padding(ciphertext []byte, blockSize int) []byte { padding := blockSize - len(ciphertext)%blockSize padtext := bytes.Repeat([]byte{byte(padding)}, padding) return append(ciphertext, padtext...) } // 去除PKCS5填充 func PKCS5UnPadding(origData []byte) ([]byte, error) { length := len(origData) unpadding := int(origData[length-1]) if length < unpadding { return nil, fmt.Errorf("invalid unpadding length") } return origData[:(length - unpadding)], nil } // 秘鑰長度驗證 func validKey(key []byte) bool { k := len(key) switch k { default: return false case 16, 24, 32: return true } } func TestAes(t *testing.T){ srcData := "hello world !" key := []byte("abcdabcdabcdabcdabcdabcdabcdabcd") //測試加密 encData ,err := ECBEncrypt([]byte(srcData),(key)) if err != nil { t.Errorf(err.Error()) return } //測試解密 decData ,err := ECBDecrypt(encData,key) if err != nil { t.Errorf(err.Error()) return } t.Log(string(decData)) }
以上就是Golang實現(xiàn)AES對稱加密的過程詳解的詳細內(nèi)容,更多關于go AES對稱加密的資料請關注腳本之家其它相關文章!
相關文章
利用GoLang?Fiber進行高性能Web開發(fā)實例詳解
這篇文章主要為大家介紹了利用GoLang?Fiber進行高性能Web開發(fā)實例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪2024-01-01GO中的時間操作總結(jié)(time&dateparse)
日常開發(fā)過程中,對于時間的操作可謂是無處不在,但是想實現(xiàn)時間自由還是不簡單的,多種時間格式容易混淆,本文為大家整理了一下GO中的時間操作,有需要的可以參考下2023-09-09go數(shù)據(jù)結(jié)構(gòu)和算法BitMap原理及實現(xiàn)示例
這篇文章主要為大家介紹了go數(shù)據(jù)結(jié)構(gòu)和算法BitMap原理及實現(xiàn)示例,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪2022-07-07golang實現(xiàn)文件上傳并轉(zhuǎn)存數(shù)據(jù)庫功能
這篇文章主要為大家詳細介紹了golang實現(xiàn)文件上傳并轉(zhuǎn)存數(shù)據(jù)庫功能,文中示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下2022-07-07golang jsoniter extension 處理動態(tài)字段的實現(xiàn)方法
這篇文章主要介紹了golang jsoniter extension 處理動態(tài)字段的實現(xiàn)方法,我們使用實例級別的 extension, 而非全局,可以針對不同業(yè)務邏輯有所區(qū)分,jsoniter 包提供了比較完善的定制能力,通過例子可以感受一下擴展性,需要的朋友可以參考下2023-04-04