MFC中CString操作指南Word格式.docx
《MFC中CString操作指南Word格式.docx》由会员分享,可在线阅读,更多相关《MFC中CString操作指南Word格式.docx(25页珍藏版)》请在冰豆网上搜索。
12(CStringSpanExcluding(LPCTSTRlpszCharSet)const;
从字符串中提取不包含在指定字符集lpszCharSet中的字符的子串。
13(MakeUpper
将字符串中所有的字符全部转化成大写形式。
14(MakeLower
将字符串中所有的字符全部转化成小写形式。
15(MakeReverse
将字符串倒置,即将字符的顺序颠倒,第一个字符变成最后一个字符。
16(intReplace(TCHARchOld,TCHARchNew);
intReplace(LPCTSTRlpszOld,LPCTSTRlpszNew);
将字符串中的字符chOld或字符子串lpszOld替换成新的字符chNew或字符串lpszNew。
17.voidCString:
:
TrimLeft(TCHARchTarget);
voidCString:
TrimLeft(LPCTSTRlpszTargets);
删除字符串左边开头的字符或字符子串,参数缺省时删除左边的空格。
18(voidCString:
TrimRight(TCHARchTarget);
TrimRight(LPCTSTRlpszTargets);
删除字符串右边开头的字符或字符子串,参数缺省时删除右边的空格。
19(Find
原型:
intFind(TCHARch)const;
intFind(LPCTSTRlpszSub)const;
intFind(TCHARch,intnStart)const;
intFind(LPCTSTRpstr,intnStart)const;
在字符串中查找指定的字符或字符串。
参数ch为要查找的字符;
lpszSub为要查找的字符子串;
nStart指定查找的起始位置,如缺省为从最左边开始;
pstr指向欲查找子串的指针。
20(ReverseFind
intReverseFind(TCHARch)const;
返回字符串中最后一个和指定的字符匹配的字符的下标。
21(FindOneOf
intFindOneOf(LPCTSTRlpszCharSet)const;
在字符串中查找第一个和指定的字符集lpszCharSet中任何一个字符匹配的字符的位置。
22(比较运算符
CString类重载了―==‖、―!
=‖、―>
‖、―<
=‖、―<
‖等比较运算符,可用于两个字符串间的各种比较运算,比较时区分大小写。
23(Compare
原形:
intCompare(LPCTSTRlpsz)const;
比较两个字符串,如果两个字符串相等,返回值等于0;
如果本对象大于参数字符串,返回值大于0;
如果本对象小于参数字符串,返回值小于0,比较时区分大小写。
24(CompareNoCase
intCompareNoCase(LPCTSTRlpsz)const;
与Compare函数类似,不同的是在忽略字符大小写的情况下比较两个字符串。
25(operator=
将一个新的值赋予CString对象。
26(operator+
将两个字符串合并成一个新的字符串。
在两个参数中必须有一个是CString类型的,而另一个参数可以是字符、字符指针或CString类型对象。
27(operator+=
在一个字符串的后面再添加一个字符串或一个字符。
二、例题
例1:
连结字符串
CStringm_str1="
工作"
;
CStringm_str2="
正常"
CStringm_str3=m_str1+m_str2;
执行第三行后,m_str3的值应该是―工作正常‖。
例2:
比较字符串
a"
b"
intresult=m_str1.Compare(m_str2);
if(result=0)
AfxMessageBox("
两者相同"
;
elseif(result>
0)
m_str1大于m_str2"
else
m_str1小于m_str2"
运行结果在信息框中显示―m_str1小于m_str2‖。
两个字符串比较大小时从第一个字
母开始,按照对应的ASCII值比较。
如果第一个字母相同,再比较下一个字母。
依
次往下直到比较出大小为止。
例3:
提取字符串
abcde"
CStringm_str2=m_str1.Left
(1)+m_str1.Mid(2,1)+m_str1.Right
(1);
AfxMessageBox(m_str2);
运行结果在信息框中显示m_str2的内容―ace‖。
m_str1.Left
(1)得到m_str1的最左
边的一个字符―a‖,m_str1.Mid(2,1)从m_str1中取得从索引为2开始的一个字符
―c‖,m_str1.Right
(1)得到m_str1的最右边的一个字符―e‖。
例4:
查找字符串
abcdef"
deq"
intindex=m_str1.Find(m_str2);
if(index>
=0)
{
chars[10];
wsprintf(s,"
匹配字符的下标为%d"
index);
MessageBox(s);
}
MessageBox("
没有匹配字符"
运行结果在信息框中显示―没有匹配字符‖。
m_str1.Find(m_str2)在m_str1中查找
m_str2的内容―deq‖,由于未找到返回–1。
例5:
变换字符串
CStringm_str="
ABCabc"
m_str.TrimLeft();
m_str.TrimRight();
m_str.MakeUpper();
MessageBox(m_str);
运行结果在信息框中显示―ABCABC‖。
m_str.TrimLeft()和m_str.TrimRight()
分别去掉m_str左边和右边的空格,m_str.MakeUpper()将m_str中的所有字母转
换成大写。
例6:
转化为字符串
把CString转化为LPTSTR和LPCTSTR可分别用如下方法
(LPTSTR)(LPCTSTR)(_T(CString))
(LPCTSTR)(_T(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[]="
charcat[]="
char*graycat=malloc(strlen(gray)+strlen(cat)+1);
strcpy(graycat,gray);
strcat(graycat,cat);
2、格式化字符串
与其用swsprintf()函数来格式化一个字符串,还不如用CStringprintf()函数或
对象的Format()方法:
CStrings;
s.Format(_T("
Thetotalis%d"
),total);
用这种方法的好处是你不用担心用来存放格式化后数据的缓冲区是否足够大,
这些工作由CString类替你完成。
格式化是一种把其它不是字符串类型的数据转化为CString类型的最常用技
巧,比如,把一个整数转化成CString类型,可用如下方法:
%d"
我总是对我的字符串使用_T()宏,这是为了让我的代码至少有Unicode的意
识,当然,关于Unicode的话题不在这篇文章的讨论范围。
_T()宏在8位字符环境
下是如下定义的:
#define_T(x)x//非Unicode版本(non-Unicodeversion)而在Unicode环境下是如下定义的:
#define_T(x)L##x//Unicode版本(Unicodeversion)所以在Unicode环境下,它的效果就相当于:
s.Format(L"
total);
如果你认为你的程序可能在Unicode的环境下运行,那么开始在意用Unicode编码。
比如说,不要用sizeof()操作符来获得字符串的长度,因为在Unicode环境下就会有2倍的误差。
我们可以用一些方法来隐藏Unicode的一些细节,比如在我需要获得字符长度的时候,我会用一个叫做DIM的宏,这个宏是在我的dim.h文件中定义的,我会在我写的所有程序中都包含这个文件:
#defineDIM(x)(sizeof((x))/sizeof((x)[0]))
这个宏不仅可以用来解决Unicode的字符串长度的问题,也可以用在编译时定义的表格上,它可以获得表格的项数,如下:
classWhatever{...};
Whateverdata[]={
{...},
...
};
for(inti=0;
i<
DIM(data);
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);
WriteFile(f,data,sizeof(data),&
//RIGHT造成以上原因是因为lstrcpyn需要一个字符个数作为参数,但是WriteFile却需要字节数作为参数。
同样需要注意的是有时候需要写出数据的所有内容。
如果你仅仅只想写出数据的真实长度,你可能会认为你应该这样做:
WriteFile(f,data,lstrlen(data),&
//WRONG但是在Unicode环境下,它不会正常工作。
正确的做法应该是这样:
WriteFile(f,data,lstrlen(data)*sizeof(TCHAR),&
因为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="
+"
或者这样:
CStringgraycat("
事实上,编译器将抱怨上面的这些尝试。
为什么呢,因为针对CString和LPCTSTR数据类型的各种各样的组合,―+‖运算符被定义成一个重载操作符。
而不是两个LPCTSTR数据类型,它是底层数据类型。
你不能对基本数据(如int、char或者char*)类型重载C++的运算符。
你可以象下面这样做:
CStringgraycat=CString("
)+CString("
)+"
研究一番就会发现:
―+‖总是使用在至少有一个CString对象和一个LPCSTR的场合。
注意,编写有Unicode意识的代码总是一件好事,比如:
CStringgraycat=CString(_T("
))+_T("
这将使得你的代码可以直接移植。
char*转化为CString
现在你有一个char*类型的数据,或者说一个字符串。
怎么样创建CString对象呢,这里有一些例子:
char*p="
Thisisatest"
或者象下面这样更具有Unicode意识:
TCHAR*p=_T("
或
LPTSTRp=_T("
你可以使用下面任意一种写法:
CStrings="
//8-bitonly
CStrings=_T("
//Unicode-aware
CStrings("
CStrings(_T("
));
CStrings=p;
CStrings(p);
用这些方法可以轻松将常量字符串或指针转换成CString。
需要注意的是,字符的赋值总是被拷贝到CString对象中去的,所以你可以象下面这样操作:
p=_T("
s+=p;
结果字符串肯定是―GrayCat‖。
CString类还有几个其它的构造函数,但是这里我们不考虑它,如果你有兴趣可以自己查看相关文档。
事实上,CString类的构造函数比我展示的要复杂,比如:
这是很草率的编码,但是实际上它在Unicode环境下能编译通过。
它在运行时调用构造函数的MultiByteToWideChar操作将8位字符串转换成16位字符串。
不管怎样,如果char*指针是网络上传输的8位数据,这种转换是很有用的。
强制类型转换为LPCTSTR;
这是一种略微硬性的转换,有关―正确‖的做法,人们在认识上还存在许多混乱,正确的使用方法有很多,但错误的使用方法可能与正确的使用方法一样多。
我们首先要了解CString是一种很特殊的C++对象,它里面包含了三个值:
一个指向某个数据缓冲区的指针、一个是该缓冲中有效的字符记数以及一个缓冲区长度。
有效字符数的大小可以是从0到该缓冲最大长度值减1之间的任何数(因为字符串结尾有一个NULL字符)。
字符记数和缓冲区长度被巧妙隐藏。
除非你做一些特殊的操作,否则你不可能知道给CString对象分配的缓冲区的长度。
这样,即使你获得了该0缓冲的地址,你也无法更改其中的内容,不能截短字符串,也绝对没有办法加长它的内容,否则第一时间就会看到溢出。
LPCTSTR操作符(或者更明确地说就是TCHAR*操作符)在CString类中被重载了,该操作符的定义是返回缓冲区的地址,因此,如果你需要一个指向CString的字符串指针的话,可以这样做:
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中就是返回的串地址。
如果你要格式化字符串怎么办呢,
s.Format("
Mew!
Ilove%s"
graycat);
注意由于在可变参数列表中的值(在函数说明中是以―...‖表示的)并没有隐含一个强制类型转换操作符。
你会得到什么结果呢,
一个令人惊讶的结果,我们得到的实际结果串是:
"
Il