From:http://www.cppblog.com/clane/archive/2010/03/10/109382.html
项目编译后给朋友试运行,发现运行出错,提示路径不正确找不到配置文件。因为朋友是放在桌面运行的,于是推测是中文路径的问题(因为路径中包含"桌面"两个汉字)。反应很诧异,什么年代了,还有中文路径的问题... 跟踪了一下ifstream的open函数,发现ifstream在打开文件之前会通过_mbstowcs_l_helper函数把文件路径从mutilbyte转换到unicode。其中的关键转换函数如下:
if
(_loc_update.GetLocaleT()
->
locinfo
->
lc_handle[LC_CTYPE]
==
_CLOCALEHANDLE)
{
/* C locale: easy and fast */
while (count < n)
{
* pwcs = (wchar_t) ((unsigned char )s[count]);
if ( ! s[count])
return count;
count ++ ;
pwcs ++ ;
}
return count;
}
而vc的默认local信息就是
_CLOCALEHANDLE,于是中文字符很悲剧的被转换成了莫名其妙的一串东西。google了下,似乎很多人碰到了这个问题,也没有特别好的解决方案,要不直接unicode,要不每次调用fstream前后都调用一遍setlocal,对代码的侵入性都很强。
{
/* C locale: easy and fast */
while (count < n)
{
* pwcs = (wchar_t) ((unsigned char )s[count]);
if ( ! s[count])
return count;
count ++ ;
pwcs ++ ;
}
return count;
}
我做了个封装的解决方案,使用个模板类对fstream做一个wrapper,代码如下:
template
<
class
T
>
struct fstream_fix
: public T
{
fstream_fix(){};
template < class T1 >
fstream_fix(T1 v1){
setlocale(LC_CTYPE, " " );
T::open(v1);
setlocale(LC_CTYPE, 0 );
}
template < class T1, class T2 >
fstream_fix(T1 v1,T2 v2){
setlocale(LC_CTYPE, " " );
T::open(v1,v2);
setlocale(LC_CTYPE, 0 );
}
template < class T1 >
void open(T1 v1){
setlocale(LC_CTYPE, " " );
T::open(v1);
setlocale(LC_CTYPE, 0 );
}
template < class T1, class T2 >
void open(T1 v1,T2 v2){
setlocale(LC_CTYPE, " " );
T::open(v1,v2);
setlocale(LC_CTYPE, 0 );
}
};
#define ifstream fstream_fix<ifstream>
#define ofstream fstream_fix<ofstream>
OK.完美,对原项目没有任何影响,ifstream fi(filepath);filepath中含有中文也能正常工作了。:) 当然要注意的是,在宏定义之后,就不能再include <fstream>,不然可能会有编译错误。
struct fstream_fix
: public T
{
fstream_fix(){};
template < class T1 >
fstream_fix(T1 v1){
setlocale(LC_CTYPE, " " );
T::open(v1);
setlocale(LC_CTYPE, 0 );
}
template < class T1, class T2 >
fstream_fix(T1 v1,T2 v2){
setlocale(LC_CTYPE, " " );
T::open(v1,v2);
setlocale(LC_CTYPE, 0 );
}
template < class T1 >
void open(T1 v1){
setlocale(LC_CTYPE, " " );
T::open(v1);
setlocale(LC_CTYPE, 0 );
}
template < class T1, class T2 >
void open(T1 v1,T2 v2){
setlocale(LC_CTYPE, " " );
T::open(v1,v2);
setlocale(LC_CTYPE, 0 );
}
};
#define ifstream fstream_fix<ifstream>
#define ofstream fstream_fix<ofstream>