Golang 获取文件 md5 校验和的方法及效率比较

本文探讨了在Golang中获取文件MD5校验和的两种方法,并通过测试对比了它们的效率。针对大量且大小不一的文件,文章提到了效率问题的重要性。在测试中,一种方法被证明在处理大文件时具有更高的效率,推荐在实际应用中采用。
摘要由CSDN通过智能技术生成

近期有一个需求:获取多个文件 md5 校验和判断是否存在重复文件,因为文件数量较多,有的文件还比较大,需要处理的文件还没有到位,我就考虑了一下效率的问题。目前我已知的 Golang 中获取 md5 校验和的方法有两个,这里直接给出实现源码。

package main

import (
	"crypto/md5"
	"flag"
	"fmt"
	"io"
	"io/ioutil"
	"os"
)

var which = flag.Bool("which", true, "")
var path = flag.String("path", "", "")
var cnt = flag.Int("cnt", 100, "")

func aaa() {
	f, err := os.Open(*path)
	if err != nil {
		fmt.Println("Open", err)
		return
	}

	defer f.Close()

	body, err := ioutil.ReadAll(f)
	if err != nil {
		fmt.Println("ReadAll", err)
		return
	}

	md5.Sum(body)
	//fmt.Printf("%x\n", md5.Sum(body))
}

func bbb() {
	f, err := os.Open(*path)
	if err != nil {
		fmt.Println("Open", err)
		return
	}

	defer f.Close()

	md5hash := md5.New()
	if _, err := io.Copy(md5hash, f); err != nil {
		fmt.Println("Copy", err)
		return
	}

	md5hash.Sum(nil)
	//fmt.Printf("%x\n", md5hash.Sum(nil))
}

func main() {
	flag.Parse()

	for i := 0; i < *cnt; i++ {
		if *which {
			aaa()
		} else {
			bbb()
		}
	}
}

还有可供参考的获取 md5 校验和的 Shell 命令

md5 -- calculate a message-digest fingerprint (checksum) for a file
md5 [-pqrtx] [-s string] [file ...]

测试文件是公司项目的日志文件

banjakukutekiiMac:shell panshiqu$ ls -an | grep by
-rw-r--r--   1 501  20   7285957 11 17 16:14 by.out

banjakukutekiiMac:shell panshiqu$ cp by.out by2.out
banjakukutekiiMac:shell panshiqu$ cat by.out >> by2.out

banjakukutekiiMac:shell panshiqu$ ls -an | grep by
-rw-r--r--   1 501  20   7285957 11 17 16:14 by.out
-rw-r--r--   1 501  20  14571914 11 17 17:03 by2.out

下面效率展示

banjakukutekiiMac:shell panshiqu$ time ./gomd5 -cnt=1 -which=true -path="by.out"

real	0m0.027s
user	0m0.017s
sys	0m0.012s
banjakukutekiiMac:shell panshiqu$ time ./gomd5 -cnt=1 -which=true -path="by2.out"

real	0m0.048s
user	0m0.033s
sys	0m0.018s
banjakukutekiiMac:shell panshiqu$ time ./gomd5 -cnt=1 -which=false -path="by.out"

real	0m0.018s
user	0m0.012s
sys	0m0.004s
banjakukutekiiMac:shell panshiqu$ time ./gomd5 -cnt=1 -which=false -path="by2.out"

real	0m0.031s
user	0m0.024s
sys	0m0.005s
banjakukutekiiMac:shell panshiqu$ time md5 by.out
MD5 (by.out) = 9d79e19a00cef1ae1bb6518ca4adf9de

real	0m0.023s
user	0m0.019s
sys	0m0.006s
banjakukutekiiMac:shell panshiqu$ time md5 by2.out
MD5 (by2.out) = 0a029a460a20e8dcb00d032d6fab74c6

real	0m0.042s
user	0m0.037s
sys	0m0.009s

总结:

  • 不管什么方法都会随着文件变大时间会变长,上面的例子大约都是2倍
  • io.Copy 方法效率最高,建议大家这样使用
你可以使用Go语言的net包来连接其他机器并获取文件MD5值。以下是一个示例代码: ```go package main import ( "crypto/md5" "fmt" "io" "net" "os" ) func main() { // 远程机器的IP地址和端口号 host := "192.168.1.100" port := "8888" // 要获取MD5值的文件路径 path := "/path/to/file.txt" // 连接远程机器 conn, err := net.Dial("tcp", host+":"+port) if err != nil { fmt.Println("Error while connecting to remote machine:", err) return } defer conn.Close() // 发送文件路径给远程机器 if _, err := conn.Write([]byte(path)); err != nil { fmt.Println("Error while sending file path:", err) return } // 接收远程机器返回的MD5md5sum := make([]byte, md5.Size) if _, err := io.ReadFull(conn, md5sum); err != nil { fmt.Println("Error while receiving MD5 sum:", err) return } // 输出MD5值 fmt.Printf("MD5 of file %s on %s: %x\n", path, host, md5sum) } ``` 你需要将`host`和`port`变量替换成远程机器的IP地址和端口号,将`path`变量替换成要获取MD5值的文件的路径。然后你需要在远程机器上编写一个简单的TCP服务器程序,用于接收文件路径并计算MD5值。例如: ```go package main import ( "crypto/md5" "fmt" "io" "net" "os" ) func main() { // 监听的端口号 port := "8888" // 监听TCP端口 listener, err := net.Listen("tcp", ":"+port) if err != nil { fmt.Println("Error while listening:", err) return } defer listener.Close() for { // 等待连接 conn, err := listener.Accept() if err != nil { fmt.Println("Error while accepting connection:", err) continue } // 接收文件路径 path := make([]byte, 1024) n, err := conn.Read(path) if err != nil { fmt.Println("Error while receiving file path:", err) continue } path = path[:n] // 计算MD5值 f, err := os.Open(string(path)) if err != nil { fmt.Println("Error while opening file:", err) continue } defer f.Close() h := md5.New() if _, err := io.Copy(h, f); err != nil { fmt.Println("Error while calculating MD5:", err) continue } md5sum := h.Sum(nil) // 发送MD5值 if _, err := conn.Write(md5sum); err != nil { fmt.Println("Error while sending MD5 sum:", err) continue } conn.Close() } } ``` 你需要将`port`变量替换成你要监听的端口号。然后你可以运行这个程序并等待其他机器连接,计算MD5值并返回。
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值