Tuxedo缓冲区的封装代码,成就你打造通用的tuxedo客户端(二)
friend string IntToString(int );
friend string DoubleToString(double);
friend string FloatToString(float);
friend string charToString(char *);
friend string& trim(string &s) ;
friend char *delspace(char *str);
friend int splitStr(char *ps,char *split,char dest[255][255]);
friend int iNum(char u[],char v[]); string m;
};
extern TuxPost gtux;
struct VeiwType//这个类的作用是建立业务数据成员,业务成员长度以及业务数据成员的类型
{
string name;//业务成员名称
string value;//业务成员的长度
string tmp;//业务成员的类型
string orivalue; //业务成员的原始长度
long begin;//业务成员开始位置,在view32缓冲区里面用到
string param;//业务类型入参还是出参
VeiwType(string a,string b,string c);//构造函数,构造这么一个对象比如:empno
4int,其中empno是业务名称,4是大小,int是类型
VeiwType();
~VeiwType();
};
string DoubleToString(double n);//声明了double到string类型的转化
string FloatToString(float n);//生命了float到string类型的转化
string IntToString(int);//声明int怎么到string类型的转化
string LongToString(int n);
void AllTrim (char sBuf[]);//删除字符串两边的空格
int splitStr(char *ps,char *split,char dest[][255]);
Trie对象的声明
struct Trie
{
string sname;//服务���称
string wsnaddr;//服务的WSNADDR
string servtype;//服务的类型
string fieldtbs;//只有FML类型使用,服务的域表头文件
string fldtbldir;//只有FML类型使用,服务的域表头的目录
string members;//只有view和FML类型使用,表示业务成员的顺序格式
map<string,VeiwType> RedotheView;//重新处理已经请求过的服务的时候,用的业务成员容器信息
int flag;//标志,保存是否成功的标志
Trie(string s1,string s2,string s3,string s4,string s5,string s6);
int TrieClear();
~Trie();
};
struct Veiw
这个类是用来处理每一行的service.def信息的.
{
string
outstr; // string instr; map<string,string> Vinbufmap; //VIEW32缓冲区使用的map vector<string> Voutbuf;
//VIEW32缓冲区使用的容器
Veiw();
~Veiw();
int DealoneRecord(char *,char *);//此函数比较重要,用户处理每一行的的服务定义文件的数据
bool PrintVeiw();//打印map里面的数据 \
bool Printmap();//打印map里面的信息
bool Printvec();
//打印vec里面的信息
map<string,VeiwType> theView;//业务成员容器的信息:此map和RedotheView一一对应.。
vector<string> theVector;
bool UpDate();//此函数比较重要,用于在view类型缓冲区中做结构体内存成员对齐和结构体本?淼哪诖娑云?
string m;//产生日志使用
int GetData(char*data,char *value);
//此函数重要,用于将业务的数据,赋值给我们定义的结构体成员,比如:GetData(empno,123),由于empno在themapview32这个map里面对应的是v1,所以赋值之后,就变成?藇1=123; int
SumVeiw();//获取themapview32这个map里面的所有的成员的大小的总和
char wsnaddr[255];//这个是wsnaddr,从服务定义文件获取
char ServType[255];//本次服务所接收的类型,从服务定义文件获取
void Clear();
};
if(gtux.VeiwInit(sout,sname)==0)
{
break;
}
int TuxPost::VeiwInit(char *a,char *name)
{ //也就是用gv这个对象,调用3个成员函数,每一行的内容,name对应请求的服务名称.
if(gview.DealoneRecord(a,name)==0 &&gview.UpDate()==0 &&SetServType()==0) {
m+=gview.m;
gview.m.clear();
return 0;
}
return -1;
}
int Veiw::DealoneRecord(char *sRecord,char *sname)
{//输入是服务定义文件里���的每一行内容,记住是每一行
char out[255][255];
char out2[255][255];
char sTmp[255];
int iNum=0;
theView.clear();
string s;
memset(out,0,sizeof(out));
memset(out2,0,sizeof(out2));
iNum=splitStr(sRecord,"@",out);//isalt.def里面的每个记录是用@来分割的
if(iNum<=2) {
m+="||@符号太少,服务配置错误,请检查";
return -1;
}
//经过xplitstr函数处理之后,那么sout[0]="INSERT",sout[1]="192.168.136.129:7200",sout[2]=view32,sout[3]=rid:in
//sout[4]=int,sout[5]=rname:in,sout[6]=int sout[7]=char[8]
memset(sTmp,0,sizeof(sTmp));
strcpy(sTmp,out[0]);
//获取每行的第一列,是service_name
sTmp[strlen(sTmp)]=0;
AllTrim(sTmp);
int i=3;
int j=1;
if(strncmp(sTmp,sname,sizeof(sname))==0) {
//判断本行的servicename是否为请求的sname,如果是,继续
m+="||在isalt.def中找到所需要处理的servicename这行记录,开始处理";
memset(wsnaddr,0,sizeof(wsnaddr));
strcpy(wsnaddr,out[1]);
wsnaddr[strlen(out[1])]='\0';
memset(ServType,0,sizeof(ServType));
strcpy(ServType,out[2]);
ServType[strlen(out[2])]='\0';
for(i=3;i<iNum-1;i++) {
if(strcmp(out[i],"")!=0) {
memset(g_out2,0,sizeof(g_out2));
if(splitStr(out[i],":",g_out2)==3) {
s=g_out2[1];
transform(s.begin(),s.end(),s.begin(),::toupper);
if(s!="IN"&&s!="OUT"&&s!="INOUT") {
memset(sTmp,0,sizeof(sTmp));
sprintf(sTmp," ||第%d个成员业务员参数类型%s错误,请改正",j,s.c_str());
m+=sTmp;
return -1;
}
VeiwType a(g_out2[0],g_out2[1],s);
//读取业务数据成员名称,和业务数据成员类型,分别填充到map里面,
//先绑定到a里面,然后和v1~v50分别绑定到map里面 string
stmp="v";
stmp+=IntToString(j);
theView.insert(map<string,VeiwType>::value_type(stmp,a));
j++;
}
}
}
return 0;
}
m+=" || 本条记录不是要请求的servicename";
return -1;
}
int splitStr(char *ps,char *split,char dest[][255])
{//ps是一段长的字符串,split是分隔符。。。dest是输出
char sTmp[1024]="";
char *sBegin="",*sEnd="";
char *sStr="";
int i;
i=0;
strcpy(sTmp,ps);
sStr=sTmp;
while(1) {
sBegin=strstr(sStr,split) ;
if(sBegin) {
if(sBegin-sStr>0)
strncpy(dest[i],sStr,sBegin-sStr);
else
dest[i][0]=0;
sStr=sBegin+strlen(split);
i++;
} else {
strcpy(dest[i],sStr);
i++;
break;
}
}
return i;
}
// 下面是update函数,用于内存对齐。。。只有VIEW32缓冲区能用到:
bool Veiw::UpDate()//此函数用于内存对齐,逻辑较复杂。。。废了哥九牛二虎之力
{ int sum1=0;
int sum=0;
int nn=0;
int nn1=0;
map<string,VeiwType>::iterator it;
map<string,VeiwType>::iterator it1;
map<string,VeiwType>::iterator it2;
map<string,VeiwType>::iterator it3;
map<string,VeiwType>::iterator it4;
//成员内存对齐
for(it=++theView.begin();it!=theView.end();++it)
{//从第二个成员开始对齐
sum=0;
int n=atoi(it->second.value.c_str());//找到它的大小
int n0=atoi((--it)->second.value.c_str());//找到它前面那个成员的大小
++it; //只带器回到当前位置
if(strncmp(it->second.tmp.c_str(),"char",4)!=0) {//char类型不用做内存对齐
for(it1=theView.begin();it1!=it;++it1) {//求取当前成员前面所有成员的大小综合
sum+=atoi(it1->second.value.c_str()); //求取前面所有成员的大小
}
if(n!=0 &&sum%n==0
) {//看前面所有成员总和是不是当前成员大小的整数倍数
continue;
}
if(n!=0&&sum%n!=0) {//如果不是整数倍数,那么当前成员的前一个成员进行扩充,扩充到是整数倍
(--it)->second.value=IntToString(n0+(n-sum%n));//对齐数据的诚信
++it;
}
}
}
//下面是结构体内存对齐
sum1=0;
for(it2=theView.begin();it2!=theView.end();++it2)
{
sum1+=atoi((it2->second).value.c_str();
}//求取所有成员的大小综
it2=--theView.end();//将指针只带器移动到末尾成员
int max=0;
for(it3=theView.begin();it3!=theView.end();++it3)
{
if(strncmp(it3->second.value.c_str(),"char ",4)!=0) {
if(atoi(it3->second.value.c_str())>max)
max=atoi(it3->second.value.c_str());
}
}
//求取容器里面占用字节最大的成员大小
// cout<<"the max is "<<max<<endl;
if(sum1%max!=0)
{//如果容器总和不是最大成员的整数倍
int n2=atoi(it2->second.value.c_str())
;
it2->second.value=IntToString(n2+(max-sum1%max));//末尾成员进行扩充
}
for(it3=theView.begin();it3!=theView.end();++it3)
{
int sumbegin=0;
for(it4=theView.begin();it4!=it3;++it4) {
sumbegin+=atoi(it4->second.value.c_str());
}
it3->second.begin=sumbegin;
}//计算每一项起始位置的死循环,对于view是这么做的
return 0;
}