昨天和一个老程序员吃饭聊起codelphi,说很久以前,经常能从这里搜索到一些好的技术文章。
最近的工作也蛮辛苦。开始接触以前从来没有接触过的GNU/linux下的基于gcc的开发。两样东西都是现学的。工作了3个星期,只写了一个 电信smgp3协议的tlv参数解析包。所谓的tlv参数就是(tag ,length,value),tag表示一个指令标志,length,表示这个指令所携带数据的长度,value表示指令所携带的数据,用这种方式传递 参数可以很大程度的在不影响效率的情况下减少空参数所占的空间,节省网络带宽。贴部分代码给大家指正

#ifndef _PTLV_HPP
#define _PTLV_HPP

#include <iostream>
#include <iomanip>
#include <string>
#include <sys/types.h>
/*sowpdu*/
typedef unsigned short WORD;
typedef unsigned char BYTE;

typedef int BOOL;
#ifndef    TRUE
    #define    TRUE        1
    #define    FALSE        0
#endif

////////////////////////////////////////////////////////////////////////////////
/*
  Name:  PTlv
  Copyright:
  Author:
  Date: 14-12-05 15:05
  Description:
*/
class PTlv{
  private:
    WORD tag;//tlv 标识
    WORD len; //value 长度
    BYTE *value; //参数数据体
    int byteOffset;

  public:
    PTlv(WORD new_tag, WORD new_len, BYTE *new_value);
    PTlv(WORD new_tag, BYTE new_value);
   
    PTlv();
   
    void Clone(PTlv &src_tlv);
   
    ~PTlv();
   
    enum VALUE_TYPE{  //
      INTEGER_1          = 0x0001, //byte
      INTEGER_2          = 0x0002, //word
      OCTET_STRING       = 0x0003  //string
    };
   
    enum TLV_Tag{
      TLV_TP_PID             = 0x0001, //GSM协议类型。详细解释请参考GSM03.40中的9.2.3.9
      TLV_TP_UDHI            = 0x0002, //GSM协议类型。详细解释请参考GSM03.40中的9.2.3.23,仅使用1位,右对齐。
      TLV_LINK_ID            = 0x0003, //交易标识,用于唯一标识一次交易
      TLV_CHARGE_USER_TYPE   = 0x0004, //计费用户类型。
                                       // 0=对短消息接收方计费;
                                       // 1=对短消息发送方计费;
                                       // 2=对SP计费;
                                       // 3=表示本字段无效,对谁计费参见ChargeTermID或ChargeTermPseudo 字段。
      TLV_CHARGE_TERM_TYPE   = 0x0005, //计费用户的号码类型。
                                       // 0=真实号码;
                                       // 1=伪码;其它保留。
      TLV_CHARGE_TERM_PSEUDO = 0x0006, //计费用户的伪码
      TLV_DEST_TERM_TYPE     = 0x0007, //短消息接收方的号码类型。
                                       // 0=真实号码;
                                       // 1=伪码;其它保留
      TLV_DEST_TERM_PSEUDO   = 0x0008, //短消息接收方的伪码,当有多个接收方伪码时,要求每个接收方伪码的长度一样。
      TLV_PK_TOTAL           = 0x0009, //相同Msg_Id的消息总条数。
      TLV_PK_NUMBER          = 0x000A, //相同Msg_Id的消息序号,从1开始。
      TLV_SUBMIT_MSG_TYPE    = 0x000B, //SP发送的消息类型。
                                       // 0=普通短消息;
                                       // 1=WEB方式定制结果消息;
                                       // 2=WEB方式取消定制结果消息;
                                       // 3=终端方式定制结果消息;
                                       // 4=终端方式取消定制结果消息;
                                       // 5=包月扣费通知消息;
                                       // 6=WEB方式定制二次确认消息;
                                       // 7=WEB方式取消定制二次确认消息;
                                       // 8=终端方式定制二次确认消息;
                                       // 9=终端方式取消定制二次确认消息;
                                       // 10=WEB方式点播二次确认消息;
                                       // 11=终端方式点播二次确认消息(暂保留);
                                       // 12=群发请求;
                                       // 13:同步订购(包括点播和定制)关系;
                                       // 14:群发结果通知消息。
                                       // 无该字段时,默认为"普通短消息"
                                       // 15:同步订购(包括点播和定制)关系回复;其它保留;
      TLV_SP_DEAL_RESLT      = 0x000C, //SP对消息的处理结果
                                       // 0=成功;
                                       // 1=失败;其它保留。
                                       // 该字段在SubmitMsgType为0、5、6、7、8、9、10、11、14时无效。

      TLV_SRC_TERM_TYPE      = 0x000D, //短消息发送方的号码类型。
                                       // 0=真实号码;
                                       // 1=伪码;其它保留。

