fstream 和中文路径

转载 2012年03月26日 12:27:19

fstream 和中文路径 
有时候用ifstream或ofstream打开带有中文路径的文件会失败。

解决办法:
1、使用C语言的函数设置为中文运行环境
setlocale(LC_ALL,"Chinese-simplified");

2、使用STL函数设置为系统语言环境
std::locale::global(std::locale(""));

当然选2啦!

另转载针对该问题的详细讨论:http://haoren.blogbus.com/logs/2278697.html

前两天写的一个程序里面,发现中文路径扫描有问题,无法得出带有中文路径的文件的CRC值以及文件大小。因为这两个获得都是通过fstream实现的,跟踪了一下发现2005中的fstream对中文路径处理存在BUG。

跟 踪后找到问题所在,原来2005中为了让std::wfstream(这个是MS自己搞的东西,C++标准中并未规定有wfstream)的路径支持 wchar_t,所有底层函数的路径都是转换成wchar_t来进行操作的。而在此之前VC2003的wfstream并不支持wchar_t作为路径, 写STL的人并未使用MS的标准API:MultiByteToWideChar及WideCharToMultiByte来进行wchar_t和 char之间的转换,而是用了C库的wcstombos来进行转换。而wcstombos并不像MultiByteToWideChar这样的函数可以在 参数中指定代码页设置,它只接受当前全句的locale设置。当全局的locale设置并不为wcstombos所要转换的文字的代码页时,BUG就出现 了。

默认的locale设置为ANSI C,通过设置代码页为".936"可以解决中文DBCS到UNICODE的转换。因此,如果要解决这个BUG可以:

setlocale(LC_CTYPE, ".936");

设置全局locale来解决。不过
这 样将导致全局的locale改变。wcstombos是标准C库中的函数,并非STL中的库函数,所以无法只用locale类来对它造成单一影响。这么设 置的话,有可能std::cout将无法正确输出中文名,造成的影响似乎有点得不偿失。所以这个BUG最好不要用这种方式解决。还是等MS的SP补丁出来 看看是否解决了这个问题。在此之前可以考虑用C库的fopen来代替fstream.

最后:上面说过fstream底层都是用wchar_t来处理路径的,如果事先把路径转换成wchar_t传入进去也是可以的,但是这样也会带来一些问题:比如说VC2005以前的编译器将无法编译。

另转载对进行上述修改后造成的cout无法输出中文的解决方案:http://elanso.com/ArticleModule/HaHGMbTDPKHaSYHQSsSEM6Ii.html

前几天发了这篇《用fstream对二进制文件的读写 》,有朋友指出了VS2005的fstream对于中文路径支持不好的bug。我想大概是因为VS2005更加重视了对字符串的全球化支持,所以鼓励我们使用unicode编码的字符串,对于MBCS之类的支持可能就疏忽了吧。

   我搜索了一下这个问题的解决,参考了如下资料写了演示代码。

  • fstream 和 中文路径   c++博客
  • About unicode settings in visual studio 2005, it really puzzled me a lot ms forum
  • MSDN

   我综合了以上的内容,总结了3种方法,能够较好解决大家的困扰,包括可能无法使用cout的问题。

    /******************************************************************** 
         created:    2008/05/10 
        created:    10:5:2008   23:56 
         filename:     k:/sj/fstreamTest/fstreamTest/main.cpp 
        file path:    k:/sj/fstreamTest/fstreamTest 
       file base:    main 
       file ext:    cpp 
        author:        Gohan 
    *********************************************************************/  
    #include  
    #include  
    #include  
    using namespace std ;  
    int main()  
    {  
        /************************************************************************/  
        /* 方法1,使用_TEXT()宏定义将字符串常量指定为TCHAR*类型                 */  
        /* 如果是我,首选此类型                                                 */  
        /************************************************************************/  
        fstream file ;  
        file.open(_TEXT("c://测试//测试文本.txt"));  
        cout<<PRE>  
        file.close();  
          
        /************************************************************************/  
        /* 方法2,使用STL中的locale类的静态方法指定全局locale                   */  
        /* 使用该方法以后,cout可能不能正常输出中文,十分蹊跷                    */  
        /* 我发现了勉强解决的方法:不要在还原区域设定前用cout或wcout 输出中文   */  
        /* 否则后果就是还原区域设定后无法使用cout wcout输出中文                 */  
        /************************************************************************/  
        locale :: global(locale(""));//将全局区域设为操作系统默认区域  
        file.open("c://测试//测试文本2.txt");//可以顺利打开文件了  
        locale :: global(locale("C"));//还原全局区域设定  
        cout<<PRE>  
        file.close();  
          
        /************************************************************************/  
        /* 方法3,使用C函数setlocale,不能用cout输出中文的问题解决方法同上      */  
        /************************************************************************/  
        setlocale(LC_ALL,"Chinese-simplified");//设置中文环境  
        file.open("c://测试//测试文本3.txt");//可以顺利打开文件了  
        setlocale(LC_ALL,"C"); //还原  
        cout<<PRE>  
        file.close();  
    }  

