C语言对通讯数据包生成校验码的例子

转载 2007年09月30日 14:55:00
转自:http://blog.csdn.net/hnwanghb/archive/2005/03/15/319834.aspx
在写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
}

 

相关文章推荐

C语言生成SHA1校验码

C语言生成SHA1校验码,记下备用。 /*sha1.h*/ #ifndef _SHA1_H #define _SHA1_H #define uint8 unsigned char #define u...

C语言生成MD5校验码

如有转载,请注明出处:http://blog.csdn.net/embedded_sky 作者:super_bert@csdn C语言生成MD5校验码源码,从Linux命令源码md5sum移植,M...

C编程获取指定网卡网络数据包并分析(附C语言源码)

 [cpp] #include    #include    #include    #include    #include      int main(){ ...

linux网络编程之:获取指定网卡网络数据包并分析(附C语言源码)

#include #include #include #include #include int main(){ char *net_dev; char *ip_addr; char...

CRC校验码生成

  • 2015-04-08 17:02
  • 68KB
  • 下载

java生成MD5校验码及算法实现

在Java中,java.security.MessageDigest (rt.jar中)已经定义了 MD5 的计算,所以我们只需要简单地调用即可得到 MD5 的128 位整数。然后将此 128 位计 ...

LRC校验码自动生成

内容举报
返回顶部
收藏助手
不良信息举报
您举报文章:深度学习:神经网络中的前向传播和反向传播算法推导
举报原因:
原因补充:

(最多只允许输入30个字)