深入理解MD5:Message Digest Algorithm 5


title: 深入理解MD5:Message Digest Algorithm 5
date: 2024/4/21 18:10:18
updated: 2024/4/21 18:10:18
tags:

  • MD5
  • 哈希函数
  • 密码学
  • 数据完整性
  • 碰撞攻击
  • 安全性
  • 替代算法

在这里插入图片描述

导论

MD5的背景和历史

MD5(Message Digest Algorithm 5)是一种广泛使用的哈希函数,用于产生128位(16字节)的哈希值,通常以32个十六进制数字表示。它由Ronald Rivest于1991年设计,并在RFC 1321中进行了描述。

MD5的设计目的是为了提供数据完整性验证和消息认证。它被广泛应用于数字签名、消息认证码(MAC)、密码学散列函数等领域。在早期,MD5曾被广泛用于密码存储、数字证书、软件完整性验证等方面。

MD5的应用领域
  1. 密码学:MD5常用于密码存储,通过将用户密码的MD5哈希值存储在数据库中,而不是明文密码,以增加安全性。然而,由于MD5存在安全性问题,现已被更安全的哈希函数如SHA-256所取代。
  2. 数据完整性验证:MD5可用于验证文件的完整性,通过比较文件的MD5哈希值,确保文件在传输或存储过程中未被篡改。
  3. 数字签名:在一些早期的应用中,MD5曾用于生成数字签名,用于验证文件的来源和完整性。然而,由于MD5的碰撞攻击,这种用法已经不再安全。
  4. 消息认证码(MAC) :MD5可以用于生成消息认证码,用于验证消息的完整性和来源。
MD5的特点和安全性问题
  1. 快速计算:MD5的计算速度相对较快,适用于对大量数据进行哈希计算。

  2. 固定长度输出:MD5生成的哈希值长度固定为128位,无论输入数据的长度如何,始终生成相同长度的哈希值。

  3. 安全性问题:然而,MD5在安全性上存在严重问题,主要表现在以下几个方面:

    • 碰撞攻击:MD5已经被证明存在碰撞攻击,即找到两个不同的输入,但生成相同的MD5哈希值。这种情况下,攻击者可以伪造数字签名或通过篡改数据而不改变哈希值来绕过完整性检查。
    • 预图攻击:MD5也容易受到预图攻击,即通过已知的输入和输出对来推导出其他输入的哈希值。
    • 彩虹表攻击:MD5哈希值的空间相对较小,容易受到彩虹表攻击,即通过事先计算好的哈希值与明文的对应关系表,从而快速破解密码。

由于MD5的安全性问题,现在已经不再建议将其用于安全敏感的应用中,而应该选择更安全的哈希函数如SHA-256等。

MD5算法原理

MD5算法的概述

MD5(Message Digest Algorithm 5)是一种哈希函数,用于生成128位(16字节)的哈希值。其设计目的是为了提供数据完整性验证和消息认证。MD5算法将任意长度的输入数据转换为固定长度的128位哈希值,且输出的哈希值在理想情况下是唯一的。

MD5算法的流程和步骤

MD5算法的流程主要包括以下步骤:

  1. 填充数据:将输入数据进行填充,使其长度满足512位(64字节)的倍数,通常采用的填充方式是在数据末尾添加一个1和若干个0,以及数据长度的二进制表示。
  2. 初始化变量:初始化四个32位的寄存器A、B、C、D,这些寄存器用于存储中间计算结果。
  3. 分块处理:将填充后的数据按照512位(64字节)一块进行处理,每块包含16个32位字。
  4. 循环压缩函数处理:对每个512位的数据块进行四轮循环压缩函数处理,每轮处理包括四个步骤:F函数G函数H函数I函数
  5. 更新寄存器:根据每轮的计算结果更新寄存器A、B、C、D的值。
  6. 生成哈希值:经过所有数据块的处理后,将四个寄存器的值连接起来,按照A、B、C、D的顺序将每个寄存器的值转换为16进制表示,即得到128位的MD5哈希值。
MD5算法中使用的运算函数

MD5算法中使用了四个非线性函数,分别为F、G、H、I函数,这些函数在循环压缩函数处理中起着重要作用:

  1. F函数: F函数采用的是逻辑与(AND)、逻辑或(OR)、逻辑非(NOT)和异或(XOR)操作,用于混淆数据块中的数据。
  2. G函数: G函数也是采用与F函数类似的操作,用于进一步混淆数据块中的数据。
  3. H函数: H函数同样采用逻辑与、逻辑或和异或操作,用于增加数据块的复杂性。
  4. I函数: I函数也是一种非线性函数,用于增加数据块的随机性,增强MD5算法的安全性。

