您的位置:首页 > 其它

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

2011-06-26 19:37 288 查看
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字符串占用的内存块
int WideCharToMultiByte(

  UINT CodePage,            // code page
  DWORD dwFlags,            // performance and mapping flags
  LPCWSTR lpWideCharStr,    // wide-character string
  int cchWideChar,          // number of chars in string
  LPSTR lpMultiByteStr,     // buffer for new string
  int cbMultiByte,          // size of buffer
  LPCSTR lpDefaultChar,     // default for unmappable chars
  LPBOOL lpUsedDefaultChar  // set when default char used
);
其步骤与MultiByteToWideChar大致相同,只是在第二步不需要乘以sizeof(wchar_t)
注意:该函数的后两个参数,只有一个字符在uCodePage中指定代码页中没有对应的表示时才使用.在遇到一个不能转换的宽字符时,pDefaultChar用来指定字符(为NULL时,函数就会使用一个系统默认的字符,通常是?).
pfUsedDefaultChar指向一个布尔变量;在宽字符串中,如果至少有一个字符不能转换为对应的多字节的形式,则变量为TRUE,如果所有字符都能成功转换则设为FALSE(传入NULL).
9. BOOL IsTextUnicode(

  CONST VOID* pBuffer, // input buffer to be examined

  int cb,                // size of input buffer

  PINT      pResult               // options

);

包含的dll----AdvApi32.dll
包含.h-------WinBase.h
参数:pvBuffer,是一个void指针.
      cb表示字节数而非字符数,我们可以测试某部分字符就行了
      pResult:整数地址.指定函数执行哪些测试.如果为NULL,函数将执行他能执行的每一项测试.
返回值:是Unicode返回TRUE.如果指定了具体的测试项目,那么函数在返回之前,还会shehi此整数中的相应位,一反应每个测试项目的结果.
注意此函数并不精确.
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: