为了正常的体验网站,请在浏览器设置里面开启Javascript功能!

MD5算法实现

2017-09-19 10页 doc 40KB 16阅读

用户头像

is_713593

暂无简介

举报
MD5算法实现md5加密算法c实现 md5的介绍的文章网上很多,关于md5的来历,用途什么的这里就不再介绍了。这里主要介绍代码。代码明白了就什么都明白了。 //////////////////////////////////////////////////////////////////// /*md5.h*/ #ifndef_MD5_H_ #define_MD5_H_ #defineR_memset(x,y,z)memset(x,y,z) #defineR_memcpy(x,y,z)memcpy(x,y,z) #defineR_memcm...
MD5算法实现
md5加密算法c实现 md5的介绍的文章网上很多,关于md5的来历,用途什么的这里就不再介绍了。这里主要介绍代码。代码明白了就什么都明白了。 //////////////////////////////////////////////////////////////////// /*md5.h*/ #ifndef_MD5_H_ #define_MD5_H_ #defineR_memset(x,y,z)memset(x,y,z) #defineR_memcpy(x,y,z)memcpy(x,y,z) #defineR_memcmp(x,y,z)memcmp(x,y,z) typedefunsignedlongUINT4; typedefunsignedchar*POINTER; /*MD5context.*/ typedefstruct{ /*state(ABCD)*/ /*四个32bits数,用于存放最终计算得到的消息摘要。当消息长度〉512bits时,也用于存放每个512bits的中间结果*/ UINT4state[4]; /*numberofbits,modulo2^64(lsbfirst)*/ /*存储原始信息的bits数长度,不包括填充的bits,最长为2^64bits,因为2^64是一个64位数的最大值*/ UINT4count[2]; /*inputbuffer*/ /*存放输入的信息的缓冲区,512bits*/ unsignedcharbuffer[64]; }MD5_CTX; voidMD5Init(MD5_CTX*); voidMD5Update(MD5_CTX*,unsignedchar*,unsignedint); voidMD5Final(unsignedchar[16],MD5_CTX*); #endif/*_MD5_H_*/ /////////////////////////////////////////////////////////////////////////// /*md5.cpp*/ #include"stdafx.h" /*ConstantsforMD5Transformroutine.*/ /*md5转换用到的常量,算法本身规定的*/ #defineS117 #defineS1212 #defineS1317 #defineS1422 #defineS215 #defineS229 #defineS2314 #defineS2420 #defineS314 #defineS3211 #defineS3316 #defineS3423 #defineS416 #defineS4210 #defineS4315 #defineS4421 staticvoidMD5Transform(UINT4[4],unsignedchar[64]); staticvoidEncode(unsignedchar*,UINT4*,unsignedint); staticvoidDecode(UINT4*,unsignedchar*,unsignedint); /* 用于bits填充的缓冲区,为什么要64个字节呢?因为当欲加密的信息的bits数被512除其余数为448时, 需要填充的bits的最大值为512=64*8。 */ staticunsignedcharPADDING[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 }; /* 接下来的这几个宏定义是md5算法规定的,就是对信息进行md5加密都要做的运算。 据说有经验的高手跟踪程序时根据这几个特殊的操作就可以断定是不是用的md5 */ /*F,G,HandIarebasicMD5functions. */ #defineF(x,y,z)(((x)&(y))|((~x)&(z))) #defineG(x,y,z)(((x)&(z))|((y)&(~z))) #defineH(x,y,z)((x)^(y)^(z)) #defineI(x,y,z)((y)^((x)|(~z))) /*ROTATE_LEFTrotatesxleftnbits. */ #defineROTATE_LEFT(x,n)(((x)<<(n))|((x)>>(32-(n)))) /*FF,GG,HH,andIItransformationsforrounds1,2,3,and4. Rotationisseparatefromadditiontopreventrecomputation. */ #defineFF(a,b,c,d,x,s,ac){/ (a)+=F((b),(c),(d))+(x)+(UINT4)(ac);/ (a)=ROTATE_LEFT((a),(s));/ (a)+=(b);/ } #defineGG(a,b,c,d,x,s,ac){/ (a)+=G((b),(c),(d))+(x)+(UINT4)(ac);/ (a)=ROTATE_LEFT((a),(s));/ (a)+=(b);/ } #defineHH(a,b,c,d,x,s,ac){/ (a)+=H((b),(c),(d))+(x)+(UINT4)(ac);/ (a)=ROTATE_LEFT((a),(s));/ (a)+=(b);/ } #defineII(a,b,c,d,x,s,ac){/ (a)+=I((b),(c),(d))+(x)+(UINT4)(ac);/ (a)=ROTATE_LEFT((a),(s));/ (a)+=(b);/ } /*MD5initialization.BeginsanMD5operation,writinganewcontext.*/ /*初始化md5的结构*/ voidMD5Init(MD5_CTX*context) { /*将当前的有效信息的长度设成0,这个很简单,还没有有效信息,长度当然是0了*/ context->count[0]=context->count[1]=0; /*Loadmagicinitializationconstants.*/ /*初始化链接变量,算法这样,这个没法解释了*/ context->state[0]=0x67452301; context->state[1]=0xefcdab89; context->state[2]=0x98badcfe; context->state[3]=0x10325476; } /*MD5blockupdateoperation.ContinuesanMD5message-digest operation,processinganothermessageblock,andupdatingthe context.*/ /*将与加密的信息传递给md5结构,可以多次调用 context:初始化过了的md5结构 input:欲加密的信息,可以任意长 inputLen:指定input的长度 */ voidMD5Update(MD5_CTX*context,unsignedchar*input,unsignedintinputLen) { unsignedinti,index,partLen; /*Computenumberofbytesmod64*/ /*计算已有信息的bits长度的字节数的模64,64bytes=512bits。 用于判断已有信息加上当前传过来的信息的总长度能不能达到512bits, 如果能够达到则对凑够的512bits进行一次处理*/ index=(unsignedint)((context->count[0]>>3)&0x3F); /*Updatenumberofbits*//*更新已有信息的bits长度*/ if((context->count[0]+=((UINT4)inputLen<<3))<((UINT4)inputLen<<3)) context->count[1]++; context->count[1]+=((UINT4)inputLen>>29); /*计算已有的字节数长度还差多少字节可以凑成64的整倍数*/ partLen=64-index; /*Transformasmanytimesaspossible. */ /*如果当前输入的字节数大于已有字节数长度补足64字节整倍数所差的字节数*/ if(inputLen>=partLen) { /*用当前输入的内容把context->buffer的内容补足512bits*/ R_memcpy((POINTER)&context->buffer[index],(POINTER)input,partLen); /*用基本函数对填充满的512bits(已经保存到context->buffer中)做一次转换,转换结果保存到context->state中*/ MD5Transform(context->state,context->buffer); /* 对当前输入的剩余字节做转换(如果剩余的字节<在输入的input缓冲区中>大于512bits的话), 转换结果保存到context->state中 */ for(i=partLen;i+63state,&input[i]); index=0; } else i=0; /*Bufferremaininginput*/ /*将输入缓冲区中的不足填充满512bits的剩余内容填充到context->buffer中,留待以后再作处理*/ R_memcpy((POINTER)&context->buffer[index],(POINTER)&input[i],inputLen-i); } /*MD5finalization.EndsanMD5message-digestoperation,writingthe themessagedigestandzeroizingthecontext.*/ /*获取加密的最终结果 digest:保存最终的加密串 context:你前面初始化并填入了信息的md5结构 */ voidMD5Final(unsignedchardigest[16],MD5_CTX*context) { unsignedcharbits[8]; unsignedintindex,padLen; /*Savenumberofbits*/ /*将要被转换的信息(所有的)的bits长度拷贝到bits中*/ Encode(bits,context->count,8); /*Padoutto56mod64.*/ /*计算所有的bits长度的字节数的模64,64bytes=512bits*/ index=(unsignedint)((context->count[0]>>3)&0x3f); /*计算需要填充的字节数,padLen的取值范围在1-64之间*/ padLen=(index<56)?(56-index):(120-index); /*这一次函数调用绝对不会再导致MD5Transform的被调用,因为这一次不会填满512bits*/ MD5Update(context,PADDING,padLen); /*Appendlength(beforepadding)*/ /*补上原始信息的bits长度(bits长度固定的用64bits示),这一次能够恰巧凑够512bits,不会多也不会少*/ MD5Update(context,bits,8); /*Storestateindigest*/ /*将最终的结果保存到digest中。ok,终于大功告成了*/ Encode(digest,context->state,16); /*Zeroizesensitiveinformation.*/ R_memset((POINTER)context,0,sizeof(*context)); } /*MD5basictransformation.Transformsstatebasedonblock.*/ /* 对512bits信息(即block缓冲区)进行一次处理,每次处理包括四轮 state[4]:md5结构中的state[4],用于保存对512bits信息加密的中间结果或者最终结果 block[64]:欲加密的512bits信息 */ staticvoidMD5Transform(UINT4state[4],unsignedcharblock[64]) { UINT4a=state[0],b=state[1],c=state[2],d=state[3],x[16]; Decode(x,block,64); /*Round1*/ 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*/ /*Round2*/ 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*/ /*Round3*/ 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*/ /*Round4*/ 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; /*Zeroizesensitiveinformation.*/ R_memset((POINTER)x,0,sizeof(x)); } /*Encodesinput(UINT4)intooutput(unsignedchar).Assumeslenis amultipleof4.*/ /*将4字节的整数copy到字符形式的缓冲区中 output:用于输出的字符缓冲区 input:欲转换的四字节的整数形式的数组 len:output缓冲区的长度,要求是4的整数倍 */ staticvoidEncode(unsignedchar*output,UINT4*input,unsignedintlen) { unsignedinti,j; for(i=0,j=0;j>8)&0xff); output[j+2]=(unsignedchar)((input[i]>>16)&0xff); output[j+3]=(unsignedchar)((input[i]>>24)&0xff); } } /*Decodesinput(unsignedchar)intooutput(UINT4).Assumeslenis amultipleof4.*/ /*与上面的函数正好相反,这一个把字符形式的缓冲区中的数据copy到4字节的整数中(即以整数形式保存) output:保存转换出的整数 input:欲转换的字符缓冲区 len:输入的字符缓冲区的长度,要求是4的整数倍 */ staticvoidDecode(UINT4*output,unsignedchar*input,unsignedintlen) { unsignedinti,j; for(i=0,j=0;j
/
本文档为【MD5算法实现】,请使用软件OFFICE或WPS软件打开。作品中的文字与图均可以修改和编辑, 图片更改请在作品中右键图片并更换,文字修改请直接点击文字进行修改,也可以新增和删除文档中的内容。
[版权声明] 本站所有资料为用户分享产生,若发现您的权利被侵害,请联系客服邮件isharekefu@iask.cn,我们尽快处理。 本作品所展示的图片、画像、字体、音乐的版权可能需版权方额外授权,请谨慎使用。 网站提供的党政主题相关内容(国旗、国徽、党徽..)目的在于配合国家政策宣传,仅限个人学习分享使用,禁止用于任何广告和商用目的。

历史搜索

    清空历史搜索