这些运算函数的设计使得MD5算法具有较好的扩散性和抗碰撞能力,但由于现代计算能力的提升和MD5算法本身的设计缺陷,使得MD5算法已经不再安全。

MD5算法细节

消息填充和处理

在MD5算法中,消息填充的目的是将输入数据填充至512位(64字节)的倍数,并在数据末尾添加一个长度信息。填充的具体步骤如下:

  1. 首先将数据长度表示为二进制形式,并附加到数据的末尾。
  2. 在数据末尾添加一个’1’比特,然后填充0直到数据长度满足对512取模的结果为448(即数据长度模512等于448)。
  3. 将数据长度(64位)附加到填充后的数据末尾,以二进制表示。
循环函数和常量

MD5算法中使用了四个循环函数(F、G、H、I)和64个常量。这些函数和常量是在循环压缩函数处理过程中使用的。循环压缩函数中的每一轮都会使用不同的常量和循环函数。

  1. F函数:F函数采用了非线性函数𝐹(𝑋,𝑌,𝑍)=(𝑋∧𝑌)∨(¬𝑋∧𝑍)。
  2. G函数:G函数采用了非线性函数𝐺(𝑋,𝑌,𝑍)=(𝑋∧𝑍)∨(𝑌∧¬𝑍)。
  3. H函数:H函数采用了非线性函数𝐻(𝑋,𝑌,𝑍)=𝑋⊕𝑌⊕𝑍。
  4. I函数:I函数采用了非线性函数𝐼(𝑋,𝑌,𝑍)=𝑌⊕(𝑋∨¬𝑍)。

在MD5算法中,每轮循环使用的常量是通过对2^32的整数部分取绝对值后的正弦函数计算得到的。这些常量用于增加算法的非线性性质和扩散性。

消息分组和处理流程

MD5算法将输入数据按512位(64字节)分组,并对每个512位数据块进行处理。处理流程如下:

  1. 初始化寄存器:将四个32位寄存器(A、B、C、D)初始化为特定的常量值。
  2. 分组处理:将填充后的数据分成若干个512位的数据块,对每个数据块进行循环压缩函数处理。
  3. 循环压缩函数处理:对每个数据块进行四轮循环处理,每轮处理包括16次操作。在每一轮中,使用一个常量和一个消息块的子分组作为输入,并对寄存器进行更新。
  4. 更新寄存器:每轮处理后,根据计算结果更新寄存器的值。
  5. 生成哈希值:经过所有数据块的处理后,将四个寄存器的值连接起来,并将每个寄存器的值转换为16进制表示,即得到128位的MD5哈希值。

这样,MD5算法通过一系列的数据处理步骤,将任意长度的输入数据转换为128位的哈希值,以实现数据的完整性验证和消息认证。

MD5碰撞攻击

原理

MD5碰撞攻击的原理基于MD5算法的弱点,主要包括以下两点:

  1. 碰撞攻击的原理:MD5算法的输出空间远远小于输入空间,这导致了可能性的碰撞。MD5算法的输出长度是128位,因此存在着不同的输入会产生相同的输出(哈希碰撞)。攻击者利用这一特性,通过精心构造的两个不同的输入数据,使得它们的MD5哈希值相同,即产生了碰撞。
  2. 哈希碰撞的搜索:MD5算法的设计存在缺陷,使得寻找碰撞不需要遍历所有可能的输入组合。攻击者可以利用已知的碰撞片段,通过巧妙的方式生成具有相同哈希值的两个不同输入。这通常涉及对MD5算法的内部结构和碰撞寻找技术的深入理解。
实例分析

经典的MD5碰撞攻击实例是2004年Wang等人发表的《Finding Collisions in the Full SHA-1》。在这项研究中,他们展示了如何在SHA-1算法上实现碰撞攻击,并利用类似的技术攻击了MD5。他们使用了巧妙的算法和大量计算资源,成功地生成了两个具有相同MD5哈希值的不同输入。

针对MD5碰撞攻击的防范措施

