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

来源:互联网 发布:卖家怎么在淘宝客推广 编辑:程序博客网 时间:2024/05/01 08:04


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