在学习最新的C++标准过程中会碰到很多新问题,这些问题有时候很难理解为什么会发生。更奇怪的是,似乎跟开发工具有关。如下面这个在VS2005下的例子:
ifstream ifs;
if (IsExit(szPRLFile, 0 ) != 0 )
... {
AfxMessageBox("file is not exit!");
return -1;
}
std::locale:: global (std::locale( "" )); // 添加的语句1
ifs.open(szPRLFile,ios:: in | ios::binary); // 文件名如果有中文此函数会失败
ifs.imbue(std::locale( " C " )); // 添加的语句2
char data[PRLMAXSIZE] = ... {0} ;
ifs.read(data, sizeof (data));
int iTotalSize = ifs.gcount(); // 该函数读取的数据为0;解决方法在调用open的前后添加上面两条语句改变
ifs.close();
上面这段代码如果将“添加的语句1”和“添加的语句2”注释,在VS2005下编译没有问题,但是运行问题就来了。可以在运行到ifs.read(data,sizeof(data))的时候查看data的值,为空。同样,运行到下一句的时候iTotalSize为0。只有将那两句的注释去掉,才能得到我们想要的结果。
同样的代码我在Dev-C++4.9.9.2上运行正常。据说还是同样的代码在VS2003下正常运行。看来是VS2005本身的问题了。一位大侠给出了下面的分析:
VC8 中fstream的陷阱
VC8的STL部分相对以前版本作了相当大的改进,但其中一个更改更象一个bug。它的file stream 类(含ifstream和ofstream)在open时会自动将文件名转换成Unicode,但转换时会使用STL库的默认locale(C locale)来做 page code,这在非英语语言操作系统上会导致文件名错误,从而无法打开文件。解决的办法是在程序初始化时(或在执行open之前),将STL的默认locale设置成系统的locale, 具体方法是加入语句 std::local::global(std::locale(“”)); 即STL需要执行一个初始化过程才能在非英语操作系统上正确运行。Locale会影响所有stream的格式化操作,比如设置了中文操作系统的本地locale后,数字会被格式化成逗号分割的三位组(形如123,456,78).这对于需要再从这些格式化后的字符串恢复成数字的操作来说是不合适的,使用C locale则不会出现这种问题。要在stream open以后再修改locale,STL提供了stream::imbue操作。可以使用stream::imbue(std::locale(“C”))来将stream的locale再修改回c locale。