尽管MD5算法已经被广泛认为不安全,并且不建议在安全敏感的场景中使用,但在实际应用中,有时仍然需要对其进行防范。以下是一些针对MD5碰撞攻击的防范措施:

  1. 停止使用MD5:首先,避免在安全相关的应用中使用MD5算法。应当选择更安全的哈希算法,如SHA-256或SHA-3。
  2. 使用盐值:对于需要使用MD5的场景,应当使用盐值(salt)来增加哈希的随机性。盐值是一个随机的字符串,与原始数据一起进行哈希运算,可以有效防止彩虹表攻击。
  3. 使用消息认证码(MAC) :对于需要数据完整性保护的场景,应当使用消息认证码(MAC)来代替简单的哈希算法。MAC结合了哈希函数和密钥,提供了更高级别的安全保护。
  4. 更新算法:如果无法立即停止使用MD5,至少应当对现有系统进行更新,以使用更安全的哈希算法。

总的来说,MD5碰撞攻击已经被广泛认为是不安全的,应当尽可能避免在安全敏感的场景中使用MD5算法,并采取必要的防范措施以保护数据安全。

MD5在实际应用中的使用

数据完整性校验

MD5常被用于数据完整性校验,特别是在文件传输过程中。发送方计算文件的MD5哈希值,并将其与接收方计算的哈希值进行比较,以确保文件在传输过程中未被篡改。虽然MD5在这种场景下被广泛使用,但由于其碰撞攻击的漏洞,已经不再是最佳选择。

import hashlibdef calculate_md5(file_path):with open(file_path, 'rb') as f:content = f.read()md5_hash = hashlib.md5(content).hexdigest()return md5_hash# 计算文件的MD5哈希值
file_path = 'example_file.txt'
md5_hash = calculate_md5(file_path)
print("MD5哈希值:", md5_hash)
密码存储和验证

在过去,MD5常被用于密码存储,通常是在数据库中存储用户密码的哈希值。然而,由于MD5容易受到碰撞攻击,现在不再推荐将其用于密码存储。推荐使用更安全的哈希算法,如bcrypt或Argon2。

import hashlibdef hash_password(password):# 添加盐值salted_password = password + 'mysalt'hashed_password = hashlib.md5(salted_password.encode()).hexdigest()return hashed_passworddef verify_password(input_password, stored_password):return hash_password(input_password) == stored_password# 存储密码
password = 'mypassword'
hashed_password = hash_password(password)# 验证密码
input_password = 'mypassword'
if verify_password(input_password, hashed_password):print("密码正确")
else:print("密码错误")
数字签名和认证

MD5也曾经被用于数字签名和认证,用于验证数据的完整性和真实性。然而,由于其安全性问题,现在不再建议在这种敏感场景中使用MD5算法。

在实践中,应使用更安全的哈希算法,如SHA-256或SHA-3,以确保数字签名和认证的安全性。

import hashlibdef create_signature(data):hashed_data = hashlib.sha256(data.encode()).hexdigest()return hashed_data# 创建数字签名
data = 'important_data'
signature = create_signature(data)
print("数字签名:", signature)

总的来说,尽管MD5在过去被广泛使用,但现在由于其安全性问题,应尽可能避免在敏感场景中使用,并选择更安全的替代方案。

MD5的替代算法

SHA系列算法

SHA(安全哈希算法)系列是MD5的替代方案,提供了更高的安全性。SHA-1、SHA-256、SHA-384和SHA-512是其中最常见的几种。

  1. SHA-1 (Secure Hash Algorithm 1) : SHA-1是MD5的后继者,拥有160位的哈希值,通常用于数字签名和证书中。
  2. SHA-256 (Secure Hash Algorithm 256) : SHA-256产生256位的哈希值,比SHA-1更安全,常用于数据完整性验证、密码存储和数字签名。
  3. SHA-384 (Secure Hash Algorithm 384) : SHA-384生成384位的哈希值,适用于安全性要求较高的场景,如金融和政府领域。
  4. SHA-512 (Secure Hash Algorithm 512) : SHA-512生成512位的哈希值,提供更高的安全性,适用于对安全性要求极高的应用。
import hashlibdef sha256_hash(data):hashed_data = hashlib.sha256(data.encode()).hexdigest()return hashed_data# 使用SHA-256哈希算法
data = 'important_data'
sha256_hashed_data = sha256_hash(data)
print("SHA-256哈希值:", sha256_hashed_data)
新兴的哈希算法

