随笔 - 5, 文章 - 0, 评论 - 13, 引用 - 0
数据加载中……

vc2008 fstream不支持中文路径的解决方案

     项目编译后给朋友试运行,发现运行出错,提示路径不正确找不到配置文件。因为朋友是放在桌面运行的,于是推测是中文路径的问题(因为路径中包含"桌面"两个汉字)。反应很诧异,什么年代了,还有中文路径的问题...
     跟踪了一下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>,不然可能会有编译错误。



posted on 2010-03-10 19:19 clane 阅读(3825) 评论(5)  编辑 收藏 引用 所属分类: C++

评论

# re: vc2008 fstream不支持中文路径的解决方案  回复  更多评论   

wifstream ??
2010-03-10 22:34 | WXX

# re: vc2008 fstream不支持中文路径的解决方案  回复  更多评论   

setlocal一次就可以了,不少函数都依赖这个。
2010-03-11 15:29 | 坏人

# re: vc2008 fstream不支持中文路径的解决方案  回复  更多评论   

@WXX
这样的话,要修改项目里所有的传入path为unicode...
2010-03-11 15:57 | clane

# re: vc2008 fstream不支持中文路径的解决方案[未登录]  回复  更多评论   

顶楼主
2010-03-11 16:15 | Bill Hsu

# re: vc2008 fstream不支持中文路径的解决方案  回复  更多评论   

setlocale(LC_CTYPE, ".936");应该改为:

setlocale(LC_ALL,"");

这样更好
2010-03-13 11:51 | Bill Hsu

只有注册用户登录后才能发表评论。
网站导航: 博客园   IT新闻   BlogJava   博问   Chat2DB   管理