Golang实现AES对称加密的过程详解

AES是一个对称密码,旨在取代DES成为广泛使用的标准,本文给大家分享Golang实现AES对称加密的过程,本文附有Golang实现AES加密ECB模式的源码,感兴趣的朋友跟随小编一起学习下吧

Posted in Golang onMay 20, 2021

AES加密

AES对称加密简介
AES是一个对称密码,旨在取代DES成为广泛使用的标准。是美国联邦政府采用的一种区块加密标准。

AES对称加密过程
加密解密算法的输入是一个128位分组。这些分组被描述成4×4的字节方阵,这个分组被复制到数组中,并在加密和解密的每一阶段都被修改。在字节方阵中,每一格都是一个字,包含了4字节。在矩阵中字是按列排序的。
加密由N轮构成,轮数依赖于密钥长度:16字节密钥对应10轮,24字节密钥对应12轮,32字节对应14轮。

AES加密模式

1.电码本模式(Electronic Codebook Book (ECB)

ECB模式是最早采用和最简单的模式,它将加密的数据分成若干组,每组的大小跟加密密钥长度相同,然后每组都用相同的密钥进行加密。

2.密码分组链接模式(Cipher Block Chaining (CBC))

这种模式是先将明文切分成若干小段,然后每一小段与初始块或者上一段的密文段进行异或运算后,再与密钥进行加密。

3.密码反馈模式(Cipher FeedBack (CFB))
隐藏了明文模式,分组密码转化为流模式,可以及时加密传送小于分组的数据

4.OFB(Output FeedBack,输出反馈)模式
隐藏了明文模式;,分组密码转化为流模式,可以及时加密传送小于分组的数据

AES填充方式
AES支持支持几种填充:NoPadding,PKCS5Padding,ISO10126Padding,PaddingMode.Zeros,PaddingMode.PKCS7。对于AES来说PKCS5Padding和PKCS7Padding是完全一样的,不同在于PKCS5限定了块大小为8bytes而PKCS7没有限定。因此对于AES来说两者完全相同

Golang实现AES加密解密

下面附上Golang实现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("源数据长度不能为0")
    }
    block, err := aes.NewCipher(key)
    if err != nil {
        return nil, err
    }
    if len(crypted)%block.BlockSize() != 0 {
        return nil, fmt.Errorf("源数据长度必须是 %d 的整数倍,当前长度为:%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("源数据长度不能为0")
    }
    src = PKCS5Padding(src, block.BlockSize())
    if len(src)%block.BlockSize() != 0 {
        return nil, fmt.Errorf("源数据长度必须是 %d 的整数倍,当前长度为:%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实现AES对称加密的过程详解的详细内容

Golang 相关文章推荐
Golang二维切片初始化的实现
Apr 08 Golang
Go语言-为什么返回值为接口类型,却返回结构体
Apr 24 Golang
Go语言使用select{}阻塞main函数介绍
Apr 25 Golang
Golang 正则匹配效率详解
Apr 25 Golang
Golang之sync.Pool使用详解
May 06 Golang
Golang Gob编码(gob包的使用详解)
May 07 Golang
Go语言基础知识点介绍
Jul 04 Golang
go开发alertmanger实现钉钉报警
Jul 16 Golang
Go语言安装并操作redis的go-redis库
Apr 14 Golang
Golang map映射的用法
Apr 22 Golang
Go gRPC进阶教程gRPC转换HTTP
Jun 16 Golang
go语言基础 seek光标位置os包的使用
May 09 #Golang
Golang 实现获取当前函数名称和文件行号等操作
May 08 #Golang
Golang 获取文件md5校验的方法以及效率对比
May 08 #Golang
GoLang中生成UUID唯一标识的实现
May 08 #Golang
聊聊golang中多个defer的执行顺序
May 08 #Golang
Golang全局变量加锁的问题解决
golang 实现并发求和
May 08 #Golang
You might like
动态新闻发布的实现及其技巧
2006/10/09 PHP
php auth_http类库进行身份效验
2009/03/19 PHP
php不用内置函数对数组排序的两个算法代码
2010/02/08 PHP
PHP之COOKIE支持详解
2010/09/20 PHP
PHP 面向对象程序设计(oop)学习笔记(三) - 单例模式和工厂模式
2014/06/12 PHP
php中final关键字用法分析
2016/12/07 PHP
PHP框架laravel的.env文件配置教程
2017/06/07 PHP
JavaScript 密码强度判断代码
2009/09/05 Javascript
JavaScript中获取元素索引的函数
2010/09/10 Javascript
javascript设计模式 接口介绍
2012/07/24 Javascript
jQuery lazyLoad图片延迟加载插件的优化改造方法分享
2013/08/13 Javascript
判断iframe里的页面是否加载完成
2014/06/06 Javascript
js 实现的可折叠留言板(附源码下载)
2014/07/01 Javascript
Javascript冒泡排序算法详解
2014/12/03 Javascript
浅谈Javascript变量作用域问题
2014/12/16 Javascript
jQuery给动态添加的元素绑定事件的方法
2015/03/09 Javascript
js实现点击链接后延迟3秒再跳转的方法
2015/06/05 Javascript
详解JavaScript中的forEach()方法的使用
2015/06/08 Javascript
jQuery模仿单选按钮选中效果
2016/06/24 Javascript
详解angular中通过$location获取路径(参数)的写法
2017/03/21 Javascript
webpack学习笔记之优化缓存、合并、懒加载
2017/08/24 Javascript
深入理解Vue官方文档梳理之全局API
2017/11/22 Javascript
JavaScript实现图片懒加载的方法分析
2018/07/05 Javascript
Vue使用.sync 实现父子组件的双向绑定数据问题
2019/04/04 Javascript
python 控制语句
2011/11/03 Python
pandas DataFrame 删除重复的行的实现方法
2019/01/29 Python
python实现从本地摄像头和网络摄像头截取图片功能
2019/07/11 Python
CSS实现半透明边框与多重边框的场景分析
2019/11/13 HTML / CSS
HTML5所有标签汇总及标签意义解释
2015/03/12 HTML / CSS
Java程序员常见面试题
2015/07/16 面试题
应届生高等护理求职信
2013/10/12 职场文书
生产文员岗位职责
2014/04/05 职场文书
盗窃罪辩护词范文
2015/05/21 职场文书
2015年社区党建工作汇报材料
2015/06/25 职场文书
卫生主题班会
2015/08/14 职场文书
教你一步步实现一个简易promise
2021/11/02 Javascript