除了SHA系列算法外,还有一些新兴的哈希算法在不断发展和应用中,其中一些包括:

  1. BLAKE2: BLAKE2是一种高速、安全的哈希函数,具有与MD5和SHA-3相似的速度,但提供了更高的安全性。它适用于数据完整性校验、密码存储等场景。
from hashlib import blake2bdef blake2_hash(data):h = blake2b(data.encode(), digest_size=32)return h.hexdigest()# 使用BLAKE2哈希算法
data = 'important_data'
blake2_hashed_data = blake2_hash(data)
print("BLAKE2哈希值:", blake2_hashed_data)
安全性更高的替代方案

除了哈希算法外,对于特别安全性要求高的场景,可以考虑使用加密哈希函数或密码哈希函数,如bcrypt和Argon2。这些算法不仅提供了哈希功能,还包含了额外的安全机制,如随机盐和迭代次数,以抵御暴力破解和彩虹表攻击。

import bcryptdef bcrypt_hash_password(password):hashed_password = bcrypt.hashpw(password.encode(), bcrypt.gensalt())return hashed_passworddef bcrypt_verify_password(input_password, hashed_password):return bcrypt.checkpw(input_password.encode(), hashed_password)# 使用bcrypt哈希算法存储和验证密码
password = 'mypassword'
hashed_password = bcrypt_hash_password(password)input_password = 'mypassword'
if bcrypt_verify_password(input_password, hashed_password):print("密码正确")
else:print("密码错误")

总的来说,选择哈希算法应根据具体应用场景和安全需求,对于不同的需求,可以选择适合的替代方案以提高安全性。

MD5的未来发展

MD5在密码学中的地位

MD5(Message Digest Algorithm 5)是一种广泛使用的哈希算法,但它在密码学中的地位已经逐渐下降。原因在于MD5存在严重的安全漏洞,易受到碰撞攻击和预图攻击的影响。因此,在安全性要求较高的领域,如密码存储和数字签名,不推荐使用MD5。

MD5的安全性评估

MD5的安全性已经受到广泛的评估,并被证明是不安全的。主要的安全问题包括:

  1. 碰撞攻击:MD5已经被证明容易受到碰撞攻击,即找到两个不同的输入产生相同的哈希值,这会对数据完整性和身份验证造成严重威胁。
  2. 预图攻击:MD5也容易受到预图攻击,攻击者可以根据已知的哈希值推导出对应的输入,这使得MD5不适合用于密码存储等安全敏感场景。
MD5的未来走向和发展趋势

考虑到MD5存在的严重安全问题,它的未来走向主要集中在以下几个方面:

  1. 淘汰和替代:由于MD5的安全性已经受到广泛认可,大多数安全专家和组织已经不再推荐使用MD5。取而代之的是更安全的哈希算法,如SHA-256、SHA-3等。
  2. 后续研究:尽管MD5不再被视为安全的哈希算法,但仍然有一些研究人员致力于分析MD5的碰撞攻击和预图攻击,以深入理解其安全性漏洞,并从中得出更深层次的密码学洞见。
  3. 历史价值:尽管MD5不再适用于安全敏感的应用,但它仍然具有一定的历史价值,可以用于非安全性要求且对速度要求较高的场景,如数据完整性校验和非关键性数据的哈希。

总的来说,MD5已经逐渐被淘汰,未来的发展趋势主要集中在其替代方案的研究和应用上,以提高密码学的安全性和可靠性。

附录

MD5在线加密 | 一个覆盖广泛主题工具的高效在线平台(amd794.com)
https://amd794.com/md5

MD5算法的伪代码实现

以下是MD5算法的简化伪代码实现:

初始化变量:
A = 0x67452301
B = 0xEFCDAB89
C = 0x98BADCFE
D = 0x10325476对消息进行填充和长度扩展:
padding 消息 M 使其长度 (bits) 为 448 mod 512
在消息末尾添加一个 1,然后添加足够的 0,直到消息长度为 64 bits 不足 512 bits 的倍数
在消息末尾添加 64 bits 表示消息长度(原始长度)对填充后的消息进行处理:
将消息分为512位的块,并对每个块执行以下操作:将当前块分为16个32位字初始化四个32位寄存器:A,B,C,D主循环:对每个消息块进行64轮迭代处理:根据当前轮数选择不同的F函数、索引值和位移数更新寄存器A,B,C,D的值更新最终哈希值:将最终的A,B,C,D寄存器值连接成一个128位哈希值返回哈希值
MD5算法的Python实现示例

