`
sabolasi
  • 浏览: 906988 次
文章分类
社区版块
存档分类
最新评论

从char/wchar_t到TCHAR

 
阅读更多

一.ANSIUNICODE

1.为什么要使用UNICODE

1) 可以很容易地在不同语言之间进行数据交换。

2) 使你能够分配支持所有语言的单个二进制.exe文件或DLL文件。

3) 提高应用程序的运行效率。

Windows 2000是使用UNICODE从头进行开发的,如果调用任何一个Windows函数并给它传递一个ANSI字符串,那么系统首先要将字符串转换成UNICODE,然后将UNICODE字符串传递给操作系统。如果希望函数返回ANSI字符串,系统就会首先将UNICODE字符串转换成ANSI字符串,然后将结果返回给你的应用程序。进行这些字符串的转换需要占用系统的时间和内存。通过从头开始用UNICODE来开发应用程序,就能够使你的应用程序更加有效地运行。

Windows 98只支持ANSI,只能为开发ANSI应用程序。 Windows CE 就是使用UNICODE的操作系统,完全不支持ANSI版函数。

MicrosoftCOMWin16转换成Win32时,所有COM接口方法都只能接受UNICODE字符串。

2ANSI字符和UNICODE字符

ANSI字符类型为CHAR,指向字符串的指针PSTR(LPSTR),指向一个常数字符串的指针PCSTR(LPCSTR);对应的Windows定义的UNICODE字符类型为WCHARtypedef WCHAR wchar_t),指向UNICODE字符串的指针PWSTR ,指向一个常数UNICODE字符串的指针PCWSTR

ANSI "ANSI";

UNICODE L"UNICODE";

ANSI/UNICODE _T("string") or _TEXT("string");

3ANSI字符和UNICODE字符串的操作

双字节(DBCS)字符集中,字符串的每个字符可以包含一个或两个字节。如果只是调用strlen()函数,那么你就无法知道字符串到底有多少个字符,它只能告诉你到达结尾的0之前有多少个字节。

标准c中的strcpy,strchr,strcat等只能用于ANSI字符串,不能正确处理UNICODE字符串,因此也提供了一组补充函数,功能等价,但用于UNICODE码。我们来看看string .h字符串头文件中是怎样处理char*wchar_t*两个字符串版本的:

// …/Microsoft Visual Studio 8/VC/include/string.h

char *strcat(char*,const char*);

wchar_t *wcschr(wchar_t*, const wchar_t*);

类似的还有strchr/wcschrstrcmp/wcscmpstrlen/wcslen etc. ANSI 操作函数以str开头 strcpy UNICODE 操作函数以wcs开头 wcscpy

MBCS 操作函数以_mbs开头 _mbscpy

ANSI/UNICODE 操作函数以_tcs开头 _tcscpyC运行期库)

ANSI/UNICODE 操作函数以lstr开头 lstrcpyWindows API-KERNEL32.DLL

所有新的和未过时的函数在Windows2000中都同时拥有ANSIUNICODE两个版本。ANSI版本函数结尾以A表示;UNICODE版本函数结尾以W表示。操作系统根据是否定义了UNICODE宏来调用合适版本的API

Windows XP默认字符集为gbk(包含gb2312)。wchar_t wcs[] = {0xCD, 0xCD, 0xEE, 0xFE, 0x00};表示汉字“屯铪”。参考在线汉字编码查询

二.ANSI/UNICODE通用字符/字符串类型TCHAR/LPTSTR/LPCTSTR

<1>_UNICODE宏用于C运行期头文件(如下_tprintf示例),UNICODE宏则用于Windows头文件(如上winnt.h),当编译代码模块时,通常必须同时定义这两个宏或都不定义。

winnt.h中定义了TEXT__TEXT)宏,用于做UNICODE环境的自适应。对于非UNICODE环境,__TEXT宏不对参数quote做任何处理;对于UNICODE环境,__TEXT宏在字符串quote前添加L宏,以表示宽字符串(wchar_t[])

// #include <winnt.h>

typedef CHAR char;

typedef WCHAR wchar_t;

#ifdef UNICODE // r_winnt

typedef WCHAR TCHAR, *PTCHAR;

typedef WCHAR *LPWSTR;

typedef LPWSTR LPTSTR;

typedef CONST WCHAR *LPCWSTR;

typedef LPCWSTR LPCTSTR;

#define __TEXT(quote) L##quote // r_winnt

#else /* UNICODE */ // r_winnt

typedef CHAR TCHAR, *PTCHAR;

typedef CHAR *LPSTR;

typedef LPSTR LPTSTR;

typedef CONST CHAR *LPCSTR,

typedef LPCSTR LPCTSTR;

#define __TEXT(quote) quote

#endif /* UNICODE */ // r_winnt

#define TEXT(quote) __TEXT(quote) // r_winnt

MicrosoftVC提供了tchar.h头文件用于字符集的自适应(To be used for compatibility between single-byte, multi-byte and Unicode text models)。对常用的函数做了T定义,例如_tprintf

// #include <tchar.h>

#ifdef _UNICODE

#define __T(x) L ## x

typedef wchar_t TCHAR;

#define _tprintf wprintf

#else /* ndef _UNICODE */

#define __T(x) x

#define _tprintf printf

#endif /* _UNICODE */

#define _T(x) __T(x)

#define _TEXT(x) __T(x)

<2>如果定义了_UNICODE,若要生成一个UNICODE字符串,字符串前要加L宏,用于告诉编译器该字符串应该作为UNICODE字符串来编译处理。但是这样又有个问题就是如果没有定义_UNICODE则编译出错。为了解决这个问题我们必须用到_T(_TEXT)宏,同winnt.h中定义的TEXT__TEXT)宏。使用该宏后,无论源文件有没有定义_UNICODE都不会出现编译错误。

<3>ANSIUNICODE的一些互操作QA

Q1:如何判断一个文本文件是ANSI还是Unicode

A1:如果文本文件的开头两个字节是0xFF0xFE,那么就是Unicode,否则是ANSI

Q2:如何判断一段字符串是ANSI还是Unicode

A2:用IsTextUnicode进行判断。IsTextUnicode使用一系列统计方法和定性方法,以便猜测缓存的内容。由于这不是一种确切的科学方法,因此 IsTextUnicode有可能返回不正确的结果。

Q3:如何在UnicodeANSI之间转换字符串?

A3Windows函数MultiByteToWideChar/mbstowcs函数用于将多字节字符串转换成宽字符串,函数WideCharToMultiByte/wcstombs将宽字符串转换成等价的多字节字符串。

三.ANSI/UNICODE字符串通用函数lstrcpy/lstrcmp/lstrcat/lstrlen

ntdll.dll中实现了许多CRT基本函数:

strlen/wcslenstrcpy/wcscpystrncpy/wcsncpystrcat/wcscatstrncat/wcsncatstrcmp/wcscmpstrncmp/wcsncmpstrchr/wcschrstrrchr/wcsrchrstrstr/wcsstrsprintf/swprintfstrtol/wcstolstrtoul/wcstoul

kernel32.dll(依赖ntdll.dll)中实现了:

lstrlen(lstrlenA/lstrlenW)lstrcpy(lstrcpyA/lstrcpyW)lstrcpyn(lstrcpynA/lstrcpynW)lstrcat(lstrcatA/lstrcatW)lstrcmp(lstrcmpA/lstrcmpW)lstrcmpi(lstrcmpiA/lstrcmpiW)

// /Microsoft Visual Studio 8/VC/PlatformSDK/Include/Winbase.h(已包含在windows.h中)

WINBASEAPI

LPSTR

WINAPI

lstrcpyA(

__out LPSTR lpString1,

__in LPCSTR lpString2

);

WINBASEAPI

LPWSTR

WINAPI

lstrcpyW(

__out LPWSTR lpString1,

__in LPCWSTR lpString2

);

#ifdef UNICODE

#define lstrcpy lstrcpyW

#else

#define lstrcpy lstrcpyA

#endif // !UNICODE

四.使用shlwapi头文件中定义的函数StrCat/StrCmp/StrCpy

Shlwapi.dll(依赖kernel32.dll)是UNCURL地址动态链接库文件,用于注册键值和色彩设置。因为操作系统字符串函数常常被大型应用程序比如操作系统的外壳进程Explorer.exe所使用。由于这些函数使用得很多,因此,在应用程序运行时,它们可能已经被装入RAM。这将有助于稍稍提高应用程序的运行性能。

注意:使用StrCatStrCmpStrCpy etc时要

#include <Shlwapi.h>

#pragma comment(lib, "Shlwapi.lib")

// …/Microsoft Visual Studio 8/VC/PlatformSDK/Include/Shlwapi.h

LWSTDAPI_(LPWSTR) StrCatW(LPWSTR psz1, LPCWSTR psz2);

LWSTDAPI_(int) StrCmpW(LPCWSTR psz1, LPCWSTR psz2);

LWSTDAPI_(LPWSTR) StrCpyW(LPWSTR psz1, LPCWSTR psz2);

#ifdef UNICODE

#define StrCat StrCatW

#define StrCmp StrCmpW

#define StrCpy StrCpyW

#else

#define StrCat lstrcatA

#define StrCmp lstrcmpA

#define StrCpy lstrcpyA

由上可以看出StrCpy调用的是lstrcpyStrCat调用的是lstrcatStrCmp调用的是lstrcmp

五.MFC动态字符串类CString

// …/Microsoft Visual Studio 8/VC/atlmfc/include/afx.h

一个CString对象由可变长度的一队字符组成。CString使用类似于Basic的语法提供函数和操作符。连接和比较操作符以及简化的内存管理使CString对象比普通字符串数组容易使用。

CString是基于TCHAR数据类型的对象。如果在你的程序中定义了符号_UNICODE,则TCHAR被定义为类型wchar_t,即16位字符类型;否则,TCHAR被定义为char,即8位字符类型。在UNICODE方式下,CString对象由16位字符组成。非UNICODE方式下,CString对象由8位字符组成。 而VS2005默认TCHARwchar而不是char.

当不使用_UNICODE时,CString是多字节字符集(MBCS,也被认为是双字节字符集,DBCS)。注意,对于MBCS字符串,CString仍然基于8位字符来计算,返回,以及处理字符串,并且你的应用程序必须自己解释MBCS的开始和结束字节。

CString 提供 operator LPCTSTR 来在 CString LPCTSTR 之间进行转换。

有关CString的操作请参考MSDN MFC类库。参考源码文件AFX.HAFX.INLSTRCORE.CPPSTREX.CPP

六.更安全的C语言字符串处理函数 Strsafe.h

// …/Microsoft Visual Studio 8/VC/PlatformSDK/Include/strsafe.h

注意:使用StringCchCopy/StringCchPrintf时要#include <Shlwapi.h>.

STRSAFEAPI是为了解决现有的 C 语言运行时函数的代码太容易产生的内存溢出问题。当我们引用 strsafe 系列函数时,原有的 C 语言字符串处理函数都将被自动进行 #undef 处理。调试过程中的警告或出错信息将会告诉我们哪些函数哪些不安全,哪些已经被相应的 strsafe 系列函数取代了。

1.不赞成使用不安全的函数,以避免产生编译错误

2.如果你不要安全处理,你可以在包含strsafe.h头文件之前,

#define STRSAFE_NO_DEPRECATE

#ifdef DEPRECATE_SUPPORTED

// First all the names that are a/w variants (or shouldn't be #defined by now anyway).

#pragma deprecated(strcpy)

#pragma deprecated(wcscpy)

#pragma deprecated(lstrcpy)

#pragma deprecated(StrCpy)

类似的Strcat/wcscat/lstrcat/StrCatsprintf/wsprintf

以下是D3D中预编译头文件dxstdafx.h

#pragma warning( disable : 4996 ) // 将报警置为无效

#include <strsafe.h>

#pragma warning( default : 4996 ) // 将报警置为默认

有关#pragma warning请参考《关于#pragma warning

以下是D3DVS2003移植到VS2005时遇到的安全警告:

warning C4996: 'wcscpy' was declared deprecated

see declaration of 'wcscpy'

Message: 'This function or variable may be unsafe.

Consider using wcscpy_s instead. To disable deprecation, use _CRT_SECURE_NO_DEPRECATE. See online help for details.'

warning C4995: 'lstrcpy': name was marked as #pragma deprecated

warning C4995: 'wsprintf': name was marked as #pragma deprecated

推荐使用新的安全可靠的TRSAFEAPI

STRSAFEAPI

StringCchCopyA(

__out_ecount(cchDest) STRSAFE_LPSTR pszDest,

__in size_t cchDest,

__in STRSAFE_LPCSTR pszSrc);

STRSAFEAPI

StringCchCopyW(

__out_ecount(cchDest) STRSAFE_LPWSTR pszDest,

__in size_t cchDest,

__in STRSAFE_LPCWSTR pszSrc);

#ifdef UNICODE

#define StringCchCopy StringCchCopyW (WWide Unicode)

#else

#define StringCchCopy StringCchCopyA (AANSI)

#endif // !UNICODE

#undef strcpy

#define strcpy strcpy_instead_use_StringCbCopyA_or_StringCchCopyA;

#undef wcscpy

#define wcscpy wcscpy_instead_use_StringCbCopyW_or_StringCchCopyW;

#undef wsprintf

#define wsprintf wsprintf_instead_use_StringCbPrintf_or_StringCchPrintf;

// Then all the windows.h names - we need to undef and redef based on UNICODE setting

#undef lstrcpy // 取消已定义的宏

#pragma deprecated(lstrcpy) // 安全警告

#ifdef UNICODE // 使用UNICODE编程

#define lstrcpy lstrcpyW // 重定义

#else

#define lstrcpy lstrcpyA // 重定义

#endif

类似的有对lstrcat/wsprintf/wvsprintf#undef#pragma deprecated#define

推荐使用新的安全可靠的TRSAFEAPI

#undef lstrcpy

#define lstrcpy lstrcpy_instead_use_StringCbCopy_or_StringCchCopy;

// Then the shlwapi names - they key off UNICODE also.

#undef StrCpy

#pragma deprecated(StrCpy)

#ifdef UNICODE

#define StrCpy StrCpyW

#else

#define StrCpy lstrcpyA

#endif

类似的有#undef StrCpyA /StrCpy /StrCatA /StrCat /StrNCat /StrCatN以及对StrCpy/StrCat/StrNCat#undef#pragma deprecated#define

推荐使用新的安全可靠的TRSAFEAPI

#undef StrCpy

#define StrCpy StrCpy_instead_use_StringCbCopy_or_StringCchCopy;

// Then all the CRT names - we need to undef/redef based on _UNICODE value.

七.VC编译UNICODE版本

VC6.0支持Unicode,但在缺省安装情况下,没有把相关的部件安装上去,所以第一步要安装相关的组件。从安装向导中选择自定义,在选择要安装的组件清单时,把vc里面带unicode的子项全部选中安装即可。

如果采用默认安装,后期需要编译发布Unicode版本,在“Project SettingàC/C++CategoryPreprocessoràPreprocessor definitions”处使用_UNICODE UNICODE替换掉_MBCS

程序运行可能会提示找不到MFC42U.DLLMFC42UD.DLLMFCO42UD.DLLUUnicodeDDebugOOle)等,这是由于VC++ 安装的时候没有选择UNICODE支持的缘故,导致没有安装debug版本的MFC42*.DLL。如果只是运行需要的话,可以从VC安装盘(Visual Studio 6安装包目录/VC98/REDIST/MFC42U.DLLVisual Studio 6安装包目录/VC98/DEBUG/MFC*.DLL)或其它机器copy过来,如果编译使用,建议还是重装来得实在。

参考:

VC中的__T

VC6 UNICODE编程总

有关UNICODEANSI字符集和相关字符串操的总结

分享到:
评论

相关推荐

    VC++中的char,wchar_t,TCHAR

    VC++中的char,wchar_t,TCHAR,主要讲解char,wchar_t,TCHAR的用法及区别,适合初学者了解相关内容

    关于char,_wchar_t,_TCHAR,__T(),L,宏__T、TEXT,_TEXT、L.

    关于char,_wchar_t,_TCHAR,__T(),L,宏__T、TEXT,_TEXT、L.

    VC++中的char wchar_T和TCHAR

    详细介绍了VC++中的char wchar_T和TCHAR的含义及区别,对于想了解各种编码的初学者有着较大用处

    VC++中的char_wchar_T和TCHAR

    详细的介绍了char wchar TCHAR的联系和区别,非常适合初学者了解

    关于char的前后兼容问题

    关于char, wchar_t, TCHAR, _T(),L,宏 _T、TEXT,_TEXT、L

    CString、LPCTSTR、LPTSTR、TCHAR、WCHAR、string、wchar_t、char解析

    CString、LPCTSTR、LPTSTR、TCHAR、WCHAR、string、wchar_t、char解析 CString、LPCTSTR、LPTSTR、TCHAR、WCHAR、string、wchar_t、char解析 CString、LPCTSTR、LPTSTR、TCHAR、WCHAR、string、wchar_t、char解析

    char TCHAR WCHAR区别

    介绍char_TCHAR_WCHAR三种类型的区别

    wchar_t,char,string,wstring之间的相互转换

    在处理中文时有时需要进行wchar_t,char,string,wstring之间的转换。 其中char和string之间、wchar_t和wstring之间的转换较为简单,代码在vs2010下测试通过。 代码如下:#include &lt;iostream&gt;#include &lt;string&gt;#...

    cpp-texport用于自动生成TCHAR类型程序的char和wchart类型的函数调用存根

    texport用于自动生成 TCHAR 类型程序的 char 和 wchar_t 类型的函数调用存根

    Unicode ANSI中一些宏的解释

    关于char, wchar_t, TCHAR, _T(),L,宏 _T、TEXT,_TEXT、L(转载) CString ,BSTR ,LPCTSTR之间关系和区别 (转载)

    TCHAR转CHAR与WCHAR类型导出函数自动生成器

    当在 windows 下开发程序库时,使用 TCHAR 类型,比直接使用 char 和 wchar_t 具有更好的移植性。但是在编写动态链接库时,不能直接导出 TCHAR 类型的函数,给 TCHAR 类型函数的使用带来了一些限制。使用 texport 在...

    C++字符串操作经验集

    lchar,wchar_t,CHAR,BYTE,UCHAR,WCHAR,TCHAR,TBYTE char是C语言标准数据类型,字符型,至于由几个字节组成通常由编译器决定,一般一个字节。 wchar_t是char的Unicode版本。相当于unsigned short。一般两个...

    VC下实现fopen支持中文的方法

    bool UTF8ToUnicode(const char* UTF8, wchar_t* strUnicode) { DWORD dwUnicodeLen; //转换后Unicode的长度 TCHAR *pwText; //保存Unicode的指针 // wchar_t* strUnicode; //返回值 //获得转换后的长度,并分配...

    VC++对字符串处理

    VC++ MFC里定义了一些宏,例如LPCTSTR表示const TCHAR*,LPTSTR表示TCHAR*,LPSTR表示char*,LPWSTR表示wchar_t*,LPCSTR表示const char*,LPCWSTR表示const wchar_t*。总之就是LP表示指针,C表示const,STR表示char...

    019-兼容型字符串(TCHAR)及相关函数.flv

    typedef wchar_t TCHAR; #else typedef char TCHAR; #endif #ifdef UNICODE #define _tcscpy wcscpy #else #define _tcscpy strcpy #endif 再来看_tmain()的定义 #ifdef UNICODE #define _tmain ...

    《深入学习c++string》2.1版

    目录 1 正文 4 一、 C++的string的使用 4 1.1 C++ string简介 4 ...2.1.4 以char和wchar_t特化char_traits 24 附录3 TCHAR.h 映射表 24 日志: 24 1.0 版 24 1.1 版 24 1.2 版 24 2.0 版 24 2.1 版 24

    《C++String深入详解2.0版》PDF

    2.1.4 以char和wchar_t特化char_traits 56 附录3:Boost.Format中文文档 57 2.1 大纲 57 2.2 它是如何工作的 57 2.3语法 58 2.3.1 boost::format( format-string ) % arg1 % arg2 % ... % argN 58 2.3.2 printf 格式...

    C++多字节字符与宽字节字符相互转换

    觉得麻烦,于是就自己写了一个类来封装wchar_t与char类型间的转换,其他的,诸如:CString\ LPWSTR\TCHAR CHAR\LPSTR之间也是一样用 代码如下: #include &lt;iostream&gt; using namespace std; class CUser { public: ...

    CString,string,char的综合比较

    CString(typedef CStringT&lt;TCHAR, StrTraitMFC&lt;TCHAR&gt;&gt; CString)为Visual C++中最常用的字符串类,继承自CSimpleStringT类,主要应用在MFC和ATL编程中,主要数据类型有char(应用于ANSI),wchar_t(unicode),TCHAR...

    LPK专杀C语言源码

    _T("出错! %s"), lpMsgBuf); LocalFree(lpMsgBuf); wprintf(L"%s\n", szBuf); return false; } bool DelRegedit(HKEY szMainKey, TCHAR *szRegName) { HKEY hResult = 0;//\\Ghijkl Nopqrstu Wxy LONG l...

Global site tag (gtag.js) - Google Analytics