在写Socket通讯程序的时候,为了校验传送的数据包的完整性,我一般会使用给数据包附加校验码
的方法,具体做法就是: 对要传输的数据从第一个字符到最后一个字符进行异或运算,最后回得到一个异或
结果字符,把这个字符转换为2进制字符串!附加在数据包后面,收取数据的一方,收到数据后,先根据预先定义
的格式拆分数据包,对收到的数据进行异或运算, 然后把结果和发送方的校验码比较,如果一致,说明数据传输
没有问题。
附上一段银行接口实际应用中的例子:
//定义报文结构
typedef struct tradePackage
{
char p_trade_code[4];
char p_mobile_no[16];
char p_agent_account[26];
char p_our_account[26];
char p_trade_fee[16];
char p_trade_date[9];
char p_trade_time[7];
char p_bank_id[3];
char p_trade_node[11];
char p_operator_no[5];
char p_bank_flow_no[13];
char p_result_code[4];
char p_our_flow_no[13];
char p_check_code[9];
}PACKAGE;
/*
功能描述:生成发送报文的校验码(显示为2进制的字符方式)
返回参数:checkcode为对input_buf串的每个字符做异或生成的校验码
*/
void make_XOR_checkcode(char * input_buf,char * checkcode)
{
int m=strlen(input_buf);
int i,n;
//保存异或的结果字符
char ret;
char s[m];
char b[8];
int x=0x80;
strcpy(s,input_buf);
//给准备返回的报文字符逐个做异或
for(i=0;i<m-1;i++)
{
if(i==0)
{
ret=s[i]^s[i+1];
}
else
{
ret=ret^s[i+1];
}
}
/*
用异或的结果ret和x做位比较,做出ret的2进制校验码 start
*/
for(n=0;n<8;n++)
{
if((ret&x)==0)
{
b[n]='0';
}
else
{
b[n]='1';
}
x=x>>1; //右移一位相当于除2
}
b[8]='\0';
/*
用异或的结果ret和x做位比较,做出ret的2进制校验码 end
*/
strcpy(checkcode,b); //返回生成的校验码
}
/*
功能描述:检验发来的数据报文的内容是否与校验码一致
返回参数:trade_package参数是由银行方传来的报文,用trade_package.p_check_code来
对其他的字符串进行校验。校验成功返回1,否则返回0
*/
int packageXOR_parse(struct tradePackage trade_package)
{
//报文格式:交易码3手机号码15代理商银行帐号25商户银行帐号25金额15交易日期8交易时间6银行编码2交易网点10
//操作员4银行流水号12结果代码3商户流水号12校验码8
char chkcode[8];
char strtmp[256];
char input_str[1024];
char checkcode[8];
memset(input_str, 0x0, sizeof(input_str));
//1:交易码 3 不空
strcpy(input_str,trade_package.p_trade_code);
//2:手机号码 15 不空 右补空格
memset(strtmp, 0x0, 15);
sprintf(strtmp,"%-15s",trade_package.p_mobile_no); //显示字符串时限定15位(不足则补空格),并靠左端对齐
strcat(input_str,strtmp);
//3:代理商银行帐号 25 不空 右补空格(如现金缴款,全空格)
memset(strtmp, 0x0, 25);
sprintf(strtmp,"%-25s",trade_package.p_agent_account); //显示字符串时限定25位(不足则右补空格),并靠左端对齐
strcat(input_str,strtmp);
//4:商户银行帐号 25 不空
memset(strtmp, 0x0, 25);
sprintf(strtmp,"%-25s",trade_package.p_our_account); //显示字符串时限定25位(不足则右补空格),并靠左端对齐
strcat(input_str,strtmp);
//5:金额 15 不空 含小数点和两位小数,右补空格
//sprintf(trade_package.p_trade_fee,"%0.2f",p_trade_fee);
memset(strtmp, 0x0, 15);
sprintf(strtmp,"%-15s",trade_package.p_trade_fee); //显示字符串时限定15位(不足则右补空格),并靠左端对齐
strcat(input_str,strtmp);
//6:交易日期 8 不空 YYYYMMDD
strcat(input_str,trade_package.p_trade_date);
//7:交易时间 6 不空 HHMMSS
strcat(input_str,trade_package.p_trade_time);
//8:银行编码 2 不空
strcat(input_str,trade_package.p_bank_id);
//9:交易网点 10 不空 根据银行方规定
strcat(input_str,trade_package.p_trade_node);
//10:操作员 4 不空 柜台:操作员其它为设备号
strcat(input_str,trade_package.p_operator_no);
//11:银行流水号 12 不空 左补0做重复性检查
sprintf(strtmp,"%12s",trade_package.p_bank_flow_no); //显示字符串时限定12位(不足则左边补空格),并靠右端对齐
strcat(input_str,strtmp);
//12:结果代码 3 空格
strcat(input_str,trade_package.p_result_code);
//13:商户流水号 12 空格 左补0
//显示字符串时限定12位(不足则左边补0),并靠右端对齐
sprintf(strtmp,"%012s",trade_package.p_our_flow_no);
strcat(input_str,strtmp);
//用发来的报文生成校验码chkcode,然后用chkcode和trade_package.p_check_code进行比较 start
make_XOR_checkcode(input_str,chkcode);
char str[128];
sprintf(str,"%d",strcmp(trade_package.p_check_code,chkcode));
strcat(str," comp");
//如果比较
if(strcmp(trade_package.p_check_code,chkcode)==0)
return 1;
else
return 0;
//用发来的报文生成校验码chkcode,然后用chkcode和trade_package.p_check_code进行比较 end
}