      TLV_SRC_TERM_PSEUDO    = 0x000E, //短消息发送方的伪码
      TLV_NODES_COUNT        = 0x000F, //经过的网关数量。该字段的初始值为1。

      TLV_MSG_SRC            = 0x0010, //信息内容的来源。
                                       // 在固定网短消息业务中,MsgSrc填写SP的服务代码。
                                       // 在移动网短消息业务中,MsgSrc填写SP的企业代码。
      TLV_SRC_TYPE           = 0x0011, //传递给SP 的源号码的类型。
                                       // 0=真实号码;
                                       // 1=伪码;其它保留。
      TLV_M_SERVICE_ID       = 0x0012  //业务代码。用于移动网业务
    };
   
    WORD getTag() const { return tag;}
    void setTag(WORD new_tag){ tag = new_tag; }
   
    WORD getTLVLen() const { return 2+2+len; }
    WORD getValueLen() const { return len;}
    int getOffset() const{ return byteOffset;}
   
    BOOL getValue(BYTE *pstr, int value_len);
   
    static BOOL IsValidTag(WORD the_tag);
    static std::string AliasByTag(WORD the_tag);
   
    BOOL Decode(BYTE *pstr, int tlv_len);
    BOOL Encode(WORD the_tag, WORD the_len, BYTE *the_value);
    BOOL Encode(WORD new_tag, BYTE new_value);
   
    void PrintOn(std::ostream & strm) const;
    void ToString(std::ostream & strm) const;
   
    WORD getValueType(WORD the_tag) const;
  protected: 
   
};
////////////////////////////////////////////////////////////////////////////////
#endif //_TLV_HPP








//////////////////////////////tlv.cpp///////////////////////////////////////////////
#include "TLV.hpp"

// PTLV////////////////////////////////////////////////////////////////////

/**构造器
*@param */
PTlv::PTlv(WORD new_tag, WORD new_len, BYTE *new_value){
  Encode(new_tag, new_len, new_value);
}


PTlv::PTlv(WORD new_tag, BYTE new_value){
  Encode(new_tag, new_value);
}

PTlv::PTlv(){
  tag=0x0000;
  len=0x0000;
  value=0;
  byteOffset=0;
}

/**/
void PTlv::Clone(PTlv &src_tlv){
  tag = src_tlv.getTag();
  len = src_tlv.getValueLen();
  byteOffset= src_tlv.getOffset();
  if(value != NULL) {
    delete []value;
    value = NULL;
  }
  value = new BYTE[len];
  memset(value, 0, len);
  src_tlv.getValue(value,len);
}

/*析构器*/
PTlv::~PTlv(){
  if (value != NULL)
    delete []value;
  value      = NULL;
  byteOffset = 0;
  len        = 0;
}

BOOL PTlv::IsValidTag(WORD the_tag){
  switch(the_tag){
    case TLV_TP_PID://             = 0x00000001,
    case TLV_TP_UDHI://            = 0x00000002,
    case TLV_LINK_ID://            = 0x00000003,
    case TLV_CHARGE_USER_TYPE://   = 0x00000004,
    case TLV_CHARGE_TERM_TYPE://   = 0x00000005,
    case TLV_CHARGE_TERM_PSEUDO:// = 0x00000006,
    case TLV_DEST_TERM_TYPE://     = 0x00000007,
    case TLV_DEST_TERM_PSEUDO://   = 0x00000008,
    case TLV_PK_TOTAL://           = 0x00000009,
    case TLV_PK_NUMBER://          = 0x0000000A,
    case TLV_SUBMIT_MSG_TYPE://    = 0x0000000B,
    case TLV_SP_DEAL_RESLT://      = 0x0000000C,
    case TLV_SRC_TERM_TYPE://      = 0x0000000D,
    case TLV_SRC_TERM_PSEUDO://    = 0x0000000E,
    case TLV_NODES_COUNT://        = 0x0000000F,
    case TLV_MSG_SRC://            = 0x00000010,
    case TLV_SRC_TYPE://           = 0x00000011,
    case TLV_M_SERVICE_ID://       = 0x00000012,
      return TRUE;
    default :
      return FALSE;
  }
}