下面是一个简单的Python示例,展示了如何使用Python实现MD5算法:

import hashlibdef calculate_md5(message):hash_object = hashlib.md5(message.encode())md5_hash = hash_object.hexdigest()return md5_hash# 示例用法
message = "Hello, MD5!"
md5_hash = calculate_md5(message)
print("MD5 哈希值:", md5_hash)
MD5算法的安全性分析

MD5算法存在严重的安全漏洞,主要包括碰撞攻击和预图攻击。这些安全漏洞已经被广泛研究和验证,并且已经有实际攻击案例。

碰撞攻击是指找到两个不同的输入消息,但它们经过MD5哈希后产生相同的哈希值。这使得攻击者可以伪造具有相同哈希值的消息,这对于数字签名和身份验证等场景具有严重的安全威胁。

预图攻击是指根据已知的MD5哈希值,推导出与之对应的原始消息。虽然这种攻击比碰撞攻击更困难,但仍然是MD5算法的一个安全漏洞,特别是在密码破解等场景下可能被利用。

综上所述,MD5算法的安全性已经受到广泛质疑,并且不再适合用于安全敏感的应用程序。因此,建议使用更安全的哈希算法,如SHA-256、SHA-3等。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/826644.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

一堆喷儿香喷儿香的工具网站-已经收藏-搜嗖工具箱!

文心一言 https://yiyan.baidu.com/ ​ ChatGpt横空出世的横空出世好像一把钥匙,开启了大模型时代,国内也有不错的产品,比如百度的文心一言,从3.5到4.0看得见的成长,现在的文心一言是我们工作中不可缺少的好帮手&am…

javaEE初阶——多线程(五)

T04BF 👋专栏: 算法|JAVA|MySQL|C语言 🫵 小比特 大梦想 此篇文章与大家分享关于多线程的文章第五篇关于 多线程代码案例二 阻塞队列 如果有不足的或者错误的请您指出! 目录 2.阻塞队列2.1常见队列2.2 生产者消费者模型有利于进行解耦合程序进行削峰填谷…

走近网络安全公司:F5探索应用安全至简之道

伴随企业加速数字化转型工作、扩展到新的基础设施环境并采用微服务架构,企业正在拥抱混合和多云基础设施所带来的灵活性。现在跨越四种环境部署应用的企业,其平均需要管理和保护的应用数据路径比应用仅限本地部署时多10倍,复杂性呈指数级增加…

2024数学建模时间汇总与竞赛攻略

目录 2024数学建模汇总(时间、报名费、获奖率、竞赛级别、是否可跨校) 中国高校大数据挑战赛 “华数杯”国际大学生数学建模竞赛 美国大学生数学建模竞赛(美赛) 数学中国(认证杯)数学建模网络挑战赛 …

盲盒一番赏小程序:打开未知的惊喜之旅

在快节奏的生活中,人们总是渴望寻找一份属于自己的小确幸。盲盒一番赏小程序,正是这样一个为你带来无尽惊喜与乐趣的平台。我们精心打造这一小程序,让每一次点击都成为一次全新的探索,让每一次选择都充满无限可能。 盲盒一番赏小…

JavaWeb--05Vue项目简介

Vue项目简介 1 创建vue项目2 Vue项目目录结构3 运行Vue项目3 Vue项目开发流程 1 创建vue项目 环境准备好了,接下来我们需要通过Vue-cli创建一个vue项目,然后再学习一下vue项目的目录结构。Vue-cli提供了如下2种方式创建vue项目: 命令行:直接…

用Cmake编译程序时,链接到FFmpeg库

用Cmake编译程序时,链接到FFmpeg库 一、前言 可喜可贺,折腾了一晚上终于把这个勾八链接成功了,已经要吐了。看到下面控制台的输出,吾心甚慰呀😭 [100%] Linking CXX executable rknn_yolov5_demo [100%] Built targe…

SpringBoot 操作 Redis

导入对应版本的依赖 <dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-data-redis</artifactId> </dependency>修改配置文件中的信息 spring:redis:host: 127.0.0.1port: 8888注意: 我这里 xsh…

【python】深度解剖!一文搞懂字符串常用功能

【python】深度解剖!一文搞懂字符串常用功能 【先赞后看养成习惯】求点赞+关注+收藏😀 目录 【python】深度解剖!一文搞懂字符串常用功能字符串的创建:字符串的格式化——占位符.format方法字符串的分割字符串的连接字符串的重复字符串的大小写转换字符串的判断字符串的…

