VC++ CString的操作.docx

上传人:b****2 文档编号:2570323 上传时间:2023-05-04 格式:DOCX 页数:28 大小:36.67KB
下载 相关 举报
VC++ CString的操作.docx_第1页
第1页 / 共28页
VC++ CString的操作.docx_第2页
第2页 / 共28页
VC++ CString的操作.docx_第3页
第3页 / 共28页
VC++ CString的操作.docx_第4页
第4页 / 共28页
VC++ CString的操作.docx_第5页
第5页 / 共28页
VC++ CString的操作.docx_第6页
第6页 / 共28页
VC++ CString的操作.docx_第7页
第7页 / 共28页
VC++ CString的操作.docx_第8页
第8页 / 共28页
VC++ CString的操作.docx_第9页
第9页 / 共28页
VC++ CString的操作.docx_第10页
第10页 / 共28页
VC++ CString的操作.docx_第11页
第11页 / 共28页
VC++ CString的操作.docx_第12页
第12页 / 共28页
VC++ CString的操作.docx_第13页
第13页 / 共28页
VC++ CString的操作.docx_第14页
第14页 / 共28页
VC++ CString的操作.docx_第15页
第15页 / 共28页
VC++ CString的操作.docx_第16页
第16页 / 共28页
VC++ CString的操作.docx_第17页
第17页 / 共28页
VC++ CString的操作.docx_第18页
第18页 / 共28页
VC++ CString的操作.docx_第19页
第19页 / 共28页
VC++ CString的操作.docx_第20页
第20页 / 共28页
亲,该文档总共28页,到这儿已超出免费预览范围,如果喜欢就下载吧!
下载资源
资源描述

VC++ CString的操作.docx

《VC++ CString的操作.docx》由会员分享,可在线阅读,更多相关《VC++ CString的操作.docx(28页珍藏版)》请在冰点文库上搜索。

VC++ CString的操作.docx

VC++CString的操作

vc中的CString的操作

原著:

JosephM.Newcomer

翻译:

littleloach

原文出处:

codeproject:

CStringManagement

通过阅读本文你能学习怎么有效地使用CString。

  CString是一种非常有用的数据类型。

他们非常大程度上简化了MFC中的许多操作,使得MFC在做字符串操作的时候方便了非常多。

不管怎样,使用CString有非常多特别的技巧,特别是对于纯C背景下走出来的程式员来说有点难以学习。

这篇文章就来讨论这些技巧。

  使用CString能让你对字符串的操作更加直截了当。

这篇文章不是CString的完全手册,但囊括了大部分常见基本问题。

这篇文章包括以下内容:

CString对象的连接

格式化字符串(包括int型转化为CString)

CString型转化成int型

CString型和char*类型的相互转化

char*转化成CString

CString转化成char*之一:

使用LPCTSTR强制转化

CString转化成char*之二:

使用CString对象的GetBuffer方法

CString转化成char*之三:

和控件的接口

CString型转化成BSTR型;

BSTR型转化成CString型;

VARIANT型转化成CString型;

载入字符串表资源;

CString和临时对象;

CString的效率;

总结

下面我分别讨论。

1、CString对象的连接

  能体现出CString类型方便性特点的一个方面就字符串的连接,使用CString类型,你能非常方便地连接两个字符串,正如下面的例子:

CStringgray("Gray");

CStringcat("Cat");

CStringgraycat=gray+cat;

要比用下面的方法好得多:

chargray[]="Gray";

charcat[]="Cat";

char*graycat=malloc(strlen(gray)+strlen(cat)+1);

strcpy(graycat,gray);

strcat(graycat,cat);

2、格式化字符串

  和其用sprintf()函数或wsprintf()函数来格式化一个字符串,还不如用CString对象的Format()方法:

CStrings;

s.Format(_T("Thetotalis%d"),total);

  用这种方法的好处是你不用担心用来存放格式化后数据的缓冲区是否足够大,这些工作由CString类替你完成。

  格式化是一种把其他不是字符串类型的数据转化为CString类型的最常用技巧,比如,把一个整数转化成CString类型,可用如下方法:

CStrings;

