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

2008-9-19 18:34 4293 9 10 分类: 软件与OS
在写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"^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
}

PARTNER CONTENT

文章评论1条评论)

登录后参与讨论

用户377235 2013-4-21 14:15

思路不错!
相关推荐阅读
wqd520 2008-11-19 21:10
嵌入式Linux开发需要的参考资料
引导:如需获得对 vmlinux 和 zimage 之间区别的极好解释,请在 Alessandro Rubini 编写的“Kernel Configuration: dealing with the ...
wqd520 2008-11-05 16:13
队列的c语言实现
队列是一种与栈相反的数据结构,它是先进先出(first in, first out),同样它也是一种运算受限的结构。 队列的抽象数据类型: ADT QUEUE = { InitQueue(&Q...
wqd520 2008-11-05 16:11
修改了一个抓包的程序(c语言队列)
修改了一个抓包的程序(c语言队列)05 年底写了个抓包的小工具,做某些方面的测试用。但是在大流量的时候,存在一些问题,经常会漏掉一些包没有分析,即使接到了预定义好的包,也没有办法作出某些响应。周末没事...
wqd520 2008-11-05 16:08
数据结构C语言实现系列——队列
#include <stdio.h>#include <stdlib.h>typedef int elemType;/*****************************...
wqd520 2008-10-22 11:45
嵌入式软件工程师的技能修炼
点穴: 深入了解至少一种处理器/控制器的体系结构,熟悉其汇编指令。      掌法: 精通C语言,掌握加载技术和编译链接知识。 兵刃: 了解常用存储、通讯和人机接口设备,熟练编写设备驱动程序。 阵法:...
wqd520 2008-10-15 11:42
嵌入式系统硬件抽象层的建立及软件的可移植性设计
摘要:在阐述嵌入式系统软件设计方法的基础上,介绍嵌入式系统底层软件可移值性设计和硬件抽象层的建立;举例说明利用此思想的嵌入式软件的设计及测试过程。     关键词:设备驱动程序 嵌入式系统 软件设计 ...
EE直播间
更多
我要评论
1
9
关闭 站长推荐上一条 /3 下一条