补充一下,第一种方法,如果不是静态字符串当作路径的话,记得传入TCHAR*类型字符串作为路径,应该就没问题了。



C/C++中读取中文路径文件的方法

转自 http://apps.hi.baidu.com/share/detail/33007444 有时候用ifstream或ofstream打开带有中文路径的文件会失败。 解决办法: 1、...
  • sibo626
  • sibo626
  • 2011年11月03日 10:14
  • 9184

剖析ifstream打开含中文路径名文件失败的原因

最近写程序的时候遇到了使用ifstream打开含中文路径文件时失败的问题,在网上翻了一下,发现这是一个普遍遇到的问题,在很多人的博文中也都给出了一些解决技巧,但大多是转载的东西,很少对这个问题引发的原...

[C/C++标准库]_[初级]_[读写中文路径的文件--写入unicode字符串]

1. 需要写入非ascii文本并且与本地编码无关时,除了utf8,unicode编码是另外一个选择,它的好处是占两个字节,便于统计字符和对字符进行处理,因为有对应的宽字节的函数,如wcslen. 2...

vc中ifstream ofstream不能读取带有中文路径

有时候用ifstream或ofstream打开带有中文路径的文件会失败。 解决办法: 一、网友甲的推荐: 1、使用C语言的函数设置为中文运行环境 setlocale(LC_ALL,"Chinese...
  • yockie
  • yockie
  • 2012年01月09日 10:16
  • 3606

ofstream打开中文路径的三种方法

 经测试第一种方法在VS2008环境下无效 1: /******************************************************************** 2: ...
  • Rocky_
  • Rocky_
  • 2008年08月07日 15:37
  • 2580

让fstream支持中文路径

fstream 和中文路径  有时候用ifstream或ofstream打开带有中文路径的文件会失败。 解决办法: 1、使用C语言的函数设置为中文运行环境 setlocale(LC_ALL...

关于std::fstream以及std::ifstream打开中文路径名失败的问题和解决方法

碰到的问题:      今天碰到这个问题了,是在使用Crypto++库的时候遇到的,该库操作文件是使用的std::ifstream。在我给文件生成签名的时候,每每碰到中文路径名就出错,后来跟进库代码一...

fstream默认不支持中文路径和输出整数带逗号的解决办法

      我们用fstream来创建一个新文件,如果文件路径中带有中文,则创建一般会失败。如下面代码:#include #include #include #include us...

C++中文件流(fstream)的使用方法及示例

C++文件流: fstream  // 文件流 ifstream  // 输入文件流 ofstream  // 输出文件流     #include //创建一个文本文...
  • seadplus
  • seadplus
  • 2012年07月30日 15:48
  • 35212

C++文件路径的写法

文件路径 正斜杠和反斜杠 正斜杠,又称左斜杠,符号是"/";反斜杠,也称右斜杠,符号是"\"。文件路径的表示可以分为绝对路径和相对路径: 1、绝对路径表示相对容易,例如 pDummyFile...
内容举报
返回顶部
收藏助手
不良信息举报
您举报文章:fstream 和中文路径
举报原因:
原因补充:

(最多只允许输入30个字)