sherpa + ncnn 离线语音识别

目录结构 前言音视频格式转为wavsherpa-ncnn编译LinuxWindowswindows编译中遇到的问题问题“nmake -? failed with: no such file or directory”编译失败原因 成功编译截图 可执行程序说明模型下载语言识别测试LinuxWindows 参考文献 前言 小编需要实现离线音视频语言部分识…

C++ 程序的内存分配

C 程序的内存分配 C 程序的内存分配栈堆数据区程序代码区参考 C 程序的内存分配 一个 C 编译的程序占用内存分为以下几个部分&#xff08;从高地址到低地址&#xff09;&#xff1a; 内核空间&#xff1a;由操作系统创建并控制&#xff0c;用户代码不能读写。栈&#xff1a;由…

mysql 日环比 统计

接到一个任务&#xff0c;要计算日环比的情况。 16、查询销售额日环比情况 日环比&#xff1a; &#xff08;今日-昨日&#xff09;/ 昨日 的一个比率情况。 1&#xff0c;建表 DROP TABLE IF EXISTS sale; create table sale(id int not null AUTO_INCREMENT,record_date da…

IDEA插件:CodeGeex

前言 CodeGeeX是由清华大学和智谱AI联合开发的多语言代码生成模型。CodeGeeX是一款AI编程助手&#xff0c;其功能类似于Github Copilot、Codeium、CodeWhisperer、Bito等智能编程助手。CodeGeeX支持Python、C、Java、JavaScript、Go等10多种主流编程语言。它可以帮助程…

windows驱动开发-WDM框架(一)

在前面的文章中解释过&#xff0c;NT5.0之后windows确定了新的架构Windows Driver Model (WDM)&#xff0c;在Vista之后又推出了Windows Driver Framework(WDF)&#xff0c;这两个都属于驱动程序框架&#xff0c;那么它们的之间的关系是怎样的&#xff1f; WDF是对WDM进行的封…

利用大语言模型,矢量数据库实现数据库的智能搜索

目的 数据库使用SQL 语言查询数据&#xff0c;数据库的记录中要有一个关键字段&#xff08;通常称为主键字段&#xff0c;它的值在数据库列表中是唯一的&#xff09;,数据记录是结构化的. 如果你需要根据数据记录的内容来查询数据记录&#xff0c;就需要通过Select 语句在数据库…

OpenCV杂记(1):绘制OSD(cv::getTextSize, cv::putText)

1. 简述 我们使用OpenCV时&#xff0c;有时会在图像的某个位置绘制OSD信息&#xff0c;如绘制一些字符串作为指示信息。 本文将简要介绍在图像&#xff08;cv::Mat&#xff09;上绘制固定的字符串信息。 2. 使用的API &#xff08;1&#xff09;cv::getTextSize() CV_EXPORT…

vue3 删除对象中的属性,可以使用js里的delete,但需注意ts定义对象类型!

如上如&#xff0c;当使用delete 删除stateData中的属性时&#xff0c; 报错&#xff0c;意思为 TypeScript 错误“‘delete’ 运算符的操作数必须是可选的 什么原因呢&#xff1f;是因为我偷懒 缺少了ts定义类型 方法一&#xff1a; &#xff08;不推荐&#xff09; delete …

MultiHeadAttention在Tensorflow中的实现原理

前言 通过这篇文章&#xff0c;你可以学习到Tensorflow实现MultiHeadAttention的底层原理。 一、MultiHeadAttention的本质内涵 1.Self_Atention机制 MultiHeadAttention是Self_Atention的多头堆嵌&#xff0c;有必要对Self_Atention机制进行一次深入浅出的理解&#xff0c;这…

Linux Makefile用法

1、什么是makefile&#xff1f; Makefile&#xff1a;将不同模块放在不同的目录中&#xff0c;定义一系列的规则进行 “自动化编译”2、Makefile写法 vim makefile 填写样例&#xff1a; app:sub.c add.c mult.c div.c main.cgcc sub.c add.c mult.c div.c main.c -o app3、工作…

刷代码随想录有感(39):每层最大值

题干&#xff1a; 代码&#xff1a; /*** Definition for a binary tree node.* struct TreeNode {* int val;* TreeNode *left;* TreeNode *right;* TreeNode() : val(0), left(nullptr), right(nullptr) {}* TreeNode(int x) : val(x), left(nullptr), …