第二章:字符串和字符串处理

  1. char(表示8位ANSI),wchar_t(表示16位的Unicode字符,而且由于早期版本的编译器没有提供这个内建的数据类型,所以编译器只有在指定了/zc:wchar_t编译开关时才会定义这个数据类型,默认是指定的.)
  说明:在编译器内建对wchar_t的支持之前,有一个C头文件定义了一个wchar_t数据类型,如下所示: typedef unsigned short wchar_t;
  wchar_t c=L'A';//L表示在编译器该字符串应该编译为Unicode字符串.
  2. 任何修改字符串的函数都存在一个安全隐患:如果目标字符串的缓冲区不够大,无法容纳所生成的字符串,就会导致内存中的数据被破坏.在应用程序中包含strSafe.h时,string.h也会包含进来.C运行库中现有的字符串处理函数(如_tcscpy宏后的那些函数)已经标记为废弃不用.如果使用了这些函数,编译器就会发出警告(注意:必须在包含其他所有文件之后才包含StrSafe.h),每个函数都有一个对应的新版本函数.前面的名称相同,但是在最后添加_s(例如:_tcscpy相对应的_tcscpy_s)
  3. C运行时实际上允许提供自己的函数,这样一来,在他检测到了一个无效参数时就会调用此函数.然后在函数中记录失败,附上一个调试器,或者做其他我们想做的事情.为了启动这个功能,必须先定义好一个函数,原型如下:
  void InvalidParameterHandler(PCTSTR expression,PCTSTR Function,
  PCTSTR file,unsigned int line,uintptr_t/*pReserved*/)
  其中:expression描述了C运行时实现代码中可能出现的函数调用失败.Function、flie、line分别描述了出现错误的函数名称、源代码文件和源代码行号.
  然后调用_set_invalid_parameter_handler来注册这个处理程序.
  最后在应用程序的开始调用_CrtSetReportMode(_CRT_ASSERT,0);从而禁止可能有C运行时触发的所有Assertion对话框.
  4. C运行库新增了一些用户在执行字符串处理时提供更多控制的函数.其返回值为HRESULT. HRESULT值 描述 S_OK 成功,目标缓冲区中包含源字符串,并以\0结尾 STRSAFE_INVALID_PARAMETER 失败,将NULL传给了一个参数 STRSAFE_INSUFFICIENT_BUFFER 失败,指定目标缓冲区太小,无法容纳整个源字符串 _countof宏来获取字符数,sizeof获取字节数.
  5. 我们如果要比较字符串是否相等或者进行排序.较好的函数是:CompareString(Ex)和CompareStringOrdinal.对于需要以符合用户语言习惯的方式向用户显示的字符串更是如此.
  int CompareString( LCID locale,// 指定区域设置ID,使用GetThreadLocale()来获取
  DWORD dwCmdFlags,//标志位,用于修改函数在比较字符串时采用的方法
  PCTSTR pString1,//字符串1
  int cch1,//字符串1的字符数
  PCTSTR pString2,//字符串2
  int cch2)//字符串2的个数
  其中dwCmdFlags的取值有: 标志 含义 NORM_IGNORECASE LINGUISTIC_IGNORECASE 忽略大小写 NORM_IGNOREKANATYPE 不区分平假名和片假名字符 NORM_IGNORENONSPACE LINGUISTIC_IGNOREDIACRITIC 忽略non-spacing字符 NORM_IGNORESYMBOLS 忽略符号 NORM_IGNOREWIDTH 不区分同一字符串的单字节和双字节形式 SORT_STRINGSORT 将标点符号当作符号来处理 为了比较程序内部所用的字符串(如路径名、注册表项值、xml元素/属性等),应该使用CompareStringOrdinal(PCWSTR pString1,
  int cchCount1,
  PCWSTR pString2,
  int cchCount2,
  BOOL reCase)//由于这个函数执行的是码位比较,不考虑区域设置,所以速度很快.
  以上两个函数的返回值:0:表示函数调用失败,
  返回值为CSTR_LESS_THAN(定义为1)表明pString1小于pString2
  返回值为CSTR_EQUAL(定义为2)表明pString1等于pString2
  返回值为CSTR_GREATER_THAN(定义为3)表明pString1大于pString.
  为了方便起见,如果函数成功,我们可以从返回值中减去2,从而使得结果与C运行库函数的结果值(-1,0,1)保持一致.
  6. 应用程序字符处理遵循的原则如下:
  ■开始将文本字符串想象为字符的数组,而不是char或者字节的数组
  ■用通用数据类型(如TCHAR/PTSTR)来表示文本字符和字符串
  ■用明确的数据类型(如BYTE和PBYTE)来表示字节,字节指针和数据缓冲区
  ■用TEXT或_T宏来表示字面量字符和字符串,但为了保存一致性和更好的可读性,请避免两者混用.
  ■执行全局替换(用PTSTR替换PSTR)
  ■修改与字符串相关的计算.可以使用宏来避免犯错:
  #define chmalloc(nCharactes) (TCHAR*) malloc(nCharacters * sizeof(TCHAR)) ■ 避免使用printf系列函数,尤其是不要用%s和%S字段类型来进行ANSI与Unicode字符串之间的相互转换.正确的做法是使用MultiByteToWideChar和WideCharToMultiByte函数.
  ■ Unicode和_UNICODE符号要么同时指定,要么都不指定. 7. 对于字符串处理函数,应该遵循以下基本准则:
  ■始终使用安全的字符串处理函数,比如后缀为_s的函数.或者使用前缀为StringCch的函数.后者主要在我们想明确控制截断的时候使用.
  ■不要使用不安全的C运行库字符串处理函数.
  ■利用/GS和/RTCs编译器标志来自动检测缓冲区溢出.
  ■不要使用Kernel32方法来进行字符串处理(例lstrcat)
  ■在应用程序的代码中,需要要比较两种字符串.其中程序字符串包含文件名,路径,XML元素/属性以及注册表项/值等.对于这些字符串,应使用CompareStringOrdinal来进行比较.用户字符串则一般要在用户界面上显示,对于这些字符串,应使用CompareString(Ex)来比较.
  8. Unicode与ANSI字符串转换函数:
  int MultiByteToWideChar(
  UINT uCodePage,//标识了与多字节字符串关联的一个代码页值
  DWORD dwFlags,//允许我们进行额外的控制,它会影响带变音符号.一般为0
  PCSTR pMultiByteStr,//指定要转换的字符串
  int cbMultiByte,//指定字符串的长度(字节数),如果为-1,则自动判断源字符串的长度
  PWSTR pWideCharStr,//写入内存缓冲区
  int cchWideChar);//缓冲区字符长度(字符数),如果传递0,则函数不会执行//转换,而是返回一个宽字符数,(包括终止字符'\0')只有当缓冲区能够容纳该数量的宽字符时转换才成功
  一般的步骤如下: ① 调用MultiByteToWideChar为pWideCharStr参数传入NULL,为cchWideChar参数传入0,为cbMultiByte传入-1
  ② 分配一块足以容纳转化后Unicode字符的内存,大小是上次调用函数的返回值乘以sizeof(wchar_t)
  ③ 再次调用MultiByteToWideChar,这一次将缓冲区的地址作为pWideCharStr参数的值传入,将第一次得到的大小作为cchWideChar参数传入
  ④ 使用转换后的字符串
  ⑤ 释放Unicode字符串占用的内存块 UINTCodePage, // code page DWORDdwFlags, // performance and mapping flags LPCWSTRlpWideCharStr, // wide-character string intcchWideChar, // number of chars in string LPSTRlpMultiByteStr, // buffer for new string intcbMultiByte, // size of buffer LPCSTRlpDefaultChar, // default for unmappable chars LPBOOLlpUsedDefaultChar // set when default char used ); 其步骤与MultiByteToWideChar大致相同,只是在第二步不需要乘以sizeof(wchar_t)
  注意:该函数的后两个参数,只有一个字符在uCodePage中指定代码页中没有对应的表示时才使用.在遇到一个不能转换的宽字符时,pDefaultChar用来指定字符(为NULL时,函数就会使用一个系统默认的字符,通常是?).
  pfUsedDefaultChar指向一个布尔变量;在宽字符串中,如果至少有一个字符不能转换为对应的多字节的形式,则变量为TRUE,如果所有字符都能成功转换则设为FALSE(传入NULL).
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值