中利用校验码为图设置水印覆盖
四、示例程序的设计实现
首先我们把本文中最关键的用于产生CRC校验和(余数)的函数CheckOut()的实现部分主要代码做了介绍,其中为了减少运算量、方便编程、将烦琐的多项式长除取余的过程按照前面讲述的CRC算法的运算过程和原理将其预先设定在一个256字节长的数组表CheckTable中,实际运算时只须按索引对其查表即可:
BYTE CCRCDlg::CheckOut(BYTE *byteData, UINT nCount)
{
……
static BYTE CheckTable[256]=
{
0x00,0x07,0x0e,0x09,0x1c,0x1b,0x12,0x15,0x38,0x3f,
0x36,0x31,0x24,0x23,0x2a,0x2d,0x70,0x77,0x7e,0x79,
0x6c,0x6b,0x62,0x65,0x48,0x4f,0x46,0x41,0x54,0x53,
0x5a,0x5d,0xe0,0xe7,0xee,0xe9,0xfc,0xfb,0xf2,0xf5,
0xd8,0xdf,0xd6,0xd1,0xc4,0xc3,0xca,0xcd,0x90,0x97,
0x9e,0x99,0x8c,0x8b,0x82,0x85,0xa8,0xaf,0xa6,0xa1,
0xb4,0xb3,0xba,0xbd,0xc7,0xc0,0xc9,0xce,0xdb,0xdc,
0xd5,0xd2,0xff,0xf8,0xf1,0xf6,0xe3,0xe4,0xed,0xea,
0xb7,0xb0,0xb9,0xbe,0xab,0xac,0xa5,0xa2,0x8f,0x88,
0x81,0x86,0x93,0x94,0x9d,0x9a,0x27,0x20,0x29,0x2e,
0x3b,0x3c,0x35,0x32,0x1f,0x18,0x11,0x16,0x03,0x04,
0x0d,0x0a,0x57,0x50,0x59,0x5e,0x4b,0x4c,0x45,0x42,
0x6f,0x68,0x61,0x66,0x73,0x74,0x7d,0x7a,0x89,0x8e,
0x87,0x80,0x95,0x92,0x9b,0x9c,0xb1,0xb6,0xbf,0xb8,
0xad,0xaa,0xa3,0xa4,0xf9,0xfe,0xf7,0xf0,0xe5,0xe2,
0xeb,0xec,0xc1,0xc6,0xcf,0xc8,0xdd,0xda,0xd3,0xd4,
0x69,0x6e,0x67,0x60,0x75,0x72,0x7b,0x7c,0x51,0x56,
0x5f,0x58,0x4d,0x4a,0x43,0x44,0x19,0x1e,0x17,0x10,
0x05,0x02,0x0b,0x0c,0x21,0x26,0x2f,0x28,0x3d,0x3a,
0x33,0x34,0x4e,0x49,0x40,0x47,0x52,0x55,0x5c,0x5b,
0x76,0x71,0x78,0x7f,0x6a,0x6d,0x64,0x63,0x3e,0x39,
0x30,0x37,0x22,0x25,0x2c,0x2b,0x06,0x01,0x08,0x0f,
0x1a,0x1d,0x14,0x13,0xae,0xa9,0xa0,0xa7,0xb2,0xb5,
0xbc,0xbb,0x96,0x91,0x98,0x9f,0x8a,0x8d,0x84,0x83,
0xde,0xd9,0xd0,0xd7,0xc2,0xc5,0xcc,0xcb,0xe6,0xe1,
0xe8,0xef,0xfa,0xfd,0xf4,0xf3
};
……
BYTE Result = CheckTable[*byteData];
for(UINT i=1;inCount;i++)//对数据流进行CRC校验和计算
{
Result = Result^(*(byteData+i));
Result = CheckTable[Result];
}
……
return Result;//返回计算出来的CRC校验和。
}
该函数的两个入口形参BYTE *byteData和 UINT nCount分别表示要计算CRC校验码的数据流和该数据流的长度。函数返回值为计算出来的CRC校验码,其他函数只须简单的调用此函数即可得到关于某段数据序列的校验码,下面是为BMP 位图设置印鉴的部分主要代码:
……
//首先读取文件到缓存中去,并得到文件的长度
en(FileName,CFile::modeReadWrite);
int FileLen=tLength();
//为缓存动态分配内存空间
BYTE* buf;
buf=new BYTE [FileLen];
ekToBegin();
ad(buf,FileLen);
//调用CheckOut函数计算BMP位图的位图阵列部分(38字节以后部分)的CRC校验码
BYTE crc=CheckOut(buf+38,FileLen-38);
//在前面提到过模2减法,在以2为模的加、减法中,都是和异或运算相同的。
//把经过模2减法的计算结果保存到BMP位图的保留空间中
buf[6]=0^crc;
ek(6,CFile::begin);
ite(buf+6,1);
//释放申请过的内存空间、关闭打开的文件
delete[] buf;
ose();
……
至于检验文件是否经过处理的代码和前面的设置印鉴的代码是非常类似的,也是先将文件读取到缓存中,只是要把存放在第六字节保留字段中的印鉴水印重新恢复到数据序列的末尾,然后再把该序列被G(x)除,通过判断能否整除即可断定该文件是否曾被改动过:
……
en(FileName,CFile::modeReadWrite);
int FileLen=tLength();
……
//读取文件到缓存
BYTE* buf;
buf=new BYTE [FileLen+1];
可以找到仓库管理员 ekToBegin();
ad(buf,FileLen);
ose();
//将印鉴恢复到数据序列的末尾
buf[FileLen]=buf[6];
//计算带有校验码(印鉴)的序列被G(x)标准多项式整除的余数
BYTE crc=CheckOut(buf+38,FileLen-37);
//判断crc(余数)是否为0(即能否整除),来判定文件的原始性。
if(crc!=0)
AfxMessageBox(\"警告!文件被改动过了!\"); //CRC校验错误!印鉴有损。
else
AfxMessageBox(\"文件没有发生过改动!\"); //CRC校验正确,印鉴完好。
delete[] buf;
……
小结:
本文对CRC校验在文件印鉴水印方面的应用做了简要的介绍,在对CRC 校验和其他文件格式有了更深入的了解后,对本文的代码稍加改动,就可以对其他格式的文件设置类似的印鉴水印。本程序在Windows2000 Professional下,由Microsoft Visual C++ 6.0编译调试通过。 查看本文来源
昆明医院哪家白癜风医院好兰州白癜风哪里治疗好
重庆医院妇科哪家医院好