欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页  >  IT编程

Golang的md5 hash计算操作

程序员文章站 2022-03-12 21:50:14
golang计算md5值的方法都是接收byte型slice([]byte)。而且使用习惯上也觉得略奇怪。看了好几个例子才看懂。感觉golang标准库在设计这些模块的时候,都会考虑使用带new关键字工厂...

golang计算md5值的方法都是接收byte型slice([]byte)。而且使用习惯上也觉得略奇怪。

看了好几个例子才看懂。

感觉golang标准库在设计这些模块的时候,都会考虑使用带new关键字工厂生成一个该类型的结构体对象。然后再使用改对象的方法进行操作。

md5包就是这样,来看例子:

s := "api_key" + apikey + "param" + param + "time" + time + "version" + version + apisecret
signbyte := []byte(s)
hash := md5.new()
hash.write(signbyte)
return hex.encodetostring(hash.sum(nil))

第一行我拼接了一个字符串。

第二行我将这个字符串转成byte型数组并赋值给了sighbyte。

第三行我new了一个md5的实现了hash.hash的结构体。

第四行我调用这个结构体的方法write将我需要计算md5的[]byte传入进去。

第五行我调用hex.encodetosring方法来把计算结果转换成16进制字符串。其中hash.sum(nil)方法可以生成前面write进去的signbyte变量的128bit md5值。

这个hash.sum方法并不是我们在操作其它语言的那种使用习惯,将值传递进去然后返回hash,这里其实是追加一个值一并生成hash。

由于我们不需要再追加值了,所以传入nil得到signbyte的hash值。

补充:golang标准库-crypto/md5(md5加密算法)

本文讲解如何使用go封装好的md5算法,不深入剖析md5算法原理。

首先我们要知道md5算法属于hash算法的一种,所以在了解md5之前,我们先认识一下go提供的hash接口。hash算法是保证只要输入的值不同,就一定会得到两个不同的指定长度的hash值。

当前两个不同值产生相同的hash还是有可能的,只是这个可能性很小很小

先认识go的hash接口(位于go标准库-hash包):

type hash interface {
  // 通过io.writer接口的write方法向hash中添加数据
  io.writer
  // 返回添加b到当前的hash值后的新切片,不会改变底层的hash状态,这个方法就是返回计算后的hash值,只是它是字符切片
  sum(b []byte) []byte
  // 重设hash为无数据输入的状态,就是清空hash之前写入的数据
  reset()
  // 返回sum会返回的切片的长度
  size() int
  // 返回hash底层的块大小;write方法可以接受任何大小的数据,
  // 但提供的数据是块大小的倍数时效率更高
  blocksize() int
}

crypto/md5包实现的就是这个hash接口。

hash包还有两个hash接口:

type hash32 interface { // hash32是一个被所有32位hash函数实现的公共接口。
  hash
  sum32() uint32
}
type hash64 interface { // hash64是一个被所有64位hash函数实现的公共接口。
  hash
  sum64() uint64
}

md5实现的第一个hash接口是16位的hash函数(它的sum方法返回的字符切片长度为16位),hash32和hash64是属于安全性更高的两个hash函数,产生的hash值也更长。

下面来看md5算法:

crypto/md5包提供了一个方法创建md5算法:

func new() hash.hash => 返回一个新的使用md5校验的hash.hash接口

接着看示例:

package main 
import (
 "crypto/md5"
 "encoding/hex"
 "fmt"
)
 
func main() { 
 has := md5.new() // 创建md5算法
 has.write([]byte("abc123")) // 写入需要加密的数据
 b := has.sum(nil) // 获取hash值字符切片;sum函数接受一个字符切片,这个切片的内容会原样的追加到abc123加密后的hash值的前面,这里我们不需要这么做,所以传入nil
 fmt.println(b) // 打印一下 [233 154 24 196 40 203 56 213 242 96 133 54 120 146 46 3]
 // 上面可以看到加密后的数据为长度为16位的字符切片,一般我们会把它转为16进制,方便存储和传播,下一步转换16进制
 fmt.println(hex.encodetostring(b)) // 通过hex包的encodetostring函数,将数据转为16进制字符串; 打印 e99a18c428cb38d5f260853678922e03
 
 // 还有一种方法转换为16进制,通过fmt的格式化打印方法, %x表示转换为16进制
 fmt.printf("%x",b) // 打印 e99a18c428cb38d5f260853678922e03
}

运行结果:

Golang的md5 hash计算操作

crypto/md5包还提供了一个md5加密简便的方法:

func sum(data []byte) [size]byte => 直接返回数据data的md5加密值,注意它返回的是指定大小(size)的数组,而不是切片了

下面看例子:

package main 
import (
 "crypto/md5"
 "fmt"
)
 
func main() { 
 b := md5.sum([]byte("abc123")) // 加密数据
 fmt.printf("%x",b) // 转换为16进制,并打印
}

运行结果:

Golang的md5 hash计算操作

这二种方法都可以md5加密数据,甚至后面更加简洁。

以上为个人经验,希望能给大家一个参考,也希望大家多多支持。如有错误或未考虑完全的地方,望不吝赐教。

相关标签: Golang md5 hash