i社区用户: 密码:
搜索:
首页| 资讯| 信息化| 视频| 专题| 创业| 企业中心| 企业博客| 职场| 周报全文| 方案中心 |电子杂志
市场会展| Blog| i社区| 社区联盟| RSS| 直播 | 发行| wiki |i| 下载| 英才招聘|CIO文库
IT两会 | 互联网年会
产品中心| 笔记本|台式机| DIY|数码| 服务器|网络通信| 手机|软件|显示器|信息安全|存储|组网| 动漫
首页|文库|数据中心|方案中心|专家库|厂商黄页|供求招标| 责编电话:010-68130909-1053
 
编写一个STL 中的CString类
 
 
 
 
2003-06-03 16:04:26
 
  徐岩柏  
  STL英文是Standard Template Library,也就是我们常说的C++标准模板库,。该标准库于1998年被正式纳入C++标准,给全世界的C++程序员带来了福音。最让我们兴奋的应该是它的跨平台性,使得你在WINDOW,UNIX ,LINUX操作系统上面用标准C++编写的程序不用修改即可移植。(当然要有C++的编译器)。

现在的编译器虽然对标准C++支持程度不同,单总体上还是很好。WINDOWS平台的VC ,BC,Linux/UNIX平台的g++都是一流的编译器,都支持STL。而且STL是有源代码的,你可以扩展增加,避开这些微小的不同。

说到STL首先要说的当然是字符串处理类std::string,这可能是一个程序员使用最多的一个类,它的功能强大,使用非常方便。但习惯于用VC的CString开发的编程人员会感到有点不方便。幸运的是这个不方便可以很容易的解决,方法就是对标准的字符串类std::string进行包装,生成一个类似Cstring的类,我把它命名为Xstring。

下面就从Format函数说起,这可能是大部分人最希望用到的:

本函数是一个变参函数,对参数不定的函数其各式如下:

int Format(const char* pstrFormat, ... )

其中pstrFormat是格式串,三个点代表所有的参数。格式中的每个格式和后面的参数必须相对应,否则函数的执行会出现意想不到的结果;当然过多的参数将被忽略。格式分为简单字符和转换规范字符两类。具体格式规范有如下格式:

%[flags] [width] [.precision] [{h | l | I64 | L}]type

flags是标志字符,输出对齐,尾零,数值符号,进制数(八或十六)

width是宽度规范符,填补空格或0的个数

precision是精度规范符,打印字符最多个数,对于整数值,为最少数字个数

h短整型数的输出

I长整型数的输出

I64为64位的整型输出

如果你对格式还不清楚,请参考有关printf的格式资料。

对不定参数的处理也很特殊,要使用下面的几个函数

先声明一个变量va_list argList;

va_start(argList,pstrFormat);

int cnt =vsprintf(buff, pstrFormat, argList);

va_end(argList);

这样就把格式化后的结果保存在buff字符串中了。

然后最重要的就是计算这个buff有多大,如果大了就有内存浪费,小了完不成任务,所以要根据格式来动态计算,然后动态的开辟内存空间。就用一个循环来把格式串中的每一个字符读出来分别处理。先初始化一个长度变量nMaxLen =0;

for (const char * p = pstrFormat; *p != '\0';p++ )

如果读出来的不是’%’或是‘%%’则长度加一。

if (*p != '%' || *(++p) == '%')

{

nMaxLen += 1;

continue;

}

如果前一个字符是‘%’,则读取格式,如果是‘#’则长度加2,来为‘0x’预留空间;如果是‘*’,则读紧跟着的一个整数,得到指定的宽度;其他的‘+’、‘-’、‘ ’、‘0’等字符主要是填充用,忽略长度。

for (; *p != '\0'; p ++)

{

if (*p == '#')

nMaxLen += 2; // 处理 '0x'

else if (*p == '*')

nWidth = va_arg(argList, int); //如:'%5f' 中的5

else if (*p == '-' || *p == '+' || *p == '0'|| *p == ' ')

; //忽略该符号

else // 不是标志字符就退出循环

break;

}

如果下一个字符是‘.’则忽略去读取其后面一个字符,如果是‘*’则也要读出其后的宽度,来计算精度。

if (*p == '*')

{

nPrecision = va_arg(argList, int);

p ++;

}

else

{

nPrecision = atoi(p);

for (; *p != '\0' && isdigit(*p); p ++)

;

}

接下来处理字符如果是‘h’、‘l’、‘I’、‘F’、‘N’等,则忽略计算长度。

如果读取字符是‘c’、‘C’则长度加上2(考虑宽字符的情况);如果读取的是‘s’、‘S’则要计算参数中给的字符串的宽度。

switch (*p)

{

case 'c':

case 'C':

nItemLen = 2;

va_arg(argList, char);

break;

case 's': // 字符串

case 'S':

nItemLen = strlen(va_arg(argList, const char*));

nItemLen = ((1) > (nItemLen)) ? (1) : (nItemLen);//如果是空串就使用1 即保存'\0'

break;

}

如果读出的字符是‘d’、‘i’、‘u’、‘x’、‘o’,‘e’、‘f’、‘G’、‘g’等,则长度加上对应的数值型的长度,当然最好使用sizeof计算,使得具有更好的移植能力。

case 'd':

case 'i':

case 'u':

case 'x':

case 'X':

case 'o':

va_arg(argList, int);

nItemLen = 32;

在for循环体的最后当然不要忘了把长度累计nMaxLen += nItemLen;当循环结束时,长度的计算也就完成了。

For循环结束时调用va_end(argList);

下面就可以开辟恰当的内存空间,来保存你的格式串。

char* ch = new char[nMaxLen+1];

有了空间再重新接收参数就可以了。

va_start(argList, pstrFormat);

vsprintf(ch, pstrFormat, argList);

va_end(argList);

最后不要忘了把空间加到std::string中,可以直接调用append函数:

this->append(ch);

然后释放你的内存空间

delete[] ch;

其他的函数可以用std::string中的相对应的功能包装即可,下面就再写一个MakeUpper函数,它也是CString中的。

void MakeUpper()

{

std::transform(this->begin (), \

this->end (),this->begin (),\

toupper);

}

是不是很容易呢,希望本文能起到抛砖引玉的作用,给你使用STL带来方便。

以上的程序编码在VC和g++中均可使用。本人曾用XString 类替代了一个用MFC编写的项目中的所有的CString类 ,使得它顺利的用G++编译通过。

本用例的完整代码为stl.zip

 
   
  (网页编辑:李卫华  
     
      如果您对“编写一个STL 中的CString类”有任何问题想要咨询,请通过点击下面的按钮来在线提交意向单,我们编辑和信息化专家将会很快为您解答问题。
专家介绍
惠普ProCurve构筑安全网络
检索
 
 
 
检索词 >>
电信 金融 信息安全
软件 CRM 电子政务
ERP 物流 电子商务
行业 >>
电信 能源 医疗
农业 银行 政府
交通 制造业 教育

收藏此页到ViVi
  发表评论  您的姓名   您的Email   发布  
   
相关文章
关于我们 | 客服热线 | 广告服务 | 招聘信息 | 法律声明 | 投稿指南 | 联系方式 
Copyright(C) ccw.com.cn,All rights reserved
中国计算机世界出版服务公司内容版权所有
京ICP证010182