在写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 }

|