s.Format(_T("%d"),total);

  我总是对我的字符串使用_T()宏,这是为了让我的代码至少有Unicode的意识,当然,关于Unicode的话题不在这篇文章的讨论范围。

_T()宏在8位字符环境下是如下定义的:

#define_T(x)x//非Unicode版本(non-Unicodeversion)

而在Unicode环境下是如下定义的:

#define_T(x)L##x//Unicode版本(Unicodeversion)

所以在Unicode环境下,他的效果就相当于:

s.Format(L"%d",total);

  如果你认为你的程式可能在Unicode的环境下运行,那么开始在意用Unicode编码。

比如说,不要用sizeof()操作符来获得字符串的长度,因为在Unicode环境下就会有2倍的误差。

我们能用一些方法来隐藏Unicode的一些细节,比如在我需要获得字符长度的时候,我会用一个叫做DIM的宏,这个宏是在我的dim.h文件中定义的,我会在我写的所有程式中都包含这个文件:

#defineDIM(x)(sizeof((x))/sizeof((x)[0]))

  这个宏不仅能用来解决Unicode的字符串长度的问题,也能用在编译时定义的表格上,他能获得表格的项数,如下:

classWhatever{...};

Whateverdata[]={

{...},

...

{...},

};

for(inti=0;i

  这里要提醒你的就是一定要注意那些在参数中需要真实字节数的API函数调用,如果你传递字符个数给他,他将不能正常工作。

如下:

TCHARdata[20];

lstrcpyn(data,longstring,sizeof(data)-1);//WRONG!

lstrcpyn(data,longstring,DIM(data)-1);//RIGHT

WriteFile(f,data,DIM(data),&bytesWritten,NULL);//WRONG!

WriteFile(f,data,sizeof(data),&bytesWritten,NULL);//RIGHT

造成以上原因是因为lstrcpyn需要一个字符个数作为参数,不过WriteFile却需要字节数作为参数。

同样需要注意的是有时候需要写出数据的所有内容。

如果你仅仅只想写出数据的真实长度,你可能会认为你应该这样做:

WriteFile(f,data,lstrlen(data),&bytesWritten,NULL);//WRONG

不过在Unicode环境下,他不会正常工作。

正确的做法应该是这样:

WriteFile(f,data,lstrlen(data)*sizeof(TCHAR),&bytesWritten,NULL);//RIGHT

  因为WriteFile需要的是个以字节为单位的长度。

(可能有些人会想“在非Unicode的环境下运行这行代码,就意味着总是在做一个多余的乘1操作,这样不会降低程式的效率吗?

”这种想法是多余的,你必须要了解编译器实际上做了什么,没有哪一个C或C++编译器会把这种无聊的乘1操作留在代码中。

在Unicode环境下运行的时候,你也不必担心那个乘2操作会降低程式的效率,记住,这只是个左移一位的操作而已,编译器也非常乐意为你做这种替换。

  使用_T宏并不是意味着你已创建了一个Unicode的程式,你只是创建了一个有Unicode意识的程式而已。

如果你在默认的8-bit模式下编译你的程式的话,得到的将是个普通的8-bit的应用程式(这里的8-bit指的只是8位的字符编码,并不是指8位的计算机系统);当你在Unicode环境下编译你的程式时,你才会得到一个Unicode的程式。

记住,CString在Unicode环境下,里面包含的可都是16位的字符哦。

3、CString型转化成int型

  把CString类型的数据转化成整数类型最简单的方法就是使用标准的字符串到整数转换例程。

  虽然通常你怀疑使用_atoi()函数是个好的选择,他也非常少会是个正确的选择。

如果你准备使用Unicode字符,你应该用_ttoi(),他在ANSI编码系统中被编译成_atoi(),而在Unicode编码系统中编译成_wtoi()。

你也能考虑使用_tcstoul()或_tcstol(),他们都能把字符串转化成任意进制的长整数(如二进制、八进制、十进制或十六进制),不同点在于前者转化后的数据是无符号的(unsigned),而后者相反。

看下面的例子:

CStringhex=_T("FAB");

CStringdecimal=_T("4011");

ASSERT(_tcstoul(hex,0,16)==_ttoi(decimal));

4、CString型和char*类型的相互转化

  这是初学者使用CString时最常见的问题。

有了C++的帮助,非常多问题你不必深入的去考虑他,直接拿来用就行了,不过如果你不能深入了解他的运行机制,又会有非常多问题让你迷惑,特别是有些看起来没有问题的代码,却偏偏不能正常工作。

比如,你会奇怪为什么不能写向下面这样的代码呢:

CStringgraycat="Gray"+"Cat";

或这样:

CStringgraycat("Gray"+"Cat");

  事实上,编译器将抱怨上面的这些尝试。

为什么呢?

因为针对CString和LPCTSTR数据类型的各种各样的组合,“+”运算符被定义成一个重载操作符。

而不是两个LPCTSTR数据类型,他是底层数据类型。

你不能对基本数据(如int、char或char*)类型重载C++的运算符。

你能象下面这样做:

CStringgraycat=CString("Gray")+CString("Cat");

或这样:

CStringgraycat=CString("Gray")+"Cat";

研究一番就会发现:

“+”总是使用在至少有一个CString对象和一个LPCSTR的场合。

注意,编写有Unicode意识的代码总是一件好事,比如:

CStringgraycat=CString(_T("Gray"))+_T("Cat");

这将使得你的代码能直接移植。

char*转化为CString

  目前你有一个char*类型的数据,或说一个字符串。

怎么样创建CString对象呢?

这里有一些例子:

char*p="Thisisatest";

或象下面这样更具有Unicode意识:

TCHAR*p=_T("Thisisatest")

LPTSTRp=_T("Thisisatest");

你能使用下面任意一种写法:

CStrings="Thisisatest";//8-bitonly

CStrings=_T("Thisisatest");//Unicode-aware

CStrings("Thisisatest");//8-bitonly

CStrings(_T("Thisisatest"));//Unicode-aware

CStrings=p;

CStrings(p);

  用这些方法能轻松将常量字符串或指针转换成CString。

需要注意的是,字符的赋值总是被拷贝到CString对象中去的,所以你能象下面这样操作:

TCHAR*p=_T("Gray");

CStrings(p);

p=_T("Cat");

s+=p;

结果字符串肯定是“GrayCat”。

CString类更有几个其他的构造函数,不过这里我们不考虑他,如果你有兴趣能自己查看相关文件。

事实上,CString类的构造函数比我展示的要复杂,比如:

CStrings="Thisisatest";

  这是非常草率的编码,不过实际上他在Unicode环境下能编译通过。

他在运行时调用构造函数的MultiByteToWideChar操作将8位字符串转换成16位字符串。

不管怎样,如果char*指针是网络上传输的8位数据,这种转换是非常有用的。

CString转化成char*之一:

强制类型转换为LPCTSTR;

  这是一种略微硬性的转换,有关“正确”的做法,人们在认识上还存在许多混乱,正确的使用方法有非常多,但错误的使用方法可能和正确的使用方法相同多。

  我们首先要了解CString是一种非常特别的C++对象,他里面包含了三个值:

一个指向某个数据缓冲区的指针、一个是该缓冲中有效的字符记数及一个缓冲区长度。

有效字符数的大小能是从0到该缓冲最大长度值减1之间的所有数(因为字符串结尾有一个NULL字符)。

字符记数和缓冲区长度被巧妙隐藏。

  除非你做一些特别的操作,否则你不可能知道给CString对象分配的缓冲区的长度。

这样,即使你获得了该0缓冲的地址,你也无法更改其中的内容,不能截短字符串,也绝对没有办法加长他的内容,否则第一时间就会看到溢出。

  LPCTSTR操作符(或更明确地说就是TCHAR*操作符)在CString类中被重载了,该操作符的定义是返回缓冲区的地址,因此,如果你需要一个指向CString的字符串指针的话,能这样做:

CStrings("GrayCat");

LPCTSTRp=s;

  他能正确地运行。

这是由C语言的强制类型转化规则实现的。

当需要强制类型转化时,C++规测容许这种选择。

比如,你能将(浮点数)定义为将某个复数(有一对浮点数)进行强制类型转换后只返回该复数的第一个浮点数(也就是其实部)。

能象下面这样:

Complexc(1.2f,4.8f);

floatrealpart=c;

如果(float)操作符定义正确的话,那么实部的的值应该是1.2。

  这种强制转化适合所有这种情况,例如,所有带有LPCTSTR类型参数的函数都会强制执行这种转换。

于是,你可能有这样一个函数(也许在某个你买来的DLL中):

BOOLDoSomethingCool(LPCTSTRs);

你象下面这样调用他:

CStringfile("c:

\\myfiles\\coolstuff")

BOOLresult=DoSomethingCool(file);

  他能正确运行。

因为DoSomethingCool函数已说明了需要一个LPCTSTR类型的参数,因此LPCTSTR被应用于该参数,在MFC中就是返回的串地址。

如果你要格式化字符串怎么办呢?

CStringgraycat("GrayCat");

CStrings;

s.Format("Mew!

Ilove%s",graycat);

  注意由于在可变参数列表中的值(在函数说明中是以“...”表示的)并没有隐含一个强制类型转换操作符。

你会得到什么结果呢?

  一个令人惊讶的结果,我们得到的实际结果串是:

"Mew!

IloveGrayCat"。

  因为MFC的设计者们在设计CString数据类型时非常小心,CString类型表达式求值后指向了字符串,所以这里看不到所有象Format或sprintf中的强制类型转换,你仍然能得到正确的行为。

描述CString的附加数据实际上在CString名义地址之后。

  有一件事情你是不能做的,那就是修改字符串。

比如,你可能会尝试用“,”代替“.”(不要做这样的,如果你在乎国际化问题,你应该使用十进制转换的NationalLanguageSupport特性,),下面是个简单的例子:

CStringv("1.00");//货币金额,两位小数

LPCTSTRp=v;

p[lstrlen(p)-3]=’’,’’;

  这时编译器会报错,因为你赋值了一个常量串。

如果你做如下尝试,编译器也会错:

strcat(p,"each");

  因为strcat的第一个参数应该是LPTSTR类型的数据,而你却给了一个LPCTSTR。

  不要试图钻这个错误消息的牛角尖,这只会使你自己陷入麻烦!

  原因是缓冲有一个计数,他是不可存取的(他位于CString地址之下的一个隐藏区域),如果你改动这个串,缓冲中的字符计数不会反映所做的修改。

此外,如果字符串长度恰好是该字符串物理限制的长度(梢后还会讲到这个问题),那么扩展该字符串将改写缓冲以外的所有数据,那是你无权进行写操作的内存(不对吗?

),你会毁换坏不属于你的内存。

这是应用程式真正的死亡处方。

CString转化成char*之二:

使用CString对象的GetBuffer方法;

  如果你需要修改CString中的内容,他有一个特别的方法能使用,那就是GetBuffer,他的作用是返回一个可写的缓冲指针。

如果你只是打算修改字符或截短字符串,你完万能这样做:

CStrings(_T("File.ext"));

LPTSTRp=s.GetBuffer();

LPTSTRdot=strchr(p,’’.’’);//OK,shouldhaveuseds.Find...

if(p!

=NULL)

*p=_T(’’\0’’);

s.ReleaseBuffer();

  这是GetBuffer的第一种用法,也是最简单的一种,不用给他传递参数,他使用默认值0,意思是:

“给我这个字符串的指针,我确保不加长他”。

当你调用ReleaseBuffer时,字符串的实际长度会被重新计算,然后存入CString对象中。

  必须强调一点,在GetBuffer和ReleaseBuffer之间这个范围,一定不能使用你要操作的这个缓冲的CString对象的所有方法。

因为ReleaseBuffer被调用之前,该CString对象的完整性得不到保障。

研究以下代码:

CStrings(...);

LPTSTRp=s.GetBuffer();

//...这个指针p发生了非常多事情

intn=s.GetLength();//非常糟D!

!

!

!

!

有可能给出错误的答案!

!

!

s.TrimRight();//非常糟!

!

!

!

!

不能确保能正常工作!

!

!

!

s.ReleaseBuffer();//目前应该OK

intm=s.GetLength();//这个结果能确保是正确的。

s.TrimRight();//将正常工作。

  假设你想增加字符串的长度,你首先要知道这个字符串可能会有多长,好比是声明字符串数组的时候用:

charbuffer[1024];

表示1024个字符空间足以让你做所有想做得事情。

在CString中和之意义相等的表示法:

LPTSTRp=s.GetBuffer(1024);

  调用这个函数后,你不仅获得了字符串缓冲区的指针,而且同时还获得了长度至少为1024个字符的空间(注意,我说的是“字符”,而不是“字节”,因为CString是以隐含方式感知Unicode的)。

  同时,还应该注意的是,如果你有一个常量串指针,这个串本身的值被存储在只读内存中,如果试图存储他,即使你已调用了GetBuffer,并获得一个只读内存的指针,存入操作会失败,并报告存取错误。

我没有在CString上证实这一点,但我看到过大把的C程式员经常犯这个错误。

  C程式员有一个通病是分配一个固定长度的缓冲,对他进行sprintf操作,然后将他赋值给一个CString:

charbuffer[256];

sprintf(buffer,"%......",args,...);//...部分省略许多细节

CStrings=buffer;

虽然更好的形式能这么做:

CStrings;

s.Format(_T("%...."),args,...);

如果你的字符串长度万一超过256个字符的时候,不会破坏堆栈。

  另外一个常见的错误是:

既然固定大小的内存不工作,那么就采用动态分配字节,这种做法弊端更大:

intlen=lstrlen(parm1)+13lstrlen(parm2)+10+100;

char*buffer=newchar[len];

sprintf(buffer,"%sisequalto%s,validdata",parm1,parm2);

CStrings=buffer;

......

delete[]buffer;

他能能被简单地写成:

CStrings;

s.Format(_T("%sisequalto%s,validdata"),parm1,parm2);

  需要注意sprintf例子都不是Unicode就绪的,尽管你能使用tsprintf及用_T()来包围格式化字符串,不过基本思路仍然是在走弯路,这这样非常容易出错。

CStringtochar*之三:

和控件的接口;

  我们经常需要把一个CString的值传递给一个控件,比如,CTreeCtrl。

MFC为我们提供了非常多便利来重载这个操作,不过在大多数情况下,你使用“原始”形式的更新,因此需要将墨某个串指针存储到TVINSERTITEMSTRUCT结构的TVITEM成员中。

如下:

TVINSERTITEMSTRUCTtvi;

CStrings;

//...为s赋一些值。

tvi.item.pszText=s;//Compileryellsatyouhere

//...填写tvi的其他域

HTREEITEMti=c_MyTree.InsertItem(&tvi);

  为什么编译器会报错呢?

明明看起来非常完美的用法啊!

不过事实上如果你看看TVITEM结构的定义你就会明白,在TVITEM结构中pszText 成员的声明如下:

LPTSTRpszText;

intcchTextMax;

  因此,赋值不是赋给一个LPCTSTR类型的变量,而且编译器无法知道怎么将赋值语句右边强制转换成LPCTSTR。

好吧,你说,那我就改成这样:

tvi.item.pszText=(LPCTSTR)s;//编译器依然会报错。

  编译器之所以依然报错是因为你试图把一个LPCTSTR类型的变量赋值给一个LPTSTR类型的变量,这种操作在C或C++中是被禁止的。

你不能用这种方法来滥用常量指针和非常量指针概念,否则,会扰乱编译器的优化机制,使之不知怎么优化你的程式。

比如,如果你这么做:

constinti=...;

//...dolotsofstuff

...=a[i];//usage1

//...lotsmorestuff

...=a[i];//usage2

  那么,编译器会以为既然i是const,所以usage1和usage2的值是相同的,并且他甚至能事先计算好usage1处的a[i]的地址,然后保留着在后面的usage2处使用,而不是重新计算。

如果你按如下方式写的话:

constinti=...;

int*p=&i;

//...dolotsofstuff

...=a[i];//usage1

//...lotsmorestuff

(*p)++;//messovercompiler’’sassumption

//...andotherstuff

...=a[i];//usage2

  编译器将认为i是常量,从而a[i]的位置也是常量,这

展开阅读全文
相关资源
猜你喜欢
相关搜索
资源标签

当前位置:首页 > 解决方案 > 学习计划

copyright@ 2008-2023 冰点文库 网站版权所有

经营许可证编号:鄂ICP备19020893号-2