MD5算法实验C++
C语言实现MD5加密,竟如此简单!
C语言实现MD5加密,竟如此简单!本文详细讲解视频已经上传到B站:https:///video/BV1uy4y1p7on/公众号后台回复【md5】即可获得本文所有源码。
一、摘要算法摘要算法又称哈希算法。
它表示输入任意长度的数据,输出固定长度的数据,它的主要特征是加密过程不需要密钥,并且经过加密的数据无法被解密。
目前可以被解密逆向的只有CRC32算法,只有输入相同的明文数据经过相同的消息摘要算法才能得到相同的密文。
消息摘要算法不存在密钥的管理与分发问题,适合于分布式网络上使用。
由于其加密计算的工作量相当巨大,所以以前的这种算法通常只用于数据量有限的情况下的加密。
消息摘要算法分为三类:•MD(Message Digest):消息摘要•SHA(Secure Hash Algorithm):安全散列•MAC(Message Authentication Code):消息认证码这三类算法的主要作用:验证数据的完整性二、MD5简介MD5即Message-Digest Algorithm 5(信息-摘要算法)。
属于摘要算法,是一个不可逆过程,就是无论多大数据,经过算法运算后都是生成固定长度的数据,结果使用16进制进行显示的128bit的二进制串。
通常表示为32个十六进制数连成的字符串。
MD5有什么用?用于确保信息传输完整一致。
是计算机广泛使用的杂凑算法之一(又译摘要算法、哈希算法),主流编程语言普遍已有MD5实现。
更多用在文档校验上,用来生成密钥检测文档是否被篡改。
三、在线MD5加密有很多在线进行MD5加密的网站,如下:/code/c26.html举例: 给字符串 12334567 加密成。
如图结果为:32135A337F8DC8E2BB9A9B80D86BDFD0 四、C语言实现MD5算法源文件如下:md5.h#ifndef MD5_H#define MD5_Htypedef struct{unsigned int count[2];unsigned int state[4];unsigned char buffer[64];}MD5_CTX;#define F(x,y,z) ((x & y) | (~x & z))#define G(x,y,z) ((x & z) | (y & ~z))#define H(x,y,z) (x^y^z)#define I(x,y,z) (y ^ (x | ~z))#define ROTATE_LEFT(x,n) ((x << n) | (x >> (32-n)))#define FF(a,b,c,d,x,s,ac) \{ \a += F(b,c,d) + x + ac; \a = ROTATE_LEFT(a,s); \a += b; \}#define GG(a,b,c,d,x,s,ac) \{ \a += G(b,c,d) + x + ac; \a = ROTATE_LEFT(a,s); \a += b; \}#define HH(a,b,c,d,x,s,ac) \{ \a += H(b,c,d) + x + ac; \a = ROTATE_LEFT(a,s); \a += b; \}#define II(a,b,c,d,x,s,ac) \{ \a += I(b,c,d) + x + ac; \a = ROTATE_LEFT(a,s); \a += b; \}void MD5Init(MD5_CTX *context);void MD5Update(MD5_CTX *context,unsigned char *input,u nsigned int inputlen);void MD5Final(MD5_CTX *context,unsigned char digest[16]);void MD5Transform(unsigned int state[4],unsigned char block[64]);void MD5Encode(unsigned char *output,unsigned int *input ,unsigned int len);void MD5Decode(unsigned int *output,unsigned char *inpu t,unsigned int len);#endifmd5.c#include <memory.h>#include 'md5.h'unsigned char PADDING[]={0x80,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0, 0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0};void MD5Init(MD5_CTX *context){context->count[0] = 0;context->count[1] = 0;context->state[0] = 0x67452301;context->state[1] = 0xEFCDAB89;context->state[2] = 0x98BADCFE;context->state[3] = 0x10325476;}void MD5Update(MD5_CTX *context,unsigned char *input,u nsigned int inputlen){unsigned int i = 0,index = 0,partlen = 0;index = (context->count[0] >> 3) & 0x3F;partlen = 64 - index;context->count[0] += inputlen << 3;if(context->count[0] < (inputlen << 3))context->count[1]++;context->count[1] += inputlen >> 29;if(inputlen >= partlen){memcpy(&context->buffer[index],input,partlen);MD5Transform(context->state,context->buffer);for(i = partlen;i+64 <= inputlen;i+=64)MD5Transform(context->state,&input[i]);index = 0;}else{i = 0;}memcpy(&context->buffer[index],&input[i],inputlen-i);}void MD5Final(MD5_CTX *context,unsigned char digest[16]) {unsigned int index = 0,padlen = 0;unsigned char bits[8];index = (context->count[0] >> 3) & 0x3F;padlen = (index < 56)?(56-index):(120-index);MD5Encode(bits,context->count,8);MD5Update(context,PADDING,padlen);MD5Update(context,bits,8);MD5Encode(digest,context->state,16);}void MD5Encode(unsigned char *output,unsigned int *input ,unsigned int len){unsigned int i = 0,j = 0;while(j < len){output[j] = input[i] & 0xFF;output[j+1] = (input[i] >> 8) & 0xFF;output[j+2] = (input[i] >> 16) & 0xFF;output[j+3] = (input[i] >> 24) & 0xFF;i++;j+=4;}}void MD5Decode(unsigned int *output,unsigned char *inpu t,unsigned int len){unsigned int i = 0,j = 0;while(j < len){output[i] = (input[j]) |(input[j+1] << 8) |(input[j+2] << 16) |(input[j+3] << 24);i++;j+=4;}}void MD5Transform(unsigned int state[4],unsigned char block[64]){unsigned int a = state[0];unsigned int b = state[1];unsigned int c = state[2];unsigned int d = state[3];unsigned int x[64];MD5Decode(x,block,64);FF(a, b, c, d, x[ 0], 7, 0xd76aa478); /* 1 */FF(d, a, b, c, x[ 1], 12, 0xe8c7b756); /* 2 */FF(c, d, a, b, x[ 2], 17, 0x242070db); /* 3 */FF(b, c, d, a, x[ 3], 22, 0xc1bdceee); /* 4 */FF(a, b, c, d, x[ 4], 7, 0xf57c0faf); /* 5 */FF(d, a, b, c, x[ 5], 12, 0x4787c62a); /* 6 */FF(c, d, a, b, x[ 6], 17, 0xa8304613); /* 7 */FF(b, c, d, a, x[ 7], 22, 0xfd469501); /* 8 */FF(a, b, c, d, x[ 8], 7, 0x698098d8); /* 9 */FF(d, a, b, c, x[ 9], 12, 0x8b44f7af); /* 10 */FF(c, d, a, b, x[10], 17, 0xffff5bb1); /* 11 */FF(b, c, d, a, x[11], 22, 0x895cd7be); /* 12 */FF(a, b, c, d, x[12], 7, 0x6b901122); /* 13 */FF(d, a, b, c, x[13], 12, 0xfd987193); /* 14 */FF(c, d, a, b, x[14], 17, 0xa679438e); /* 15 */FF(b, c, d, a, x[15], 22, 0x49b40821); /* 16 *//* Round 2 */GG(a, b, c, d, x[ 1], 5, 0xf61e2562); /* 17 */GG(d, a, b, c, x[ 6], 9, 0xc040b340); /* 18 */GG(c, d, a, b, x[11], 14, 0x265e5a51); /* 19 */ GG(b, c, d, a, x[ 0], 20, 0xe9b6c7aa); /* 20 */GG(a, b, c, d, x[ 5], 5, 0xd62f105d); /* 21 */ GG(d, a, b, c, x[10], 9, 0x2441453); /* 22 */ GG(c, d, a, b, x[15], 14, 0xd8a1e681); /* 23 */ GG(b, c, d, a, x[ 4], 20, 0xe7d3fbc8); /* 24 */ GG(a, b, c, d, x[ 9], 5, 0x21e1cde6); /* 25 */ GG(d, a, b, c, x[14], 9, 0xc33707d6); /* 26 */ GG(c, d, a, b, x[ 3], 14, 0xf4d50d87); /* 27 */ GG(b, c, d, a, x[ 8], 20, 0x455a14ed); /* 28 */ GG(a, b, c, d, x[13], 5, 0xa9e3e905); /* 29 */ GG(d, a, b, c, x[ 2], 9, 0xfcefa3f8); /* 30 */ GG(c, d, a, b, x[ 7], 14, 0x676f02d9); /* 31 */ GG(b, c, d, a, x[12], 20, 0x8d2a4c8a); /* 32 *//* Round 3 */HH(a, b, c, d, x[ 5], 4, 0xfffa3942); /* 33 */ HH(d, a, b, c, x[ 8], 11, 0x8771f681); /* 34 */ HH(c, d, a, b, x[11], 16, 0x6d9d6122); /* 35 */ HH(b, c, d, a, x[14], 23, 0xfde5380c); /* 36 */ HH(a, b, c, d, x[ 1], 4, 0xa4beea44); /* 37 */ HH(d, a, b, c, x[ 4], 11, 0x4bdecfa9); /* 38 */ HH(c, d, a, b, x[ 7], 16, 0xf6bb4b60); /* 39 */ HH(b, c, d, a, x[10], 23, 0xbebfbc70); /* 40 */ HH(a, b, c, d, x[13], 4, 0x289b7ec6); /* 41 */ HH(d, a, b, c, x[ 0], 11, 0xeaa127fa); /* 42 */ HH(c, d, a, b, x[ 3], 16, 0xd4ef3085); /* 43 */ HH(b, c, d, a, x[ 6], 23, 0x4881d05); /* 44 */ HH(a, b, c, d, x[ 9], 4, 0xd9d4d039); /* 45 */ HH(d, a, b, c, x[12], 11, 0xe6db99e5); /* 46 */ HH(c, d, a, b, x[15], 16, 0x1fa27cf8); /* 47 */ HH(b, c, d, a, x[ 2], 23, 0xc4ac5665); /* 48 *//* Round 4 */II(a, b, c, d, x[ 0], 6, 0xf4292244); /* 49 */II(d, a, b, c, x[ 7], 10, 0x432aff97); /* 50 */II(c, d, a, b, x[14], 15, 0xab9423a7); /* 51 */II(b, c, d, a, x[ 5], 21, 0xfc93a039); /* 52 */II(a, b, c, d, x[12], 6, 0x655b59c3); /* 53 */II(d, a, b, c, x[ 3], 10, 0x8f0ccc92); /* 54 */II(c, d, a, b, x[10], 15, 0xffeff47d); /* 55 */II(b, c, d, a, x[ 1], 21, 0x85845dd1); /* 56 */II(a, b, c, d, x[ 8], 6, 0x6fa87e4f); /* 57 */II(d, a, b, c, x[15], 10, 0xfe2ce6e0); /* 58 */II(c, d, a, b, x[ 6], 15, 0xa3014314); /* 59 */II(b, c, d, a, x[13], 21, 0x4e0811a1); /* 60 */II(a, b, c, d, x[ 4], 6, 0xf7537e82); /* 61 */II(d, a, b, c, x[11], 10, 0xbd3af235); /* 62 */II(c, d, a, b, x[ 2], 15, 0x2ad7d2bb); /* 63 */II(b, c, d, a, x[ 9], 21, 0xeb86d391); /* 64 */state[0] += a;state[1] += b;state[2] += c;state[3] += d;}五、MD5加密实例MD5加密步骤如下:1.定义MD5_CTX md5c;2.初始化/******************************************************** * 名称: MD5Init()* 功能: 初始化MD5结构体* 入口参数:context:要初始化的MD5结构体* 出口参数: 无*********************************************************/MD5Init(MD5_CTX *context);3.MD5值计算实现MD5值的计算及结构体的更新:/********************************************************** 名称: MD5Update()* 功能: 将要加密的信息传递给初始化过的MD5结构体,无返回值* 入口参数:context:初始化过了的MD5结构体input:需要加密的信息,可以任意长度inputLen:指定input的长度* 出口参数: 无*********************************************************/MD5Update(MD5_CTX *context,(unsigned char *)input,inpu tLen);4.输出转换/********************************************************** 名称: MD5Update()* 功能: 将加密结果存储到,无返回值* 入口参数:context:初始化过了的MD5结构体digest :加密过的结果* 出口参数: 无*********************************************************/MD5Final(MD5_CTX *context,unsigned char digest[16]);5.格式整理转换成32位的16进制字符串。
实验三 MD5算法的设计与实现
实验三MD5算法的设计与实现MD5算法及C++实现一、理论部分:1、预备知识1.1什么是数据校验通俗的说,就是为保证数据的完整性,用一种指定的算法对原始数据计算出的一个校验值。
接收方用同样的算法计算一次校验值,如果和随数据提供的校验值一样,就说明数据是完整的。
1.2最简单的检验实现方法:最简单的校验就是把原始数据和待比较数据直接进行比较,看是否完全一样这种方法是最安全最准确的。
同时也是效率最低的。
适用范围:简单的数据量极小的通讯。
应用例子:龙珠cpu在线调试工具bbug.exe。
它和龙珠cpu间通讯时,bbug发送一个字节cpu返回收到的字节,bbug确认是刚才发送字节后才继续发送下一个字节的。
1.3奇偶校验Parity Check实现方法:在数据存储和传输中,字节中额外增加一个比特位,用来检验错误。
校验位可以通过数据位异或计算出来。
应用例子:单片机串口通讯有一模式就是8位数据通讯,另加第9位用于放校验值。
1.4 bcc异或校验法(block check character)实现方法:很多基于串口的通讯都用这种既简单又相当准确的方法。
它就是把所有数据都和一个指定的初始值(通常是0)异或一次,最后的结果就是校验值,通常把她附在通讯数据的最后一起发送出去。
接收方收到数据后自己也计算一次异或和校验值,如果和收到的校验值一致就说明收到的数据是完整的。
校验值计算的代码类似于:unsigned uCRC=0;//校验初始值for(int i=0;i<DataLenth;i++) uCRC^=Data[i];适用范围:适用于大多数要求不高的数据通讯。
应用例子:ic卡接口通讯、很多单片机系统的串口通讯都使用。
1.5 crc循环冗余校验(Cyclic Redundancy Check)实现方法:这是利用除法及余数的原理来进行错误检测的.将接收到的码组进行除法运算,如果除尽,则说明传输无误;如果未除尽,则表明传输出现差错。
md5 算法分片计算 c语言
md5 算法分片计算 c语言MD5算法是一种广泛使用的哈希函数,它可以将任意长度的消息作为输入,产生一个128位(16字节)的哈希值作为输出。
在C 语言中,我们可以使用MD5算法对数据进行分片计算,以下是一个简单的示例:c.#include <stdio.h>。
#include <string.h>。
#include <openssl/md5.h>。
void calculate_md5(const char data, size_t data_len, unsigned char md5_sum) {。
MD5_CTX context;MD5_Init(&context);MD5_Update(&context, data, data_len);MD5_Final(md5_sum, &context);}。
int main() {。
const char data = "Hello, MD5!";unsigned char md5_sum[MD5_DIGEST_LENGTH];calculate_md5(data, strlen(data), md5_sum);printf("MD5 sum: ");for (int i = 0; i < MD5_DIGEST_LENGTH; i++) {。
printf("%02x", md5_sum[i]);}。
printf("\n");return 0;}。
在这个示例中,我们使用了OpenSSL库中提供的MD5函数来计算MD5哈希值。
首先,我们定义了一个`calculate_md5`函数,它接受输入数据、数据长度和一个指向存储MD5哈希值的缓冲区的指针。
然后,在`main`函数中,我们调用`calculate_md5`函数来计算给定数据的MD5哈希值,并打印输出结果。
MD5实验报告(word文档良心出品)
一、MD5算法描述MD5算法是计算机安全领域广泛使用的一种散列算法之一,用以提供消息的完整性保护。
本次实验中,对于输入的任意长度的一段数据,输出唯一固定的128位MD5散列值。
MD噂法分为以下步骤:1 、初始化(填充)首先对于任意长度的消息,将其按照512 位进行分组,为保证每个分组都是512 位,需要对消息进行填充。
填充方法是先在消息末尾填一个1,再在后面填若干个0,使其长度I恰好满足I=448(mod512),然后在其后附加上64 位的(填充前)消息长度, 确保不同的消息在填充后不相同消息在填充后不相同。
其次将每个分组512 位的数据划分为16个32 位的子分组,在实现过程中,将其表示为16 个32 位整数进行运算。
初始化四个32 位链接变量为:A=0x67452301B=0xefcdab89C=0x98badcfeD=0x103254762、单步操作与主循环MD5共分为4轮,每轮进行16步单步操作,MD5的单步操作如下其中,g k (k=0,123)是每次运算中用到的非线性函数(每轮一个),用C 语言定义如下^define gO(b> c,. d) ((b&c) (、b 赴d))ftdefine gl (b, c> d) ((b&d) )ffdef ine g2(b, c f d) (b e d) tfdef ine g3(b, c f d) (c (b d))数据块X[ d(i)]来自于初始化中分好的16个32位数据分组,根据循环 步数按以下置换函数选取(i 表示循环的第i 步)T i 是一些随机数序列,T i 的值如下(按顺序选取)Pdefine rO(i)#define rl (i)Sdefine r2(i)Sdefine r3 (i)(i%16) <(l+5*i)%16) ((5+3*i)%16) <(7*i)%16)unsigned int T .64.=:0xd76aa478;0xe8c7b756?Ox242070dL Oxclbdce-ee,0xf57c0faf,0x4787c62a T0xa8304613:0xfd469501? 0x698098d8,0x8b44f7af7OxffffSbbl. 0x895cd7te7 0x6b90U22f0xfd987193,0x3679438^ 0x49b40821, Ch<f61u2562, OxcO4Ob34O, 0x265e5a51?Oxe^GcTaa, 0xd62fl05d. 0x02141453, 0xd8ale68L?0xe7d3fbc8?Ox21.elcde6, 0xc337O7d6T0xf4d50d87r0x455al4ed f0^a9e3e905,OxfcefaofS, 0x676f02d9. 0x8d2a4c8a, 0xfffa3942,0x8771f681, 0x6旳d6122, 0xfM538(k, 0xa4beea44:0x4bdecfa9?0xf6bb4b60?Oxbebfbc7O? 0x289b7ec6:0xeaaI27£a T0s64ef3085:0x04851 dO5? 0xd9d4d039, 0xe6db99e5?DxlfaSycfS, 0xc4ac566570xf4292244?0x432aff97I0xab9423a7J0xfc93a039, 0x655b59c3,0x8f0ccc92, Oxffeff47d. 0x85845ddl? 0x6faS7e4f, 0xfe2ce6e0,0xa3014314?Ox4eO811al, Oxf7537eS2. 0xbd3af2o570x2ad7d2bb.0xeb86d391?};vv<S i表示循环左移S i位,S i的数值如下(按顺序选取)unsigned int S[64]={7,12417, 22, 7,12f17f 22, 7, 12,17, 22t 1, 12,17t 22,5, 9,14, 20, 5, 9, 14, 20, 5, 9, 14,20, 5, 9,14, 20,4, 11,16, 23, 4,11,16, 23, 4,11,16, 23* 4,11, 16, 23,6,10,15, 21, 6,10,15, 21, 6F10,15, 21, 6, 10, 15, 21};因此,MD5主要的一个单步操作可以表示为a =b ((a f(b,c,d) T[i] X[J(i)])尺S[i])每轮16步结束后的4个32位数作为下一轮的输入继续循环束后的输入再加上初始值即得到最终结果128位的散列值' 循环如下图四轮结64步主A| B| C| Bjn 彳第1轮16步便用卧P Q; 电 B ] cl 叫+1第2轮"步便用驻Pl |A| Bl C| D| "■►f 第3轮16步便用知pt1 ~Al Bl Cl Dl ~程序验证为了验证程序的正确性,我们通过查找 Wikipedia 上使用的3个 测试向量来进行测试,测试向量如下:ND5(*The quick bxown fox jumps over the lazy dog")=9e 107d9d372bb6326b d8Id3542a419d6KD5 (*The quick brown fos jumps over the lazy dog- ■*)= edd909c290dOfb IcaOGSff addf 2 2cbdOMD5(^)=d41J8cd98f0Ob204e980O99SecfS427e运行程序,得到的结果如下:X * - ■ <■>第4轮"步使用g 尹P3v ! I ! I 1 + 模屮加C ^D 工cyniF可以看到,实验结果与测试向量相同,验证了程序的正确性。
MD5算法实验报告
MD5算法实验报告实验报告:MD5算法的原理与应用一、实验目的本实验旨在通过研究MD5(Message-Digest Algorithm 5)算法的原理和应用,了解其工作原理和实现过程,并掌握其具体应用技巧。
二、实验内容1.了解MD5算法的基本原理和特点;2.分析MD5算法的具体实现过程;3.实现一个简单的MD5加密程序,并进行测试;4.掌握MD5算法的应用技巧。
三、实验原理1. 输入任意长度的数据,输出固定长度的Hash值,通常为128位;2. 安全性较高,Hash值的变化能较好地反映原始数据的变化;3. 不可逆性:无法通过Hash值反推出原始数据;4. Hash值相同的概率很低,冲突概率较小。
1.数据填充:对输入数据进行填充,使其长度满足一定要求;2.划分数据:将填充后的数据划分为多个512位的数据块;3.初始化变量:设置四个32位的变量,作为初始值;4.处理数据块:对每个数据块进行处理,分为四轮,每轮包括四个步骤,即置换、模运算、加法和循环左移操作。
5. 输出结果:将四个32位变量连接起来,即得到最终的128位Hash值。
四、实验过程1.学习MD5算法的原理和实现细节;2. 使用Python编程语言实现一个简单的MD5加密程序,并进行测试。
实验代码如下:```import hashlibdef md5_encrypt(source):md5 = hashlib.md5md5.update(source.encode('utf-8'))return md5.hexdigestif __name__ == '__main__':source_str = input("请输入要加密的字符串:")encrypted_str = md5_encrypt(source_str)print("加密后的字符串为:", encrypted_str)```五、实验结果与分析通过上述实验代码,可以输入一个字符串,程序将会对该字符串进行MD5加密,并将加密结果输出。
MD5加密算法-c源代码
context->count[1] += ((UINT4)inputLen >> 29);
/*计算已有的字节数长度还差多少字节可以凑成64的整倍数*/
partLen = 64 - index;
/* Transform as many times as possible.
static void Decode(UINT4 *, unsigned char *, unsigned int);
/*
用于bits填充的缓冲区,为什么要64个字节呢?因为当欲加密的信息的bits数被512除其余数为448时,
需要填充的bits的最大值为512=64*8。
*/
static unsigned char PADDING[64] = {
转换结果保存到context->state中
*/
for(i = partLen; i + 63 < inputLen; i += 64)/*把i+63<inputlen改为i+64<=inputlen更容易理解*/
MD5Transform(context->state, &input[i]);
index = 0;
0x80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0
MD5加密C语言实现
MD5加密C语言实现MD5 (Message Digest Algorithm 5) 是一种常用的密码散列函数,用于将数据加密为128位长度的摘要。
在C语言中,可以通过一系列步骤来实现MD5加密算法。
1.准备工作:首先需要包含一些C标准头文件和预定义常量。
在C语言中,可以使用以下代码来实现:```c#include <stdio.h>#include <stdlib.h>#include <string.h>#include <stdint.h>#define HASH_LENGTH 16```2.定义数据结构:MD5算法涉及到一个64字节的消息块和一个4字节的消息摘要块。
在C语言中,可以使用以下代码来定义这些结构:```ctypedef structuint8_t data[64];uint32_t datalen;uint32_t bitlen[2];uint32_t state[4];}MD5_CTX;typedef uint8_t (*hash_function)(uint8_t *);```3.定义常量和函数:MD5算法使用到一些常量和函数。
在C语言中,可以使用以下代码来定义这些常量和函数:```cconst uint32_t k[64] =// more constants ...};const uint32_t r[64] =7,12,17,22,7,12,17,22,// more constants ...};void md5_transform(MD5_CTX *ctx, uint8_t data[]);void md5_init(MD5_CTX *ctx)ctx->datalen = 0;ctx->bitlen[0] = 0;ctx->bitlen[1] = 0;ctx->state[1] = 0xEFCDAB89;ctx->state[2] = 0x98BADCFE;void md5_update(MD5_CTX *ctx, uint8_t data[], uint32_t len) for (uint32_t i = 0; i < len; i++)ctx->data[ctx->datalen] = data[i];ctx->datalen++;if (ctx->datalen == 64)md5_transform(ctx, ctx->data);ctx->bitlen[0] += 512;ctx->bitlen[1] += (ctx->bitlen[0] < 512);ctx->datalen = 0;}}void md5_final(MD5_CTX *ctx, uint8_t hash[])uint32_t i = ctx->datalen;if (ctx->datalen < 56)ctx->data[i++] = 0x80;while (i < 56)ctx->data[i++] = 0x00;}} elsectx->data[i++] = 0x80;while (i < 64)ctx->data[i++] = 0x00;}md5_transform(ctx, ctx->data);memset(ctx->data, 0, 56);}ctx->bitlen[0] += ctx->datalen * 8;ctx->bitlen[1] += (ctx->bitlen[0] < ctx->datalen * 8); ctx->data[63] = ctx->bitlen[0] & 0xff;ctx->data[62] = (ctx->bitlen[0] >> 8) & 0xff;ctx->data[61] = (ctx->bitlen[0] >> 16) & 0xff;ctx->data[60] = (ctx->bitlen[0] >> 24) & 0xff;ctx->data[59] = ctx->bitlen[1] & 0xff;ctx->data[58] = (ctx->bitlen[1] >> 8) & 0xff;ctx->data[57] = (ctx->bitlen[1] >> 16) & 0xff;ctx->data[56] = (ctx->bitlen[1] >> 24) & 0xff;md5_transform(ctx, ctx->data);for (i = 0; i < 4; i++)hash[i] = (ctx->state[0] >> (i * 8)) & 0xff;hash[i + 4] = (ctx->state[1] >> (i * 8)) & 0xff;hash[i + 8] = (ctx->state[2] >> (i * 8)) & 0xff;hash[i + 12] = (ctx->state[3] >> (i * 8)) & 0xff;}void md5_transform(MD5_CTX *ctx, uint8_t data[])uint32_t a, b, c, d, f, g, temp;uint32_t m[16], i, j;for (i = 0, j = 0; i < 16; i++, j += 4)m[i] = (data[j]) + (data[j + 1] << 8) + (data[j + 2] << 16) + (data[j + 3] << 24);}a = ctx->state[0];b = ctx->state[1];c = ctx->state[2];d = ctx->state[3];for (i = 0; i < 64; i++)if (i < 16)f=(b&c),((~b)&d);g=i;} else if (i < 32)f=(d&b),((~d)&c);g=(5*i+1)%16;} else if (i < 48)f=b^c^d;g=(3*i+5)%16;} elsef=c^(b,(~d));g=(7*i)%16;}temp = d;d=c;c=b;b = b + leftrotate((a + f + k[i] + m[g]), r[i]);a = temp;}ctx->state[0] += a;ctx->state[1] += b;ctx->state[2] += c;ctx->state[3] += d;```4.实现加密函数:最后,可以编写一个简单的调用MD5算法的加密函数。
C语言实现MD5算法
C语言实现MD5算法MD5(Message-Digest Algorithm 5)是一种常用的哈希函数算法,广泛用于验证数据完整性、密码存储和数字证书等领域。
下面是使用C语言实现MD5算法的代码。
这段代码包含了MD5算法的各个步骤,包括初始化MD5结构体、填充数据、更新状态、计算摘要等。
```c#include <stdio.h>#include <stdint.h>#include <string.h>//定义MD5常量#define B 0xEFCDAB89#define C 0x98BADCFE//循环左移宏定义#define LEFT_ROTATE(x, n) (((x) << (n)) , ((x) >> (32-(n)))) //填充消息void padMessage(uint8_t *message, uint32_t length)//计算需要填充的字节数uint32_t padLength = (length % sizeof(uint32_t) == 56) ? 64 : 56;padLength = padLength - (length % sizeof(uint32_t));//填充1位1message[length++] = 0x80;//填充0位for (uint32_t i = 0; i < padLength; i++) message[length++] = 0x00;}//在消息末尾添加原始消息的长度(以位表示)for (uint32_t i = 0; i < sizeof(uint32_t); i++) message[length++] = (length << 3) >> (i * 8); }//初始化MD5结构体void initMD5(uint32_t *state)state[0] = A;state[1] = B;state[2] = C;state[3] = D;//更新状态void updateState(uint32_t *state, uint32_t *M)uint32_t A = state[0], B = state[1], C = state[2], D = state[3];//定义MD5循环运算函数#define MD5_FUNCTION(a, b, c, d, k, s, i) \a=b+LEFT_ROTATE((a+F(b,c,d)+M[k]+T[i]),s)//迭代压缩消息MD5_FUNCTION(A,B,C,D,0,7,1);MD5_FUNCTION(D,A,B,C,1,12,2);MD5_FUNCTION(C,D,A,B,2,17,3);MD5_FUNCTION(B,C,D,A,3,22,4);MD5_FUNCTION(A,B,C,D,4,7,5);MD5_FUNCTION(D,A,B,C,5,12,6);MD5_FUNCTION(C,D,A,B,6,17,7);MD5_FUNCTION(B,C,D,A,7,22,8);MD5_FUNCTION(A,B,C,D,8,7,9);MD5_FUNCTION(D,A,B,C,9,12,10);MD5_FUNCTION(C,D,A,B,10,17,11);MD5_FUNCTION(B,C,D,A,11,22,12);MD5_FUNCTION(A,B,C,D,12,7,13);MD5_FUNCTION(C,D,A,B,14,17,15); MD5_FUNCTION(B,C,D,A,15,22,16); MD5_FUNCTION(A,B,C,D,1,5,17); MD5_FUNCTION(D,A,B,C,6,9,18); MD5_FUNCTION(C,D,A,B,11,14,19); MD5_FUNCTION(B,C,D,A,0,20,20); MD5_FUNCTION(A,B,C,D,5,5,21); MD5_FUNCTION(D,A,B,C,10,9,22); MD5_FUNCTION(C,D,A,B,15,14,23); MD5_FUNCTION(B,C,D,A,4,20,24); MD5_FUNCTION(A,B,C,D,9,5,25); MD5_FUNCTION(D,A,B,C,14,9,26); MD5_FUNCTION(C,D,A,B,3,14,27); MD5_FUNCTION(B,C,D,A,8,20,28); MD5_FUNCTION(A,B,C,D,13,5,29); MD5_FUNCTION(D,A,B,C,2,9,30); MD5_FUNCTION(C,D,A,B,7,14,31); MD5_FUNCTION(B,C,D,A,12,20,32);MD5_FUNCTION(D,A,B,C,8,11,34); MD5_FUNCTION(C,D,A,B,11,16,35); MD5_FUNCTION(B,C,D,A,14,23,36); MD5_FUNCTION(A,B,C,D,1,4,37); MD5_FUNCTION(D,A,B,C,4,11,38); MD5_FUNCTION(C,D,A,B,7,16,39); MD5_FUNCTION(B,C,D,A,10,23,40); MD5_FUNCTION(A,B,C,D,13,4,41); MD5_FUNCTION(D,A,B,C,0,11,42); MD5_FUNCTION(C,D,A,B,3,16,43); MD5_FUNCTION(B,C,D,A,6,23,44); MD5_FUNCTION(A,B,C,D,9,4,45); MD5_FUNCTION(D,A,B,C,12,11,46); MD5_FUNCTION(C,D,A,B,15,16,47); MD5_FUNCTION(B,C,D,A,2,23,48); MD5_FUNCTION(A,B,C,D,0,6,49); MD5_FUNCTION(D,A,B,C,7,10,50); MD5_FUNCTION(C,D,A,B,14,15,51);MD5_FUNCTION(A,B,C,D,12,6,53); MD5_FUNCTION(D,A,B,C,3,10,54); MD5_FUNCTION(C,D,A,B,10,15,55); MD5_FUNCTION(B,C,D,A,1,21,56); MD5_FUNCTION(A,B,C,D,8,6,57); MD5_FUNCTION(D,A,B,C,15,10,58); MD5_FUNCTION(C,D,A,B,6,15,59); MD5_FUNCTION(B,C,D,A,13,21,60); MD5_FUNCTION(A,B,C,D,4,6,61); MD5_FUNCTION(D,A,B,C,11,10,62); MD5_FUNCTION(C,D,A,B,2,15,63); MD5_FUNCTION(B,C,D,A,9,21,64); #undef MD5_FUNCTION//更新状态state[0] += A;state[1] += B;state[2] += C;state[3] += D;//计算MD5摘要void md5(uint8_t *message, uint32_t length, uint32_t *digest) //初始化MD5结构体uint32_t state[4];initMD5(state);//填充消息padMessage(message, length);//计算消息分组数量uint32_t numOfBlocks = length / 64;//处理每个分组for (uint32_t i = 0; i < numOfBlocks; i++)uint32_t M[16];memcpy(M, message + (i * 64), 64);//更新状态updateState(state, M);}//获取MD5摘要memcpy(digest, state, 16);int mai//测试用例uint8_t message[] = "Hello, MD5!";uint32_t length = sizeof(message) - 1;//计算MD5摘要uint32_t digest[4];md5(message, length, digest);//输出摘要printf("MD5 Digest: ");for (int i = 0; i < 4; i++)printf("%02x", ((uint8_t*)digest)[i]);}printf("\n");return 0;```以上是使用C语言实现MD5算法的代码。
密码实验md5
实验一MD5哈希算法报告1 实验目的(1)理解MD5算法的基本思想。
(2)实现MD5算法。
(3)通过该算法加深对HASH 算法的理解。
2 实验内容基于Visual C++,编程实现各类消息的摘要值。
要求:输入任意字符串,以16进制形式输出其MD5值。
3 实验原理MD5是由国际著名密码学家、“图灵奖”获得者兼公钥加密算法RSA 的创始人、麻省理工大学的RonaldRivest 教授于1991年设计的。
它与SHA 共同构成两大哈希算法,广泛应用于金融、证券等电子商务领域。
MD5输入是任意长的消息,分组长度为512比特,输出128比特。
MD5算法框图如下:由图中可看出,MD5是四轮运算,每轮又要进行16步迭代运算,4轮共需64步完成。
其中T[1,…,64]为64个常数,参与不同轮的计算。
X[k]=M[q ×16+k],即消息第q 个分组中的第k 个32比特字(k=1,…,16)。
F 、G 、H 和I 为各轮逻辑函数,定义如下:4轮处理过程中,每轮以不同的次序使用16个字。
在第1轮以字的初始次序使用,第2轮到第4轮分别对字的次序i 做置换后得到一个新次序,然后以新次序使用16个字。
3个置换分别为ρ2(i)=(1+5i) mod 16,ρ3(i)=(5+3i) mod 16和ρ4(i)=7i mod 16。
每轮的每步变换为a=b+CLSs(a+g(b,c,d)+X[k]+T[i),其中g为逻辑函数,CLSs为循环左移s位,X[k]为分组的第k个32bit字,T[i]为常数表中第i个32-bit字,+为模232加法。
特别当g=F,该变换记为FF(a, b, c, d, xj, s, ti)。
1轮中的16步变化按照一定的次序进行,4轮进行64步。
每轮输出、中间结果和最终Hash值存储在缓冲区a、b、c和d。
缓冲区初始化以十六进制表示为a=01234567,b=89ABCDEF,c=FEDCBA98,d=76543210。
c语言md5 base算法
c语言md5 base算法
C语言中可以使用MD5算法来进行消息摘要的计算。
MD5是一种广泛使用的哈希函数,可以将任意长度的消息转换为一个128位的哈希值。
在C语言中,可以使用现成的MD5库来实现MD5算法,也可以手动实现MD5算法。
一种常见的C语言MD5算法实现是使用开源的MD5库,比如OpenSSL库。
使用OpenSSL库可以很方便地计算MD5值,只需要包含相应的头文件,并连接相应的库即可。
通过调用库中的函数,可以对需要计算MD5值的数据进行处理,得到对应的MD5值。
另一种方式是手动实现MD5算法。
MD5算法的核心是对消息进行分块处理、填充、循环移位和模运算等操作,最终得到128位的哈希值。
手动实现MD5算法需要对这些操作进行逐步实现,包括消息预处理、填充、循环处理和结果输出等步骤。
这需要对MD5算法的细节有较深入的理解和编程能力。
无论是使用现成的MD5库还是手动实现MD5算法,都需要注意数据的输入输出格式、内存管理和错误处理等方面,以确保MD5算
法的正确性和安全性。
另外,在实际应用中,还需要考虑数据的保护、加盐等安全机制,以防止MD5值被破解或篡改。
c语言md5加密函数
c语言md5加密函数C语言MD5加密函数MD5(Message Digest Algorithm 5)是一种广泛使用的密码散列函数,常用于数据加密和数据完整性验证。
在C语言中,我们可以通过编写MD5加密函数来实现对数据的加密操作。
本文将介绍如何使用C语言编写一个简单的MD5加密函数,并详细解释其原理和步骤。
一、MD5加密原理MD5加密算法基于消息摘要算法,它将任意长度的消息作为输入,通过一系列复杂的数学运算,生成一个固定长度的密文。
MD5算法的核心思想是将输入的消息进行分块处理,并对每个分块进行位运算和逻辑运算,最终得到一个128位的摘要。
MD5算法具有以下特点:1. 不可逆性:无法从加密后的密文推导出原始消息。
2. 唯一性:不同的输入会产生不同的摘要。
3. 完整性:对于相同的输入,产生的摘要是稳定的。
二、MD5加密步骤MD5算法的加密过程包括以下几个步骤:1. 填充消息:将消息的位数填充为448的倍数,填充方式为在消息末尾添加一个1和若干个0。
2. 添加长度:将填充后的消息长度添加到消息末尾,以64位二进制数表示。
3. 初始化缓冲区:初始化四个32位的缓冲区A、B、C、D,用于存储最终的摘要。
4. 分块处理:将填充后的消息分为若干个512位的分组,对每个分组进行处理。
5. 迭代压缩:对每个分组进行64轮的迭代压缩操作,更新缓冲区的值。
6. 输出结果:最后将缓冲区的值按照小端序输出,得到128位的摘要。
三、C语言实现MD5加密函数以下是一个简单的C语言实现MD5加密函数的示例代码:#include <stdio.h>#include <string.h>#include <stdint.h>// 定义MD5加密函数void md5_encrypt(const uint8_t *message, uint32_t len, uint8_t *digest) {// 初始化缓冲区uint32_t A = 0x67452301;uint32_t B = 0xEFCDAB89;uint32_t C = 0x98BADCFE;uint32_t D = 0x10325476;// 填充消息// ...// 添加长度// ...// 分块处理// ...// 迭代压缩// ...// 输出结果// ...}int main() {// 测试示例uint8_t message[] = "Hello, MD5!";uint8_t digest[16] = {0};md5_encrypt(message, strlen(message), digest); // 输出结果for (int i = 0; i < 16; i++) {printf("%02x", digest[i]);}printf("\n");return 0;}四、总结通过上述的示例代码,我们可以看到使用C语言编写MD5加密函数并不复杂。
md5加密算法的C(C++)代码实现(完整)
正是因为这个原因,现在被黑客使用最多的一种破译密码的方法就是一种被称为"跑字典"的方法。有两种方法得到字典,一种是日常搜集的用做密码的字符串表,另一种是用排列组合方法生成的,先用md5程序计算出这些字典项的md5值,然后再用目标的md5值在这个字典中检索。我们假设密码的最大长度为8位字节(8 bytes),同时密码只能是字母和数字,共26+26+10=62个字符,排列组合出的字典的项数则是p(62,1)+p(62,2)…。+p(62,8),那也已经是一个很天文的数字了,存储这个字典就需要tb级的磁盘阵列,而且这种方法还有一个前提,就是能获得目标账户的密码md5值的情况下才可以。这种加密技术被广泛的应用于unix系统中,这也是为什么unix系统比一般操作系统更为坚固一个重要原因。
<< 这四轮(64步)是:
第一轮
ff(a,b,c,d,m0,7,0xd76aa478)
ff(d,a,b,c,m1,12,0xe8c7b756)
ff(c,d,a,b,m2,17,0x242070db)
ff(b,c,d,a,m3,22,0xc1bdceee)
将上面四个链接变量复制到另外四个变量中:a到a,b到b,c到c,d到d.
主循环有四轮(md4只有三轮),每轮循环都很相似。第一轮进行16次操作。每次操作对a、b、c和d中的其中三个作一次非线性函数运算,然后将所得结果加上第四个变量,文本的一个子分组和一个常数。再将所得结果向右环移一个不定的数,并加上a、b、c或d中之一。最后用该结果取代a、b、c或d中之一。
rivest在1989年开发出md2算法。在这个算法中,首先对信息进行数据补位,使信息的字节长度是16的倍数。然后,以一个16位的检验和追加到信息末尾。并且根据这个新产生的信息计算出散列值。后来,rogier和chauvaud发现如果忽略了检验和将产生md2冲突。md2算法的加密后结果是唯一的——既没有重复。
MD5加密C语言实现
MD5加密C语言实现MD5(Message-Digest Algorithm 5)是一种广泛使用的散列函数,它将任意长度的数据映射为固定长度的散列值。
在本文中,我们将实现一个简单的MD5加密算法的C语言版本。
1.MD5加密算法的基本原理:MD5算法由四个基本步骤组成:填充,初始化,循环计算和摘要。
-填充:将需要加密的数据填充到一个长度为64的倍数的消息块,填充内容为一个1和若干个0,在结尾添加64位表示原始消息长度的二进制表示。
-初始化:初始化四个64位的缓冲区,用于存储计算结果。
-循环计算:将填充后的数据分为若干个消息块,每个消息块都经过四个循环处理(共64轮),根据一系列的位运算对缓冲区进行更新。
-摘要:将经过循环计算后的结果拼接起来,得到最终的128位摘要。
2.C语言实现:下面是一个简单的C语言版本的MD5加密算法实现。
该实现基于RFC1321中的伪代码。
首先,我们需要定义一些常数和辅助函数:```c#include <stdio.h>#include <stdint.h>//初始化缓冲区//将字符串转换为二进制消息块void stringToBinary(const char *str, uint32_t *msg, uint32_t len)uint32_t i;for (i = 0; i < len; i++)msg[i] = ((uint32_t)str[i * 4 + 3] << 24) ,((uint32_t)str[i * 4 + 2] << 16)((uint32_t)str[i * 4 + 1] << 8) , (uint32_t)str[i * 4];}//辅助函数:循环左移uint32_t rotateLeft(uint32_t x, uint32_t n)return (x << n) , (x >> (32 - n));```接下来,我们可以实现MD5算法的四个步骤:-填充:```c//填充消息块void padding(uint32_t *msg, uint32_t len)uint32_t midBitLength = len * 4 * 8;uint32_t padLen = ((midBitLength + 8) / 512 + 1) * 512 / 32 - len; // 计算填充长度msg[len] = 0x80; // 添加一个1uint32_t i;for (i = len + 1; i < len + padLen - 8; i++)msg[i] = 0x00; // 其余位置0}```-初始化:```c//初始化缓冲区void initBufferbuffer[1] = 0xEFCDAB89;buffer[2] = 0x98BADCFE;```-循环计算:```c//对每个消息块进行循环计算void processBlock(uint32_t *block)uint32_t a = buffer[0], b = buffer[1], c = buffer[2], d = buffer[3];uint32_t i;//主循环,共64轮for (i = 0; i < 64; i++)uint32_t f, g;//根据i的值选取相应的函数和常数if (i < 16)f=(b&c),((~b)&d);g=i;}else if (i < 32)f=(d&b),((~d)&c);g=(5*i+1)%16;}else if (i < 48)f=b^c^d;g=(3*i+5)%16;}elsef=c^(b,(~d));g=(7*i)%16;}uint32_t temp = d;d=c;c=b;b = b + rotateLeft(a + f + k[i] + block[g], s[i]);a = temp;}//更新缓冲区buffer[0] += a;buffer[1] += b;buffer[2] += c;buffer[3] += d;```-摘要:```c//生成最终的摘要void generateDigest(uint32_t *msg, uint32_t len)uint32_t i;for (i = 0; i < len / 16; i++)processBlock(msg + i * 16);}```最后,我们可以编写一个简单的主函数来使用这些函数:```cint main//待加密的字符串const char *str = "Hello, MD5!";uint32_t msgLength = (uint32_t)(strlen(str) + 1) / 4; //将字符串转换为二进制消息块uint32_t msg[msgLength];stringToBinary(str, msg, msgLength);//加密padding(msg, msgLength);initBuffer(;generateDigest(msg, msgLength + 64 / 4);//输出结果printf("MD5: ");for (int i = 0; i < 4; i++)printf("%02x", buffer[i] >> 0 & 0xff);printf("%02x", buffer[i] >> 8 & 0xff);printf("%02x", buffer[i] >> 16 & 0xff);printf("%02x", buffer[i] >> 24 & 0xff);}printf("\n");return 0;```这个简单的C语言版本的MD5加密算法实现了填充,初始化,循环计算和摘要等基本步骤。
C语言实现MD5校验
C语⾔实现MD5校验根据⽹上资料,整理验证C程序代码。
接⼝函数:1/******************************************************2*函数名称:Compute_data_md53*输⼊:data 校验数据⾸地址4 len 校验数据长度5 md5_str 字符串形式的MD5值6*输出:⽆7*功能:计算数据MD5值,并以字符串形式返回8*******************************************************/9int Compute_data_md5(unsigned char *data, unsigned int len,unsigned char *md5_str)View CodeMD5.h 的内容:1 #ifndef _MD5_H_2#define _MD5_H_34#define MD5_SIZE 165#define MD5_STR_LEN (MD5_SIZE * 2)67 typedef struct8 {9 unsigned int count[2];10 unsigned int state[4];11 unsigned char buffer[64];12 } MD5_CTX;1314#define F(x,y,z) ((x & y) | (~x & z))15#define G(x,y,z) ((x & z) | (y & ~z))16#define H(x,y,z) (x^y^z)17#define I(x,y,z) (y ^ (x | ~z))18#define ROTATE_LEFT(x,n) ((x << n) | (x >> (32-n)))1920#define FF(a,b,c,d,x,s,ac) \21 { \22 a += F(b,c,d) + x + ac; \23 a = ROTATE_LEFT(a,s); \24 a += b; \25 }26#define GG(a,b,c,d,x,s,ac) \27 { \28 a += G(b,c,d) + x + ac; \29 a = ROTATE_LEFT(a,s); \30 a += b; \31 }32#define HH(a,b,c,d,x,s,ac) \33 { \34 a += H(b,c,d) + x + ac; \35 a = ROTATE_LEFT(a,s); \36 a += b; \37 }38#define II(a,b,c,d,x,s,ac) \39 { \40 a += I(b,c,d) + x + ac; \41 a = ROTATE_LEFT(a,s); \42 a += b; \43 }44void MD5Init(MD5_CTX *context);45void MD5Update(MD5_CTX *context, unsigned char *input, unsigned int inputlen);46void MD5Final(MD5_CTX *context, unsigned char digest[16]);47void MD5Transform(unsigned int state[4], unsigned char block[64]);48void MD5Encode(unsigned char *output, unsigned int *input, unsigned int len);49void MD5Decode(unsigned int *output, unsigned char *input, unsigned int len);5051int Compute_data_md5(unsigned char *data, unsigned int len,unsigned char *md5_str);5253#endifView CodeMD5.c 的内容:1 #include "md5.h"2 #include <memory.h>3 #include <stdio.h>45 unsigned char PADDING[] =6 {70x80,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,80,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,90,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,100,0,0,0,0,0,0,0,0,0,0,0,0,0,0,011 };1213void MD5Init(MD5_CTX *context)14 {15 context->count[0] = 0;16 context->count[1] = 0;17 context->state[0] = 0x67452301;18 context->state[1] = 0xEFCDAB89;19 context->state[2] = 0x98BADCFE;20 context->state[3] = 0x10325476;21 }2223void MD5Update(MD5_CTX *context, unsigned char *input, unsigned int inputlen)24 {25 unsigned int i = 0;26 unsigned int index = 0;27 unsigned int partlen = 0;2829 index = (context->count[0] >> 3) & 0x3F;30 partlen = 64 - index;31 context->count[0] += inputlen << 3;3233if(context->count[0] < (inputlen << 3))34 context->count[1]++;35 context->count[1] += inputlen >> 29;3637if(inputlen >= partlen)38 {39 memcpy(&context->buffer[index], input,partlen);40 MD5Transform(context->state, context->buffer);4142for(i = partlen; i+64 <= inputlen; i+=64)43 MD5Transform(context->state, &input[i]);4445 index = 0;46 }47else48 {49 i = 0;50 }51 memcpy(&context->buffer[index], &input[i], inputlen-i);52 }5354void MD5Final(MD5_CTX *context, unsigned char digest[16])55 {56 unsigned int index = 0,padlen = 0;57 unsigned char bits[8];5859 index = (context->count[0] >> 3) & 0x3F;60 padlen = (index < 56)?(56-index):(120-index);61 MD5Encode(bits, context->count, 8);62 MD5Update(context, PADDING, padlen);63 MD5Update(context, bits, 8);64 MD5Encode(digest, context->state, 16);65 }6667void MD5Encode(unsigned char *output,unsigned int *input,unsigned int len)68 {69 unsigned int i = 0;70 unsigned int j = 0;7172while(j < len)73 {74 output[j] = input[i] & 0xFF;75 output[j+1] = (input[i] >> 8) & 0xFF;76 output[j+2] = (input[i] >> 16) & 0xFF;77 output[j+3] = (input[i] >> 24) & 0xFF;78 i++;79 j += 4;80 }81 }8283void MD5Decode(unsigned int *output, unsigned char *input, unsigned int len)84 {85 unsigned int i = 0;86 unsigned int j = 0;8788while(j < len)89 {90 output[i] = (input[j]) |91 (input[j+1] << 8) |92 (input[j+2] << 16) |93 (input[j+3] << 24);94 i++;95 j += 4;96 }97 }9899void MD5Transform(unsigned int state[4], unsigned char block[64]) 100 {101 unsigned int a = state[0];102 unsigned int b = state[1];103 unsigned int c = state[2];104 unsigned int d = state[3];105 unsigned int x[64];106107 MD5Decode(x,block,64);108109 FF(a, b, c, d, x[ 0], 7, 0xd76aa478); /* 1 */110 FF(d, a, b, c, x[ 1], 12, 0xe8c7b756); /* 2 */111 FF(c, d, a, b, x[ 2], 17, 0x242070db); /* 3 */112 FF(b, c, d, a, x[ 3], 22, 0xc1bdceee); /* 4 */113 FF(a, b, c, d, x[ 4], 7, 0xf57c0faf); /* 5 */114 FF(d, a, b, c, x[ 5], 12, 0x4787c62a); /* 6 */115 FF(c, d, a, b, x[ 6], 17, 0xa8304613); /* 7 */116 FF(b, c, d, a, x[ 7], 22, 0xfd469501); /* 8 */117 FF(a, b, c, d, x[ 8], 7, 0x698098d8); /* 9 */118 FF(d, a, b, c, x[ 9], 12, 0x8b44f7af); /* 10 */119 FF(c, d, a, b, x[10], 17, 0xffff5bb1); /* 11 */120 FF(b, c, d, a, x[11], 22, 0x895cd7be); /* 12 */121 FF(a, b, c, d, x[12], 7, 0x6b901122); /* 13 */122 FF(d, a, b, c, x[13], 12, 0xfd987193); /* 14 */123 FF(c, d, a, b, x[14], 17, 0xa679438e); /* 15 */124 FF(b, c, d, a, x[15], 22, 0x49b40821); /* 16 */125126/* Round 2 */127 GG(a, b, c, d, x[ 1], 5, 0xf61e2562); /* 17 */128 GG(d, a, b, c, x[ 6], 9, 0xc040b340); /* 18 */129 GG(c, d, a, b, x[11], 14, 0x265e5a51); /* 19 */130 GG(b, c, d, a, x[ 0], 20, 0xe9b6c7aa); /* 20 */131 GG(a, b, c, d, x[ 5], 5, 0xd62f105d); /* 21 */132 GG(d, a, b, c, x[10], 9, 0x2441453); /* 22 */133 GG(c, d, a, b, x[15], 14, 0xd8a1e681); /* 23 */134 GG(b, c, d, a, x[ 4], 20, 0xe7d3fbc8); /* 24 */135 GG(a, b, c, d, x[ 9], 5, 0x21e1cde6); /* 25 */136 GG(d, a, b, c, x[14], 9, 0xc33707d6); /* 26 */137 GG(c, d, a, b, x[ 3], 14, 0xf4d50d87); /* 27 */138 GG(b, c, d, a, x[ 8], 20, 0x455a14ed); /* 28 */139 GG(a, b, c, d, x[13], 5, 0xa9e3e905); /* 29 */140 GG(d, a, b, c, x[ 2], 9, 0xfcefa3f8); /* 30 */141 GG(c, d, a, b, x[ 7], 14, 0x676f02d9); /* 31 */142 GG(b, c, d, a, x[12], 20, 0x8d2a4c8a); /* 32 */143144/* Round 3 */145 HH(a, b, c, d, x[ 5], 4, 0xfffa3942); /* 33 */146 HH(d, a, b, c, x[ 8], 11, 0x8771f681); /* 34 */147 HH(c, d, a, b, x[11], 16, 0x6d9d6122); /* 35 */148 HH(b, c, d, a, x[14], 23, 0xfde5380c); /* 36 */149 HH(a, b, c, d, x[ 1], 4, 0xa4beea44); /* 37 */150 HH(d, a, b, c, x[ 4], 11, 0x4bdecfa9); /* 38 */151 HH(c, d, a, b, x[ 7], 16, 0xf6bb4b60); /* 39 */152 HH(b, c, d, a, x[10], 23, 0xbebfbc70); /* 40 */153 HH(a, b, c, d, x[13], 4, 0x289b7ec6); /* 41 */154 HH(d, a, b, c, x[ 0], 11, 0xeaa127fa); /* 42 */155 HH(c, d, a, b, x[ 3], 16, 0xd4ef3085); /* 43 */156 HH(b, c, d, a, x[ 6], 23, 0x4881d05); /* 44 */157 HH(a, b, c, d, x[ 9], 4, 0xd9d4d039); /* 45 */158 HH(d, a, b, c, x[12], 11, 0xe6db99e5); /* 46 */159 HH(c, d, a, b, x[15], 16, 0x1fa27cf8); /* 47 */160 HH(b, c, d, a, x[ 2], 23, 0xc4ac5665); /* 48 */161162/* Round 4 */163 II(a, b, c, d, x[ 0], 6, 0xf4292244); /* 49 */164 II(d, a, b, c, x[ 7], 10, 0x432aff97); /* 50 */165 II(c, d, a, b, x[14], 15, 0xab9423a7); /* 51 */166 II(b, c, d, a, x[ 5], 21, 0xfc93a039); /* 52 */167 II(a, b, c, d, x[12], 6, 0x655b59c3); /* 53 */168 II(d, a, b, c, x[ 3], 10, 0x8f0ccc92); /* 54 */169 II(c, d, a, b, x[10], 15, 0xffeff47d); /* 55 */170 II(b, c, d, a, x[ 1], 21, 0x85845dd1); /* 56 */171 II(a, b, c, d, x[ 8], 6, 0x6fa87e4f); /* 57 */172 II(d, a, b, c, x[15], 10, 0xfe2ce6e0); /* 58 */173 II(c, d, a, b, x[ 6], 15, 0xa3014314); /* 59 */174 II(b, c, d, a, x[13], 21, 0x4e0811a1); /* 60 */175 II(a, b, c, d, x[ 4], 6, 0xf7537e82); /* 61 */176 II(d, a, b, c, x[11], 10, 0xbd3af235); /* 62 */177 II(c, d, a, b, x[ 2], 15, 0x2ad7d2bb); /* 63 */178 II(b, c, d, a, x[ 9], 21, 0xeb86d391); /* 64 */179 state[0] += a;180 state[1] += b;181 state[2] += c;182 state[3] += d;183 }184185186/******************************************************187*函数名称:Compute_data_md5188*输⼊:data 校验数据⾸地址189 len 校验数据长度190 md5_str 字符串形式的MD5值191*输出:⽆192*功能:计算数据MD5值,并以字符串形式返回193*******************************************************/194int Compute_data_md5(unsigned char *data, unsigned int len,unsigned char *md5_str) 195 {196 unsigned char md5_value[MD5_SIZE];197 unsigned int i = 0;198199 MD5_CTX md5;200201 MD5Init(&md5);202 MD5Update(&md5, data, len);203 MD5Final(&md5, md5_value);204for(i = 0; i < MD5_SIZE; i++)205 {206 snprintf(md5_str + i*2, 2+1, "%02x", md5_value[i]);207 }208 md5_str[MD5_STR_LEN] = '\0'; // add end209 }210211#if 0212int Compute_file_md5(const char *file_path, char *md5_str)213 {214int i;215int fd;216int ret;217 unsigned char data[READ_DATA_SIZE];218 unsigned char md5_value[MD5_SIZE];219 MD5_CTX md5;220221 fd = open(file_path, O_RDONLY);222if (-1 == fd)223 {224 perror("open");225return -1;226 }227228// init md5229 MD5Init(&md5);230231while (1)232 {233 ret = read(fd, data, READ_DATA_SIZE);234if (-1 == ret)235 {236 perror("read");237return -1;238 }239240 MD5Update(&md5, data, ret);241242if (0 == ret || ret < READ_DATA_SIZE)243 {244break;245 }246 }247248 close(fd);249250 MD5Final(&md5, md5_value);251252for(i = 0; i < MD5_SIZE; i++)253 {254 snprintf(md5_str + i*2, 2+1, "%02x", md5_value[i]); 255 }256 md5_str[MD5_STR_LEN] = '\0'; // add end257258return0;259 }260#endifView Code。
C语言实现MD5算法
C语言实现MD5算法1.理解MD5算法原理:-MD5算法是一种常用的哈希函数,用于将任意长度的消息转换为固定长度的哈希值,通常为128位。
-MD5算法主要包括四个循环运算和四个非线性函数,通过迭代运算将输入的消息分块处理并输出最终的哈希值。
2.定义MD5算法所需的常量和函数:-定义一个64个元素的常数列表,用于在算法运算中使用。
-定义四个非线性的F,G,H,I函数,用于在循环运算中使用。
3.定义MD5算法所需的全局变量:-定义一个128位的缓冲区保存最终输出的哈希值。
-定义一个64位的计数指示器,记录输入消息的长度。
-定义四个32位的寄存器变量A、B、C、D,用于循环运算过程中保存中间结果。
4.实现MD5算法的主要函数:- 实现Padding函数,将输入消息按照MD5算法的规则进行填充,使其长度满足对512位的整数倍。
-实现FF,GG,HH,II四个函数,用于在每个循环运算中进行非线性转换。
-实现MD5算法的核心循环函数,将输入消息分块处理,并对每个分块进行四轮循环运算,更新寄存器变量的值。
-实现MD5算法的输出函数,将最终运算得到的寄存器变量的值按照一定顺序连接起来,得到最终的128位哈希值。
5.实现MD5算法的入口函数:- 在main函数中,读取输入消息,并调用MD5算法的相关函数,得到最终的哈希值。
-打印输出哈希值。
以下为C语言实现MD5算法的伪代码:```c//定义MD5算法所需的常量和函数const uint32_t s[64] = { ... }; // 常数表const uint32_t k[64] = { ... }; // F,G,H,I函数对应的逻辑常数uint32_t F(uint32_t x, uint32_t y, uint32_t z) { ... } // F 函数uint32_t G(uint32_t x, uint32_t y, uint32_t z) { ... } // G 函数uint32_t H(uint32_t x, uint32_t y, uint32_t z) { ... } // H 函数uint32_t I(uint32_t x, uint32_t y, uint32_t z) { ... } // I 函数//定义MD5算法所需的全局变量uint32_t buffer[4]; // 128位缓冲区uint64_t count = 0; // 输入消息的长度uint32_t A, B, C, D; // 寄存器变量// 实现Padding函数void padding(char* message) { ... }//实现FF,GG,HH,II四个函数void FF(...) { ... }void GG(...) { ... }void HH(...) { ... }void II(...) { ... }//实现MD5算法的核心循环函数void MD5Block(uint32_t* block) { ... }//实现MD5算法的输出函数void MD5Output(uint32_t* digest) { ... }//实现MD5算法的入口函数int maichar message[MAX_LENGTH];uint32_t digest[4];//读取输入消息gets(message);//填充输入消息padding(message);//分块处理for (uint32_t i = 0; i < count; i += 64)MD5Block(&message[i]);}//输出最终结果MD5Output(digest);//打印哈希值printf("%08x%08x%08x%08x", digest[0], digest[1], digest[2], digest[3]);return 0;```以上是一个简单的伪代码,实现了C语言下的MD5算法。
C++篇实现MD5算法
C++篇实现MD5算法1、头⽂件#ifndef MD5_H #define MD5_H #include <string> #include <fstream> /* Type define */ typedef unsigned char byte; typedef unsigned int uint32; using std::string; using std::ifstream; /* MD5 declaration. */ class MD5 { public: MD5(); MD5(const void* input, size_t length); MD5(const string& str); MD5(ifstream& in); void update(const void* input, size_t length); void update(const string& str); void update(ifstream& in); const byte* digest(); string toString(); void reset(); private: void update(const byte* input, size_t length); void final(); void transform(const byte block[64]); void encode(const uint32* input, byte* output, size_t length); void decode(const byte* input, uint32* output, size_t length); string bytesToHexString(const byte* input, size_t length); /* class uncopyable */ MD5(const MD5&); MD5& operator=(const MD5&); private: uint32 _state[4]; /* state (ABCD) */ uint32 _count[2]; /* number of bits, modulo 2^64 (low-order word first) */ byte _buffer[64]; /* input buffer */ byte _digest[16]; /* message digest */ bool _finished; /* calculate finished ? */ static const byte PADDING[64]; /* padding for calculate */ static const char HEX[16]; enum { BUFFER_SIZE = 1024 }; }; #endif /*MD5_H*/2、CPP⽂件#include "md5.h" using namespace std; /* Constants for MD5Transform routine. */ #define S11 7 #define S12 12 #define S13 17 #define S14 22 #define S21 5 #define S22 9 #define S23 14 #define S24 20 #define S31 4 #define S32 11 #define S33 16 #define S34 23 #define S41 6 #define S42 10 #define S43 15 #define S44 21 /* F, G, H and I are basic MD5 functions. */ #define F(x, y, z) (((x) & (y)) | ((~x) & (z))) #define G(x, y, z) (((x) & (z)) | ((y) & (~z))) #define H(x, y, z) ((x) ^ (y) ^ (z)) #define I(x, y, z) ((y) ^ ((x) | (~z))) /* ROTATE_LEFT rotates x left n bits. */ #define ROTATE_LEFT(x, n) (((x) << (n)) | ((x) >> (32-(n)))) /* FF, GG, HH, and II transformations for rounds 1, 2, 3, and 4. Rotation is separate from addition to prevent recomputation. */ #define FF(a, b, c, d, x, s, ac) { / (a) += F ((b), (c), (d)) + (x) + ac; / (a) = ROTATE_LEFT ((a), (s)); / (a) += (b); / } #define GG(a, b, c, d, x, s, ac) { / (a) += G ((b), (c), (d)) + (x) + ac; / (a) = ROTATE_LEFT ((a), (s)); / (a) += (b); / } #define HH(a, b, c, d, x, s, ac) { / (a) += H ((b), (c), (d)) + (x) + ac; / (a) = ROTATE_LEFT ((a), (s)); / (a) += (b); / } #define II(a, b, c, d, x, s, ac) { / (a) += I ((b), (c), (d)) + (x) + ac; / (a) =ROTATE_LEFT ((a), (s)); / (a) += (b); / } const byte MD5::PADDING[64] = { 0x80 }; const char MD5::HEX[16] = { '0', '1', '2', '3', '4', '5', '6', '7', '8', '9', 'a', 'b', 'c', 'd', 'e', 'f' }; /* Default construct. */ MD5::MD5() { reset(); } /* Construct a MD5 object with a input buffer. */ MD5::MD5(const void* input, size_t length) { reset(); update(input, length); } /* Construct a MD5 object with a string. */ MD5::MD5(const string& str) { reset(); update(str); } /* Construct a MD5 object with a file. */ MD5::MD5(ifstream& in) { reset(); update(in); } /* Return the message-digest */ const byte*MD5::digest() { if (!_finished) { _finished = true; final(); } return _digest; } /* Reset the calculate state */ void MD5::reset() { _finished = false; /* reset number of bits. */ _count[0] = _count[1] = 0; /* Load magic initialization constants. */ _state[0] = 0x67452301; _state[1] = 0xefcdab89; _state[2] = 0x98badcfe; _state[3] = 0x10325476; } /* Updating the context with a input buffer. */ void MD5::update(const void* input, size_t length) { update((const byte*)input, length); } /* Updating the context with a string. */ void MD5::update(const string& str) { update((const byte*)str.c_str(), str.length()); } /* Updating the context with a file. */ void MD5::update(ifstream& in) { if (!in) { return; } std::streamsize length; char buffer[BUFFER_SIZE]; while (!in.eof()) { in.read(buffer, BUFFER_SIZE); length = in.gcount(); if (length > 0) { update(buffer, length); } } in.close(); } /* MD5 block update operation. Continues an MD5 message-digest operation, processing another message block, and updating the context. */ void MD5::update(const byte* input, size_t length) { uint32 i, index, partLen; _finished = false; /* Compute number of bytes mod 64 */ index = (uint32)((_count[0] >> 3) & 0x3f); /* update number of bits */ if ((_count[0] += ((uint32)length << 3)) < ((uint32)length << 3)) { ++_count[1]; } _count[1] += ((uint32)length >> 29); partLen = 64 - index; /* transform as many times as possible. */ if (length >= partLen) { memcpy(&_buffer[index], input, partLen); transform(_buffer); for (i = partLen; i + 63 < length; i += 64) { transform(&input[i]); } index = 0; } else { i = 0; } /* Buffer remaining input */ memcpy(&_buffer[index], &input[i], length - i); } /* MD5 finalization. Ends an MD5 message-_digest operation, writing the the message _digest and zeroizing the context. */ void MD5::final() { byte bits[8]; uint32 oldState[4]; uint32 oldCount[2]; uint32 index, padLen; /* Save current state and count. */ memcpy(oldState, _state, 16); memcpy(oldCount, _count, 8); /* Save number of bits */ encode(_count, bits, 8); /* Pad out to 56 mod 64. */ index = (uint32)((_count[0] >> 3) & 0x3f); padLen = (index < 56) ? (56 - index) : (120 - index); update(PADDING, padLen); /* Append length (before padding) */ update(bits, 8); /* Store state in digest */ encode(_state, _digest, 16); /* Restore current state and count. */ memcpy(_state, oldState, 16); memcpy(_count, oldCount, 8); } /* MD5 basic transformation. Transforms _state based on block. */ void MD5::transform(const byte block[64]) { uint32 a = _state[0], b = _state[1], c = _state[2], d = _state[3], x[16]; decode(block, x, 64); /* Round 1 */ FF (a, b, c, d, x[ 0], S11, 0xd76aa478); /* 1 */ FF (d, a, b, c, x[ 1], S12, 0xe8c7b756); /* 2 */ FF (c, d, a, b, x[ 2], S13, 0x242070db); /* 3 */ FF (b, c, d, a, x[ 3], S14, 0xc1bdceee); /* 4 */ FF (a, b, c, d, x[ 4], S11, 0xf57c0faf); /* 5 */ FF (d, a, b, c, x[ 5], S12, 0x4787c62a); /* 6 */ FF (c, d, a, b, x[ 6], S13, 0xa8304613); /* 7 */ FF (b, c, d, a, x[ 7], S14,0xfd469501); /* 8 */ FF (a, b, c, d, x[ 8], S11, 0x698098d8); /* 9 */ FF (d, a, b, c, x[ 9], S12, 0x8b44f7af); /* 10 */ FF (c, d, a, b,x[10], S13, 0xffff5bb1); /* 11 */ FF (b, c, d, a, x[11], S14, 0x895cd7be); /* 12 */ FF (a, b, c, d, x[12], S11, 0x6b901122); /* 13 */ FF (d, a, b, c, x[13], S12, 0xfd987193); /* 14 */ FF (c, d, a, b, x[14], S13, 0xa679438e); /* 15 */ FF (b, c, d, a, x[15], S14, 0x49b40821); /* 16 */ /* Round 2 */ GG (a, b, c, d, x[ 1], S21, 0xf61e2562); /* 17 */ GG (d, a, b, c, x[ 6], S22, 0xc040b340); /* 18 */ GG (c, d, a, b, x[11], S23, 0x265e5a51); /* 19 */ GG (b, c, d, a, x[ 0], S24, 0xe9b6c7aa); /* 20 */ GG (a, b, c, d, x[ 5], S21, 0xd62f105d); /* 21 */ GG (d, a, b, c, x[10], S22, 0x2441453); /* 22 */ GG (c, d, a, b, x[15], S23, 0xd8a1e681); /* 23 */ GG (b, c, d, a, x[ 4], S24, 0xe7d3fbc8); /* 24 */ GG (a, b, c, d, x[ 9], S21, 0x21e1cde6); /* 25 */ GG (d, a, b, c, x[14], S22, 0xc33707d6); /* 26 */ GG (c, d, a, b, x[ 3], S23,0xf4d50d87); /* 27 */ GG (b, c, d, a, x[ 8], S24, 0x455a14ed); /* 28 */ GG (a, b, c, d, x[13], S21, 0xa9e3e905); /* 29 */ GG (d, a, b, c, x[ 2], S22, 0xfcefa3f8); /* 30 */ GG (c, d, a, b, x[ 7], S23, 0x676f02d9); /* 31 */ GG (b, c, d, a, x[12], S24, 0x8d2a4c8a); /* 32 */ /* Round 3 */ HH (a, b, c, d, x[ 5], S31, 0xfffa3942); /* 33 */ HH (d, a, b, c, x[ 8], S32, 0x8771f681); /* 34 */ HH (c, d, a, b, x[11], S33, 0x6d9d6122); /* 35 */ HH (b, c, d, a, x[14], S34, 0xfde5380c); /* 36 */ HH (a, b, c, d, x[ 1], S31, 0xa4beea44); /* 37 */ HH (d, a, b,c, x[ 4], S32, 0x4bdecfa9); /* 38 */ HH (c, d, a, b, x[ 7], S33, 0xf6bb4b60); /* 39 */ HH (b, c, d, a, x[10], S34, 0xbebfbc70); /* 40 */ HH (a, b, c, d, x[13], S31, 0x289b7ec6); /* 41 */ HH (d, a, b, c, x[ 0], S32, 0xeaa127fa); /* 42 */ HH (c, d, a, b, x[ 3], S33, 0xd4ef3085); /* 43 */ HH (b, c, d, a, x[ 6], S34, 0x4881d05); /* 44 */ HH (a, b, c, d, x[ 9], S31, 0xd9d4d039); /* 45 */ HH (d, a, b, c, x[12], S32,0xe6db99e5); /* 46 */ HH (c, d, a, b, x[15], S33, 0x1fa27cf8); /* 47 */ HH (b, c, d, a, x[ 2], S34, 0xc4ac5665); /* 48 */ /* Round 4 */ II (a, b, c, d, x[ 0], S41, 0xf4292244); /* 49 */ II (d, a, b, c, x[ 7], S42, 0x432aff97); /* 50 */ II (c, d, a, b, x[14], S43, 0xab9423a7); /* 51 */ II (b, c, d, a, x[ 5], S44, 0xfc93a039); /* 52 */ II (a, b, c, d, x[12], S41, 0x655b59c3); /* 53 */ II (d, a, b, c, x[ 3], S42,0x8f0ccc92); /* 54 */ II (c, d, a, b, x[10], S43, 0xffeff47d); /* 55 */ II (b, c, d, a, x[ 1], S44, 0x85845dd1); /* 56 */ II (a, b, c, d, x[ 8],S41, 0x6fa87e4f); /* 57 */ II (d, a, b, c, x[15], S42, 0xfe2ce6e0); /* 58 */ II (c, d, a, b, x[ 6], S43, 0xa3014314); /* 59 */ II (b, c, d, a, x[13], S44, 0x4e0811a1); /* 60 */ II (a, b, c, d, x[ 4], S41, 0xf7537e82); /* 61 */ II (d, a, b, c, x[11], S42, 0xbd3af235); /* 62 */ II (c, d, a, b, x[ 2], S43, 0x2ad7d2bb); /* 63 */ II (b, c, d, a, x[ 9], S44, 0xeb86d391); /* 64 */ _state[0] += a; _state[1] += b; _state[2] += c; _state[3] += d; } /* Encodes input (ulong) into output (byte). Assumes length is a multiple of 4. */ voidMD5::encode(const uint32* input, byte* output, size_t length) { for (size_t i = 0, j = 0; j < length; ++i, j += 4) { output[j]= (byte)(input[i] & 0xff); output[j + 1] = (byte)((input[i] >> 8) & 0xff); output[j + 2] = (byte)((input[i] >> 16) & 0xff); output[j + 3] = (byte)((input[i] >> 24) & 0xff); } } /* Decodes input (byte) into output (ulong). Assumes length is a multiple of 4. */ void MD5::decode(const byte* input, uint32* output, size_t length) { for (size_t i = 0, j = 0; j < length; ++i, j += 4) { output[i] = ((uint32)input[j]) | (((uint32)input[j + 1]) << 8) | (((uint32)input[j + 2]) << 16) | (((uint32)input[j + 3]) << 24); } } /* Convert byte array to hex string. */ stringMD5::bytesToHexString(const byte* input, size_t length) { string str; str.reserve(length << 1); for (size_t i = 0; i < length; ++i) { int t = input[i]; int a = t / 16; int b = t % 16; str.append(1, HEX[a]); str.append(1, HEX[b]); } return str; } /* Convert digest to string value */ string MD5::toString() { return bytesToHexString(digest(), 16); } #include <iostream> using namespace std; void PrintMD5(const string& str, MD5& md5) { cout << "MD5(/"" << str << "/") = " << md5.toString() << endl; } string FileDigest(const string& file) { ifstream in(file.c_str(), ios::binary); if (!in) { return ""; } MD5 md5; std::streamsize length; char buffer[1024]; while (!in.eof()) { in.read(buffer, 1024); length = in.gcount(); if (length > 0) { md5.update(buffer, length); } } in.close(); return md5.toString(); }3、测试⽂件#include "md5.h" int main(int argc, _TCHAR* argv[]) { //cout << MD5("abc").toString() << endl; //cout <<MD5(ifstream("D://test.txt")).toString() << endl; //cout << MD5(ifstream("D://test.exe", ios::binary)).toString() << endl; //cout << FileDigest("D://test.exe") << endl; MD5 md5; md5.update(""); PrintMD5("", md5); md5.update("a"); PrintMD5("a", md5); md5.update("bc"); PrintMD5("abc", md5); md5.update("defghijklmnopqrstuvwxyz"); PrintMD5("abcdefghijklmnopqrstuvwxyz", md5); md5.reset(); md5.update("message digest"); PrintMD5("message digest",md5); //md5.reset(); //md5.update(ifstream("D://test.txt")); //PrintMD5("D://test.txt", md5); return 0; }。
实验四、杂凑函数MD5
实验四、实验报告
姓名:学号:实验日期:
一、实验名称: 杂凑函数MD5
二、实验目的
1、掌握MD5算法的工作原理。
2、了解字典攻击的工作原理。
三、实验基本方法
1、用C或C++语言编写一个MD5算法;
2、编程实现对一个字符串计算其MD5杂凑值;
3、编程实现利用字典攻击的MD5反算法。
*
说明:
1、MD5算法可以自编,也可以网上下载现成算法。
四、实验步骤
1、用C或C++语言编写一个MD5算法。
2、编程实现对一个字符串计算其MD5杂凑值。
3、对一个字符计算并记录其杂凑值,改变1位明文信息观察并记录MD5的输出。
重
复3次。
4、编程实现利用字典攻击的MD5反算法。
5、假设字符串是全数字的口令,利用上面程序破解口令。
*
五、实验结果
1、程序设计的思想,及程序关键原代码。
2、分析明文信息改变后的MD5输出变化。
3、报告字典攻击数字口令的结果。
*
4、简述MD5的用途。
打*号的是选做。
MD5算法的C语言实现
MD5算法的C语⾔实现11 #include <stdio.h>2 #include <stdint.h>3 #include <stdlib.h>4 #include <string.h>5 #include <sys/types.h>6 #include "md5.h"78 #ifdef __cplusplus9extern"C" {10#endif1112#define ROTATELEFT(value, bits) (((value) << (bits)) | ((value) >> (32 - (bits))))1314#define TO_HEX_FMT_L "%02x"15#define TO_HEX_FMT_U "%02X"1617/**18 * @desc: convert message and mes_bkp string into integer array and store them in w19*/20static void md5_process_part1(uint32_t *w, const char *message, uint32_t *pos, uint32_t mes_len, const unsigned char *mes_bkp)21 {22 uint32_t i; // used in for loop2324for(i = 0; i <= 15; i++)25 {26 int32_t count = 0;27while(*pos < mes_len && count <= 24)28 {29 w[i] += (((uint32_t)message[*pos]) << count);30 (*pos)++;31 count += 8;32 }33while(count <= 24)34 {35 w[i] += (((uint32_t)mes_bkp[*pos - mes_len]) << count);36 (*pos)++;37 count += 8;38 }39 }40 }4142/**43 * @desc: start encryption based on w44*/45static void md5_process_part2(uint32_t abcd[4], uint32_t *w, const uint32_t k[64], const uint32_t s[64])46 {47 uint32_t i; // used in for loop4849 uint32_t a = abcd[0];50 uint32_t b = abcd[1];51 uint32_t c = abcd[2];52 uint32_t d = abcd[3];53 uint32_t f = 0;54 uint32_t g = 0;5556for(i = 0; i < 64; i++)57 {58if(i >= 0 && i <= 15)59 {60 f = (b & c) | ((~b) & d);61 g = i;62 }else if(i >= 16 && i <= 31)63 {64 f = (d & b) | ((~d) & c);65 g = (5 * i + 1) % 16;66 }else if(i >= 32 && i <= 47)67 {68 f = b ^ c ^ d;69 g = (3 * i + 5) % 16;70 }else if(i >= 48 && i <= 63)71 {72 f = c ^ (b | (~d));73 g = (7 * i) % 16;74 }75 uint32_t temp = d;76 d = c;77 c = b;78 b = ROTATELEFT((a + f + k[i] + w[g]), s[i]) + b;79 a = temp;80 }8182 abcd[0] += a;83 abcd[1] += b;84 abcd[2] += c;85 abcd[3] += d;86 }8788/**89 * @desc: format the output, convert numbers to hexdecimal string and store them in result90*/91static void format_output(char *result, size_t size, uint32_t *abcd, uint32_t flag)92 {93 uint32_t i; // used in for loop9495 memset(result, 0, size);9697 uint32_t ptr = 0;98for(i = 0; i < 4; i++)99 {100 ptr += snprintf(result + ptr, size - ptr, (flag == 0)?TO_HEX_FMT_U:TO_HEX_FMT_L, (abcd[i] & 0x000000FF));101 ptr += snprintf(result + ptr, size - ptr, (flag == 0)?TO_HEX_FMT_U:TO_HEX_FMT_L, (abcd[i] & 0x0000FF00) >> 8 ); 102 ptr += snprintf(result + ptr, size - ptr, (flag == 0)?TO_HEX_FMT_U:TO_HEX_FMT_L, (abcd[i] & 0x00FF0000) >> 16); 103 ptr += snprintf(result + ptr, size - ptr, (flag == 0)?TO_HEX_FMT_U:TO_HEX_FMT_L, (abcd[i] & 0xFF000000) >> 24); 104 }105 }106107108/**109 * @input: result -- store the calculation result110 * size -- size of result. Make sure it's at least 33111 * since the result is a 32-byte hexdecimal string.112 * message-- string to be encrypted113 * flag -- 0 means upper case output, 1 means lower case output114 * @return: 0 -- success115 * 1 -- result size less than 33116 * 2 -- calloc failed117*/118 int32_t cal_md5(char *result, size_t size, const char *message, uint32_t flag){119if (result == NULL || size < 33)120 {121return1;122 }123124 uint32_t *w = (uint32_t *)calloc(16, sizeof(uint32_t));125if(w == NULL)126 {127return2;128 }129130 uint32_t i; // used in for loop131132 uint32_t mes_len = strlen(message);133 uint32_t looptimes = (mes_len + 8) / 64 + 1;134 uint32_t abcd[] = {0x67452301, 0xEFCDAB89, 0x98BADCFE, 0x10325476};135136const uint32_t k[]={1370xd76aa478,0xe8c7b756,0x242070db,0xc1bdceee,1380xf57c0faf,0x4787c62a,0xa8304613,0xfd469501,0x698098d8,1390x8b44f7af,0xffff5bb1,0x895cd7be,0x6b901122,0xfd987193,1400xa679438e,0x49b40821,0xf61e2562,0xc040b340,0x265e5a51,1410xe9b6c7aa,0xd62f105d,0x02441453,0xd8a1e681,0xe7d3fbc8,1420x21e1cde6,0xc33707d6,0xf4d50d87,0x455a14ed,0xa9e3e905,1430xfcefa3f8,0x676f02d9,0x8d2a4c8a,0xfffa3942,0x8771f681,1440x6d9d6122,0xfde5380c,0xa4beea44,0x4bdecfa9,0xf6bb4b60,1450xbebfbc70,0x289b7ec6,0xeaa127fa,0xd4ef3085,0x04881d05,1460xd9d4d039,0xe6db99e5,0x1fa27cf8,0xc4ac5665,0xf4292244,1470x432aff97,0xab9423a7,0xfc93a039,0x655b59c3,0x8f0ccc92,1480xffeff47d,0x85845dd1,0x6fa87e4f,0xfe2ce6e0,0xa3014314,1490x4e0811a1,0xf7537e82,0xbd3af235,0x2ad7d2bb,0xeb86d391150 };151152const uint32_t s[]={1537,12,17,22,7,12,17,22,7,12,17,22,7,15412,17,22,5,9,14,20,5,9,14,20,5,9,14,20,5,9,14,20,1554,11,16,23,4,11,16,23,4,11,16,23,4,11,16,23,6,10,15615,21,6,10,15,21,6,10,15,21,6,10,15,21157 };158159 uint32_t pos = 0; // position pointer for message string160 uint32_t bkp_len = 64 * looptimes - mes_len;161 unsigned char *bkp_mes = (unsigned char *)calloc(1, bkp_len);162if(bkp_mes == NULL)163 {164 free(w);165return2;166 }167168 bkp_mes[0] = (unsigned char)(0x80);169 uint64_t mes_bit_len = ((uint64_t)mes_len) * 8;170for(i = 0; i < 8; i++)171 {172 bkp_mes[bkp_len-i-1] = (unsigned char)((mes_bit_len & (0x00000000000000FF << (8 * (7 - i)))) >> (8 * (7 - i))); 173 }174175for(i = 0; i < looptimes; i++)176 {177 memset(w, 0, 16 * sizeof(uint32_t));178179 md5_process_part1(w, message, &pos, mes_len, bkp_mes); // compute w180181 md5_process_part2(abcd, w, k, s); // calculate md5 and store the result in abcd182 }183184 free(w);185 free(bkp_mes);186187 format_output(result, size, abcd, flag);188189return0;190 }191192 #ifdef __cplusplus193 }194#endif21/**2 * @author Horst Xu3 * @date 2015-07-104 * @contact 271021733@5*/6 #ifndef __MD5_H__7#define __MD5_H__89 #include <stdint.h>10 #include <stddef.h>1112 #ifdef __cplusplus13extern"C" {14#endif1516/**17 * @input: result -- store the calculation result18 * size -- size of result. Make sure it's at least 3319 * since the result is a 32-byte hexdecimal string.20 * message-- string to be encrypted21 * flag -- 0 means upper case output, 1 means lower case output22 * @return: 0 -- success23 * 1 -- result size less than 3324 * 2 -- calloc failed25*/26 int32_t cal_md5(char *result, size_t size, const char *message, uint32_t flag);2728 #ifdef __cplusplus29 }30#endif3132#endif//__MD5_H__31 #include <stdio.h>2 #include <stdint.h>3 #include <string.h>45 #include "md5.h"67 int32_t main(void)8 {9char result[41];10 int32_t ret = -1;1112//test 113 ret = cal_md5(result, sizeof(result), "abcdegfhijklmnopqrstuvwxyzabcdefghijklmnopqrstuvwxyz123", 1); 14if(ret == 0 && 0 == strncmp(result, "dd2fc541b65e2202d55beae0ecaf6528", strlen(result)))15 {16 printf("test 1 successful!\n");17 }else18 {19 printf("test 1 failed!\n");20 }2122//test 223 ret = cal_md5(result, sizeof(result), "abcdegfhijklmnopqrstuvwxyzabcdefghijklmnopqrstuvwxyz1234", 0); 24if(ret == 0 && 0 == strncmp(result, "27FF2E8344E7E3F36F9C7E18D0EC82DF", strlen(result)))25 {26 printf("test 2 successful!\n");27 }else28 {29 printf("test 2 failed!\n");30 }3132//test 333 ret = cal_md5(result, sizeof(result), "abcdegfhijklmnopqrstuvwxyzabcdefghijklmnopqrstuvwxyz12345", 0); 34if(ret == 0 && 0 == strncmp(result, "7A7B1279343946E3A5949BEA1B3BF8AF", strlen(result)))35 {36 printf("test 3 successful!\n");37 }else38 {39 printf("test 3 failed!\n");40 }4142//test 443 ret = cal_md5(result, sizeof(result), "", 1);44if(ret == 0 && 0 == strncmp(result, "d41d8cd98f00b204e9800998ecf8427e", strlen(result)))45 {46 printf("test 4 successful!\n");47 }else48 {49 printf("test 4 failed!\n");50 }5152return0;53 }。
- 1、下载文档前请自行甄别文档内容的完整性,平台不提供额外的编辑、内容补充、找答案等附加服务。
- 2、"仅部分预览"的文档,不可在线预览部分如存在完整性等问题,可反馈申请退款(可完整预览的文档不适用该条件!)。
- 3、如文档侵犯您的权益,请联系客服反馈,我们会尽快为您处理(人工客服工作时间:9:00-18:30)。
一、实验目的验证MD5加密算法,了解加密过程。
二、实验环境软件工具:Visual C++ 6.0操作系统:windows xp三、实验思想对MD5算法简要的叙述可以为:MD5以512位分组来处理输入的信息,且每一分组又被划分为16个32位子分组,经过了一系列的处理后,算法的输出由四个32位分组组成,将这四个32位分组级联后将生成一个128位散列值。
在MD5算法中,首先需要对信息进行填充,使其位长对512求余的结果等于448。
因此,信息的位长(Bits Length)将被扩展至N*512+448,N为一个非负整数,N可以是零。
填充的方法如下,在信息的后面填充一个1和无数个0,直到满足上面的条件时才停止用0对信息的填充。
然后,在这个结果后面附加一个以64位二进制表示的填充前信息长度。
经过这两步的处理,现在的信息的位长=N*512+448+64=(N+1)*512,即长度恰好是512的整数倍。
这样做的原因是为满足后面处理中对信息长度的要求。
MD5中有四个32位被称作链接变量(Chaining Variable)的整数参数,他们分别为:A=0x67452301,B=0xefcdab89,C=0x98badcfe,D=0x10325476。
当设置好这四个链接变量后,就开始进入算法的四轮循环运算。
循环的次数是信息中512位信息分组的数目。
将上面四个链接变量复制到另外四个变量中:A到a,B到b,C到c,D到d。
主循环有四轮(MD4只有三轮),每轮循环都很相似。
第一轮进行16次操作。
每次操作对a、b、c和d中的其中三个作一次非线性函数运算,然后将所得结果加上第四个变量,文本的一个子分组和一个常数。
再将所得结果向左环移一个不定的数,并加上a、b、c或d中之一。
最后用该结果取代a、b、c或d中之一。
四、实验数据(源代码)#include "global.h"#include "md5.h"/* Constants for MD5Transform routine.*/#define S11 7#define S12 12#define S13 17#define S14 22#define S21 5#define S22 9#define S23 14#define S24 20#define S31 4#define S32 11#define S33 16#define S34 23#define S41 6#define S42 10#define S43 15#define S44 21static void MD5Transform PROTO_LIST ((UINT4 [4], unsigned char [64])); static void Encode PROTO_LIST((unsigned char *, UINT4 *, unsigned int));static void Decode PROTO_LIST((UINT4 *, unsigned char *, unsigned int));static void MD5_memcpy PROTO_LIST ((POINTER, POINTER, unsigned int)); static void MD5_memset PROTO_LIST ((POINTER, int, unsigned int));static unsigned char PADDING[64] = {0x80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0};/* F, G, H and I are basic MD5 functions.*/#define F(x, y, z) (((x) & (y)) | ((~x) & (z)))#define G(x, y, z) (((x) & (z)) | ((y) & (~z)))#define H(x, y, z) ((x) ^ (y) ^ (z))#define I(x, y, z) ((y) ^ ((x) | (~z)))/* ROTATE_LEFT rotates x left n bits.*/#define ROTATE_LEFT(x, n) (((x) << (n)) | ((x) >> (32-(n))))/* FF, GG, HH, and II transformations for rounds 1, 2, 3, and 4.Rotation is separate from addition to prevent recomputation.*/#define FF(a, b, c, d, x, s, ac) { \(a) += F ((b), (c), (d)) + (x) + (UINT4)(ac); \(a) = ROTATE_LEFT ((a), (s)); \(a) += (b); \}#define GG(a, b, c, d, x, s, ac) { \(a) += G ((b), (c), (d)) + (x) + (UINT4)(ac); \(a) = ROTATE_LEFT ((a), (s)); \(a) += (b); \}#define HH(a, b, c, d, x, s, ac) { \(a) += H ((b), (c), (d)) + (x) + (UINT4)(ac); \(a) = ROTATE_LEFT ((a), (s)); \(a) += (b); \}#define II(a, b, c, d, x, s, ac) { \(a) += I ((b), (c), (d)) + (x) + (UINT4)(ac); \(a) = ROTATE_LEFT ((a), (s)); \(a) += (b); \}/* MD5 initialization. Begins an MD5 operation, writing a new context.*/void MD5Init (context)MD5_CTX *context; /* context */ {context->count[0] = context->count[1] = 0;/* Load magic initialization constants.*/context->state[0] = 0x67452301;context->state[1] = 0xefcdab89;context->state[2] = 0x98badcfe;context->state[3] = 0x10325476;}/* MD5 block update operation. Continues an MD5 message-digestoperation, processing another message block, and updating thecontext.*/void MD5Update (context, input, inputLen)MD5_CTX *context; /* context */ unsigned char *input; /* input block */ unsigned int inputLen; /* length of input block */{unsigned int i, index, partLen;/* Compute number of bytes mod 64 */index = (unsigned int)((context->count[0] >> 3) & 0x3F);/* Update number of bits */if ((context->count[0] += ((UINT4)inputLen << 3))< ((UINT4)inputLen << 3))context->count[1]++;context->count[1] += ((UINT4)inputLen >> 29);partLen = 64 - index;/* Transform as many times as possible.*/if (inputLen >= partLen) {MD5_memcpy((POINTER)&context->buffer[index], (POINTER)input, partLen);MD5Transform (context->state, context->buffer);for (i = partLen; i + 63 < inputLen; i += 64)MD5Transform (context->state, &input[i]);index = 0;}elsei = 0;/* Buffer remaining input */MD5_memcpy((POINTER)&context->buffer[index], (POINTER)&input[i],inputLen-i);}/* MD5 finalization. Ends an MD5 message-digest operation, writing thethe message digest and zeroizing the context.*/void MD5Final (digest, context)unsigned char digest[16]; /* message digest */MD5_CTX *context; /* context */ {unsigned char bits[8];unsigned int index, padLen;/* Save number of bits */Encode (bits, context->count, 8);/* Pad out to 56 mod 64.*/index = (unsigned int)((context->count[0] >> 3) & 0x3f);padLen = (index < 56) ? (56 - index) : (120 - index);MD5Update (context, PADDING, padLen);/* Append length (before padding) */MD5Update (context, bits, 8);/* Store state in digest */Encode (digest, context->state, 16);/* Zeroize sensitive information.*/MD5_memset ((POINTER)context, 0, sizeof (*context));}/* MD5 basic transformation. Transforms state based on block.*/static void MD5Transform (state, block)UINT4 state[4];unsigned char block[64];{UINT4 a = state[0], b = state[1], c = state[2], d = state[3], x[16];Decode (x, block, 64);/* Round 1 */FF (a, b, c, d, x[ 0], S11, 0xd76aa478); /* 1 */FF (d, a, b, c, x[ 1], S12, 0xe8c7b756); /* 2 */FF (c, d, a, b, x[ 2], S13, 0x242070db); /* 3 */FF (b, c, d, a, x[ 3], S14, 0xc1bdceee); /* 4 */FF (a, b, c, d, x[ 4], S11, 0xf57c0faf); /* 5 */FF (c, d, a, b, x[ 6], S13, 0xa8304613); /* 7 */ FF (b, c, d, a, x[ 7], S14, 0xfd469501); /* 8 */ FF (a, b, c, d, x[ 8], S11, 0x698098d8); /* 9 */ FF (d, a, b, c, x[ 9], S12, 0x8b44f7af); /* 10 */ FF (c, d, a, b, x[10], S13, 0xffff5bb1); /* 11 */ FF (b, c, d, a, x[11], S14, 0x895cd7be); /* 12 */ FF (a, b, c, d, x[12], S11, 0x6b901122); /* 13 */ FF (d, a, b, c, x[13], S12, 0xfd987193); /* 14 */ FF (c, d, a, b, x[14], S13, 0xa679438e); /* 15 */ FF (b, c, d, a, x[15], S14, 0x49b40821); /* 16 */ /* Round 2 */GG (a, b, c, d, x[ 1], S21, 0xf61e2562); /* 17 */ GG (d, a, b, c, x[ 6], S22, 0xc040b340); /* 18 */ GG (c, d, a, b, x[11], S23, 0x265e5a51); /* 19 */ GG (b, c, d, a, x[ 0], S24, 0xe9b6c7aa); /* 20 */ GG (a, b, c, d, x[ 5], S21, 0xd62f105d); /* 21 */ GG (d, a, b, c, x[10], S22, 0x2441453); /* 22 */ GG (c, d, a, b, x[15], S23, 0xd8a1e681); /* 23 */ GG (b, c, d, a, x[ 4], S24, 0xe7d3fbc8); /* 24 */ GG (a, b, c, d, x[ 9], S21, 0x21e1cde6); /* 25 */ GG (d, a, b, c, x[14], S22, 0xc33707d6); /* 26 */ GG (c, d, a, b, x[ 3], S23, 0xf4d50d87); /* 27 */ GG (b, c, d, a, x[ 8], S24, 0x455a14ed); /* 28 */ GG (a, b, c, d, x[13], S21, 0xa9e3e905); /* 29 */ GG (d, a, b, c, x[ 2], S22, 0xfcefa3f8); /* 30 */ GG (c, d, a, b, x[ 7], S23, 0x676f02d9); /* 31 */ GG (b, c, d, a, x[12], S24, 0x8d2a4c8a); /* 32 */ /* Round 3 */HH (a, b, c, d, x[ 5], S31, 0xfffa3942); /* 33 */ HH (d, a, b, c, x[ 8], S32, 0x8771f681); /* 34 */ HH (c, d, a, b, x[11], S33, 0x6d9d6122); /* 35 */ HH (b, c, d, a, x[14], S34, 0xfde5380c); /* 36 */ HH (a, b, c, d, x[ 1], S31, 0xa4beea44); /* 37 */ HH (d, a, b, c, x[ 4], S32, 0x4bdecfa9); /* 38 */ HH (c, d, a, b, x[ 7], S33, 0xf6bb4b60); /* 39 */ HH (b, c, d, a, x[10], S34, 0xbebfbc70); /* 40 */ HH (a, b, c, d, x[13], S31, 0x289b7ec6); /* 41 */ HH (d, a, b, c, x[ 0], S32, 0xeaa127fa); /* 42 */ HH (c, d, a, b, x[ 3], S33, 0xd4ef3085); /* 43 */ HH (b, c, d, a, x[ 6], S34, 0x4881d05); /* 44 */ HH (a, b, c, d, x[ 9], S31, 0xd9d4d039); /* 45 */ HH (d, a, b, c, x[12], S32, 0xe6db99e5); /* 46 */ HH (c, d, a, b, x[15], S33, 0x1fa27cf8); /* 47 *//* Round 4 */II (a, b, c, d, x[ 0], S41, 0xf4292244); /* 49 */II (d, a, b, c, x[ 7], S42, 0x432aff97); /* 50 */II (c, d, a, b, x[14], S43, 0xab9423a7); /* 51 */II (b, c, d, a, x[ 5], S44, 0xfc93a039); /* 52 */II (a, b, c, d, x[12], S41, 0x655b59c3); /* 53 */II (d, a, b, c, x[ 3], S42, 0x8f0ccc92); /* 54 */II (c, d, a, b, x[10], S43, 0xffeff47d); /* 55 */II (b, c, d, a, x[ 1], S44, 0x85845dd1); /* 56 */II (a, b, c, d, x[ 8], S41, 0x6fa87e4f); /* 57 */II (d, a, b, c, x[15], S42, 0xfe2ce6e0); /* 58 */II (c, d, a, b, x[ 6], S43, 0xa3014314); /* 59 */II (b, c, d, a, x[13], S44, 0x4e0811a1); /* 60 */II (a, b, c, d, x[ 4], S41, 0xf7537e82); /* 61 */II (d, a, b, c, x[11], S42, 0xbd3af235); /* 62 */II (c, d, a, b, x[ 2], S43, 0x2ad7d2bb); /* 63 */II (b, c, d, a, x[ 9], S44, 0xeb86d391); /* 64 */state[0] += a;state[1] += b;state[2] += c;state[3] += d;/* Zeroize sensitive information.*/MD5_memset ((POINTER)x, 0, sizeof (x));}/* Encodes input (UINT4) into output (unsigned char). Assumes len is a multiple of 4.*/static void Encode (output, input, len)unsigned char *output;UINT4 *input;unsigned int len;{unsigned int i, j;for (i = 0, j = 0; j < len; i++, j += 4) {output[j] = (unsigned char)(input[i] & 0xff);output[j+1] = (unsigned char)((input[i] >> 8) & 0xff);output[j+2] = (unsigned char)((input[i] >> 16) & 0xff);output[j+3] = (unsigned char)((input[i] >> 24) & 0xff);}}/* Decodes input (unsigned char) into output (UINT4). Assumes len is a multiple of 4.*/static void Decode (output, input, len)UINT4 *output;unsigned char *input;unsigned int len;{unsigned int i, j;for (i = 0, j = 0; j < len; i++, j += 4)output[i] = ((UINT4)input[j]) | (((UINT4)input[j+1]) << 8) |(((UINT4)input[j+2]) << 16) | (((UINT4)input[j+3]) << 24);}/* Note: Replace "for loop" with standard memcpy if possible.*/static void MD5_memcpy (output, input, len)POINTER output;POINTER input;unsigned int len;{unsigned int i;for (i = 0; i < len; i++)output[i] = input[i];}/* Note: Replace "for loop" with standard memset if possible.*/static void MD5_memset (output, value, len)POINTER output;int value;unsigned int len;{unsigned int i;for (i = 0; i < len; i++)((char *)output)[i] = (char)value;}五、实验心得通过这次实验了解到哈希函数的特点:压缩性、容易计算、抗修改性、弱抗碰撞、强抗碰撞。