CString


CString

文章插图
CStringCString对象可以被认为是字元数组. 将CString看作实际的字元串,而不是指向字元串的指针.
基本介绍中文名:字元数组
外文名:CString
套用学科:计算机软体
软体语言:Visual C++
适用系统:MFC ATL
属性:数据类型定义
前言CString位于头档案afx.h中 。这篇文章就来讨论这些技巧 。使用CString可以让你对字元串的操作更加直截了当 。这篇文章不是CString的完全手册,但囊括了大部分常见基本问题 。这篇文章包括以下内容:CString 对象的连线格式化字元串(包括 int 型转化为 CString)CString类的成员函式CString 型转化成 int 型CString 型和 char* 类型的相互转化char* 转化成 CStringCString 转化成 char* 之一:使用LPCTSTR强制转化CString 转化成 char* 之二:使用CString对象的GetBuffer方法CString 转化成 char* 之三: 和控制项的接口CString 型转化成BSTR型;BSTR 型转化成 CString 型;VARIANT型转化成 CString 型;载入字元串表资源;CString 和临时对象;CString 的效率;对象连线能体现出 CString 类型方便性特点的一个方面就是字元串的连线,使用 CString 类型,你能很方便地连线两个字元串,正如下面的例子:CString gray("Gray");CString cat("Cat");CString graycat = gray + cat;//then graycat="GrayCat"要比用下面的方法好得多:char gray[] = "Gray";char cat[] = "Cat";char * graycat =(char*)malloc(strlen(gray) + strlen(cat) + 1);strcpy(graycat,gray);strcat(graycat,cat);字元串与其用 sprintf() 函式或 wsprintf() 函式来格式化一个字元串,还不如用 CString 对象的Format()方法:CString s;s.Format(_T("The total is %d"),total);用这种方法的好处是你不用担心用来存放格式化后数据的缓冲区是否足够大,这些工作由CString类替你完成 。格式化是一种把其它不是字元串类型的数据转化为CString类型的最常用技巧,比如,把一个整数转化成CString类型,可用如下方法:CString s;s.Format(_T("%d"),total);我总是对我的字元串使用_T()宏,这是为了让我的代码至少有Unicode的意识,当然,关于Unicode的话题不在这篇文章的讨论範围 。_T()宏在8位字元环境下是如下定义的:#define _T(x) x // 非Unicode版本(non-Unicode version)而在Unicode环境下是如下定义的:#define _T(x) L##x // Unicode版本(Unicode version)所以在Unicode环境下,它的效果就相当于:s.Format(L"%d",total);如果你认为你的程式可能在Unicode的环境下运行,那幺开始在意用 Unicode 编码 。比如说,不要用 sizeof() 操作符来获得字元串的长度,因为在Unicode环境下就会有2倍的误差 。我们可以用一些方法来隐藏Unicode的一些细节,比如在我需要获得字元长度的时候,我会用一个叫做DIM的宏,这个宏是在我的dim.h档案中定义的,我会在我写的所有程式中都包含这个档案:#define DIM(x) (sizeof((x)) / sizeof((x)[0]))这个宏不仅可以用来解决Unicode的字元串长度的问题,也可以用在编译时定义的表格上,它可以获得表格的项数,如下:class Whatever { ... };Whatever data[] = {{ ... },...{ ... },};for(int i = 0; i < DIM(data); i++) // 扫描表格寻找匹配项 。这里要提醒你的就是一定要注意那些在参数中需要真实位元组数的API函式调用,如果你传递字元个数给它,它将不能正常工作 。如下:TCHAR data[20];lstrcpyn(data,longstring,sizeof(data) - 1); // WRONG!lstrcpyn(data,longstring,DIM(data) - 1); // RIGHTWriteFile(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位的字元喔 。成员函式1) CString类的构造函式CString类有很多构造函式,这里只介绍几个比较常用的:CString(const CString& stringSrc);将一个已经存在的CString对象stringSrc的内容拷贝到该CString对象 。例如:CString str1(_T(jizhuomi)); // 将常量字元串拷贝到str1CString str2(str1); // 将str1的内容拷贝到str2CString(LPCTSTR lpch,int nLength);将字元串lpch中的前nLength个字元拷贝到该CString对象 。例如:CString str(_T("wwwjizhuomi"),3); // 构造的字元串对象内容为"www"CString(TCHAR ch,int nLength = 1);使用此函式构造的CString对象中将含有nLength个重複的ch字元 。例如:CString str(_T('w'),3); // str为"www"2)CString类的大小写转换及顺序转换函式CString& MakeLower(); 将字元串中的所有大写字元转换为小写字元 。CString& MakeUpper(); 将字元串中的所有小写字元转换为大写字元 。CString& MakeReverse(); 将字元串中所有字元的顺序颠倒 。例如:CString str(_T("JiZhuoMi"));str.MakeLower(); // str为"jizhuomi"str.MakeUpper(); // str为"JIZHUOMI"str.MakeReverse(); // str为"IMOUHZIJ"3)CString对象的连线多个CString对象的连线可以通过重载运算符+、+=实现 。例如:CString str(_T("jizhuomi")); // str内容为"jizhuomi"str = _T("www") + str + _T("-"); // str为"wwwjizhuomi-"str += _T("com"); // str为wwwjizhuomi-com4)CString对象的比较CString对象的比较可以通过==、!=、<;、>;、<=、>=等重载运算符实现,也可以使用Compare和CompareNoCase成员函式实现 。int Compare(PCXSTR psz) const; 将该CString对象与psz字元串比较,如果相等则返回0,如果小于psz则返回值小于0,如果大于psz则返回值大于0 。int CompareNoCase(PCXSTR psz) const throw();此函式与Compare功能类似,只是不区分大小写 。例如:CString str1 = _T("JiZhuoMi");CString str2 = _T("jizhuomi");if (str1 == str2){// 因为str1、str2不相等,所以不执行下面的代码...}if (0 == str1.CompareNoCase(str2)){// 因为不区分大小写比较时,CompareNoCase函式返回0,所以执行下面的代码...}