/**
* 取得tag的别名
* @param the_tag tag标识
* @return string 别名
*/
std::string PTlv::AliasByTag(WORD the_tag){
  switch(the_tag){
    case TLV_TP_PID:
      return "TLV_TP_PID";
    case TLV_TP_UDHI:
      return "TLV_TP_UDHI";
    case TLV_LINK_ID:
      return "TLV_LINK_ID";
    case TLV_CHARGE_USER_TYPE:
      return "TLV_CHARGE_USER_TYPE";
    case TLV_CHARGE_TERM_TYPE:
      return "TLV_CHARGE_TERM_TYPE";
    case TLV_CHARGE_TERM_PSEUDO:
      return "TLV_CHARGE_TERM_PSEUDO";
    case TLV_DEST_TERM_TYPE:
      return "TLV_DEST_TERM_TYPE";
    case TLV_DEST_TERM_PSEUDO:
      return "TLV_DEST_TERM_PSEUDO";
    case TLV_PK_TOTAL:
      return "TLV_PK_TOTAL";
    case TLV_PK_NUMBER:
      return "TLV_PK_NUMBER";
    case TLV_SUBMIT_MSG_TYPE:
      return "TLV_SUBMIT_MSG_TYPE";
    case TLV_SP_DEAL_RESLT:
      return "TLV_SP_DEAL_RESLT";
    case TLV_SRC_TERM_TYPE:
      return "TLV_SRC_TERM_TYPE";
    case TLV_SRC_TERM_PSEUDO:
      return "TLV_SRC_TERM_PSEUDO";
    case TLV_NODES_COUNT:
      return "TLV_NODES_COUNT";
    case TLV_MSG_SRC:
      return "TLV_MSG_SRC";
    case TLV_SRC_TYPE:
      return "TLV_SRC_TYPE";
    case TLV_M_SERVICE_ID:
      return "TLV_M_SERVICE_ID";
    default :
      return "TLV_UNKNOWN_TAG_ID";
  }
}



void PTlv::PrintOn(std::ostream &strm) const{
  strm << "SMGP3_TLV:{\n";
  strm << std::setw(15) << "tag:" << AliasByTag(tag) << " 0x" << std::hex << std::setw(sizeof(tag))<< std::setfill('0') << tag << '\n';
  strm << std::setfill(' ') << std::setw(15) << "len:" << len << std::endl;
  strm << std::setfill(' ') << std::setw(17) << "value:" << value <<std::endl;
  strm <<  "}\n";
}

void PTlv::ToString(std::ostream &strm) const{
  strm <<  "alias:" << AliasByTag(tag) << " len:" << len <<   " value:" << value <<'\n';
}

WORD PTlv::getValueType(WORD the_tag) const{
  switch (the_tag) {
    case TLV_TP_PID:
    case TLV_CHARGE_USER_TYPE:
    case TLV_CHARGE_TERM_TYPE:
    case TLV_DEST_TERM_TYPE:
    case TLV_PK_TOTAL:
    case TLV_PK_NUMBER:
    case TLV_SUBMIT_MSG_TYPE:
    case TLV_SP_DEAL_RESLT:
    case TLV_SRC_TERM_TYPE:
    case TLV_NODES_COUNT:
    case TLV_SRC_TYPE:
    case TLV_TP_UDHI:{
      return INTEGER_1;
      break;
    }
    case TLV_CHARGE_TERM_PSEUDO:
    case TLV_SRC_TERM_PSEUDO:
    case TLV_DEST_TERM_PSEUDO:
    case TLV_MSG_SRC:
    case TLV_M_SERVICE_ID:
    case TLV_LINK_ID:{
      return OCTET_STRING; //20
      break;
    }
    default :
      return 0x00000000;
  }
}

BOOL PTlv::getValue(BYTE *pstr, int value_len){
  assert(len >0 );
  assert(value != NULL );
  if (value_len < len) return FALSE;
  memset(pstr,0,value_len);
  memcpy(pstr,value,len);
  return TRUE;
}

BOOL PTlv::Decode(BYTE *pstr, int tlv_len){
  memcpy(&tag, pstr, 2);
  memcpy(&len, pstr+2, 2);
  if (tlv_len < len)  //tlv包长度必须大于value长度
    return FALSE;
  if (value != NULL) { //如果之前value有申请过空间,则先释放他
    delete[] value;
    value = NULL;
  }
 
  byteOffset = 0;
  value = new BYTE[len+1];
  //memset(value, 0, len+1);
  memcpy(value, pstr+4, len);
  return TRUE;
}

BOOL PTlv::Encode(WORD the_tag, WORD the_len, BYTE *the_value){
  assert(the_value != NULL);
  assert(the_len > 0);
  tag=the_tag;
 
  if(value != NULL ){
    delete [] value;
    value = NULL;
  }
  value = new BYTE[the_len];
  memset(value, 0, sizeof(BYTE)*the_len);
 
  memcpy(value, (BYTE *)the_value, the_len);
  len = the_len;
  byteOffset = 0;
}

BOOL PTlv::Encode(WORD the_tag, BYTE the_value){
  len=sizeof(BYTE);
  tag = the_tag;
  if (value != NULL){
    delete [] value;
    value = NULL;
  }
 
  value = new BYTE[len];
  memset(value,0,sizeof(BYTE)*len);
  byteOffset = 0;
  value[byteOffset++] = the_value;
}