项目编译后给朋友试运行,发现运行出错,提示路径不正确找不到配置文件。因为朋友是放在桌面运行的,于是推测是中文路径的问题(因为路径中包含"桌面"两个汉字)。反应很诧异,什么年代了,还有中文路径的问题...
跟踪了一下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,对代码的侵入性都很强。
我做了个封装的解决方案,使用个模板类对fstream做一个wrapper,代码如下:
template<class T>
struct fstream_fix
:public T
{
fstream_fix(){};
template<class T1>
fstream_fix(T1 v1){
setlocale(LC_CTYPE, ".936");
T::open(v1);
setlocale(LC_CTYPE, 0);
}
template<class T1,class T2>
fstream_fix(T1 v1,T2 v2){
setlocale(LC_CTYPE, ".936");
T::open(v1,v2);
setlocale(LC_CTYPE, 0);
}
template<class T1>
void open(T1 v1){
setlocale(LC_CTYPE, ".936");
T::open(v1);
setlocale(LC_CTYPE, 0);
}
template<class T1,class T2>
void open(T1 v1,T2 v2){
setlocale(LC_CTYPE, ".936");
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>,不然可能会有编译错误。