- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
这是这个问题的扩展:fstream not opening files with accent marks in pathname
问题如下:打开一个简单的 NTFS 文本文件的程序在路径名中带有重音符号(例如à、ò、. ..).在我的测试中,我使用了一个路径名为 I:\università\foo.txt 的文件(università 是 university 的意大利语翻译)
下面是测试程序:
#include <iostream>
#include <fstream>
#include <string>
#include <cstdio>
#include <errno.h>
#include <Windows.h>
using namespace std;
LPSTR cPath = "I:/università/foo.txt";
LPWSTR widecPath = L"I:/università/foo.txt";
string path("I:/università/foo.txt");
void tryWithStandardC();
void tryWithStandardCpp();
void tryWithWin32();
int main(int argc, char **argv) {
tryWithStandardC();
tryWithStandardCpp();
tryWithWin32();
return 0;
}
void tryWithStandardC() {
FILE *stream = fopen(cPath, "r");
if (stream) {
cout << "File opened with fopen!" << endl;
fclose(stream);
}
else {
cout << "fopen() failed: " << strerror(errno) << endl;
}
}
void tryWithStandardCpp() {
ifstream s;
s.exceptions(ifstream::failbit | ifstream::badbit | ifstream::eofbit);
try {
s.open(path.c_str(), ifstream::in);
cout << "File opened with c++ open()" << endl;
s.close();
}
catch (ifstream::failure f) {
cout << "Exception " << f.what() << endl;
}
}
void tryWithWin32() {
DWORD error;
HANDLE h = CreateFile(cPath, GENERIC_READ, 0, NULL, OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, NULL);
if (h == INVALID_HANDLE_VALUE) {
error = GetLastError();
cout << "CreateFile failed: error number " << error << endl;
}
else {
cout << "File opened with CreateFile!" << endl;
CloseHandle(h);
return;
}
HANDLE wideHandle = CreateFileW(widecPath, GENERIC_READ, 0, NULL, OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, NULL);
if (wideHandle == INVALID_HANDLE_VALUE) {
error = GetLastError();
cout << "CreateFileW failed: error number " << error << endl;
}
else {
cout << "File opened with CreateFileW!" << endl;
CloseHandle(wideHandle);
}
}
源文件以UTF-8编码保存。我正在使用 Windows 8。
这是用VC++(Visual Studio 2012)编译程序的输出
fopen() failed: No such file or directory
Exception ios_base::failbit set
CreateFile failed: error number 3
CreateFileW failed: error number 3
这是使用 MinGW g++ 的输出
fopen() failed: No such file or directory
Exception basic_ios::clear
CreateFile failed: error number 3
File opened with CreateFileW!
那么让我们来回答问题:
我希望打开一个带有generic 路径名的generic 文件可以在不需要特定于平台的代码的情况下完成,但我不知道该怎么做。
提前致谢。
最佳答案
你写:
“The source file is saved with UTF-8 encoding.”
好吧,如果您使用的是 g++ 编译器,那一切都很好(到目前为止),它以 UTF-8 作为其默认的基本源字符集。但是,Visual C++ 默认假定源文件是用 Windows ANSI 编码的,除非另有明确说明。因此,请务必确保它在开头有一个 BOM(字节顺序标记),据我所知,它未记录在案 - 导致 Visual C++ 将其视为使用 UTF-8 编码。
然后你问,
“1. Why fopen() and std::ifstream works in a similar test in Linux but they don't in Windows?”
对于 Linux,它可能可以工作,因为 (1) 现代 Linux 是面向 UTF-8 的,所以如果文件名看起来相同,它可能与中的相同的 UTF-8 编码文件名相同源代码,以及 (2) 在 *nix 中,文件名只是字节序列,而不是字符序列。这意味着无论它看起来如何,如果您传递相同的字节序列、相同的值,那么您就有一个匹配项,否则就没有匹配项。
相比之下,在 Windows 中,文件名是可以用多种方式编码的字符序列。
在您的情况下,源代码中的 UTF-8 编码文件名在可执行文件中存储为 Windows ANSI(是的,使用 Visual C++ 构建的结果取决于 Windows 中选定的 ANSI 代码页,据我所知知道是无证)。然后这个 gobbledegook 字符串向下传递到例程层次结构并转换为 UTF-16,这是 Windows 中的标准字符编码。结果与文件名完全不匹配。
你进一步问,
“2. Why CreateFileW() works only compiling with g++?”
大概是因为您没有在源代码文件的开头包含 BOM(见上文)。
有了 BOM,一切都可以很好地与 Visual C++ 配合使用,至少在 Windows 7 中是这样:
File opened with fopen!File opened with c++ open()File opened with CreateFile!
最后,你问,
“3. Does exist a cross-platform alternative to CreateFile?”
不是真的。有升压文件系统。但是虽然它的第 2 版确实有针对标准库的基于有损窄字符的编码的解决方法,但该解决方法在第 3 版中被删除,它只使用 Visual C++ 标准库的扩展,其中 Visual C++ 实现提供流构造函数和 open
的宽字符参数版本。即,至少据我所知(我最近没有检查是否已修复问题),Boost 文件系统通常只适用于 Visual C++,不适用于例如g++——尽管它适用于无问题字 rune 件名。
v2 的解决方法是尝试转换为 Windows ANSI(由 GetACP
函数指定的代码页),如果不起作用,请尝试 GetShortPathName
,实际上保证可以用 Windows ANSI 表示。
Boost 文件系统中的解决方法被删除的部分原因是,据我所知,原则上用户可以关闭 Windows 短名称功能,至少在 Windows Vista 和更早版本中是这样。然而,这不是一个实际问题。这只是意味着如果用户因故意对系统进行脑叶切除术而遇到问题,则有一个简单的修复方法(即重新打开它)。
关于C++ 和 NTFS 文件 : pathname VS opening,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/14527845/
OpenAL.org && 创意开发网站已关闭。我选择替代版本 OpenAL Soft .我很担心,因为在 OpenAL Soft 的二进制安装中我找不到 alut.h header 。 alut.h
我使用 Android Studio 已经有一段时间了,但有一天应用程序突然出错了。当我尝试单击我的目录以查找要导入或打开的文件时,应用程序变得异常缓慢并且根本没有响应。当我最终成功切换到存储我的文件
自 Firefox 4 以来,这似乎是一个奇怪的功能变化。在使用 window.open() 打开一个窗口后,当用鼠标中键单击打开的窗口中的链接时(或右键单击并选择“在新窗口中打开”选项卡') 导致链
我无法从 Open::URI 的 rdoc 中得知当我这样做时返回的是什么: result = open(url) URL 返回 XML,但我如何查看/解析 XML? 最佳答案 open 返回一个 I
经常开发asp但对于细致的说法,真实不太清楚,这里简单的介绍下。 一般情况下 读取数据都是用rs.open sql,conn,1,1 修改数据:rs.open sql,conn,1,3 删除
关于 pathlib 标准库中的模块,是 path.open() 方法只是内置 open() 的“包装器”功能? 最佳答案 如果您阅读了 source code的 pathlib.Path.open你
我想将 Open Liberty 运行时的语言更改为 en_US从 Eclipse IDE 中,但我不知道如何。 也尝试使用 JVM 参数的首选项来设置它,但它没有用。 -Duser.language
这是我所拥有的: 参数“opener”未在可能的函数调用参数中列出。这是 PyCharm 错误还是其他原因? PyCharm 2018.3.5 社区版,Windows 7 上的 Python 3.6.
我正在使用 Tinkerpop 的 GraphFactory.open(Configuration 配置) Java 命令来访问 Neo4j 数据库。 一个最低限度的工作示例是: Configurat
这个问题在这里已经有了答案: What is the python "with" statement designed for? (11 个答案) 关闭 7 年前。 我没有使用过 with 语句,但
我正在玩 python 3.5 中的 open 函数。我不明白 opener 参数(最后一个参数)在 open 函数中的用法。根据 python 文档:可以通过将可调用对象作为打开器传递来使用自定义打
关闭。此题需要details or clarity 。目前不接受答案。 想要改进这个问题吗?通过 editing this post 添加详细信息并澄清问题. 已关闭 5 年前。 Improve th
我试图用 Python 来做一些模拟 3D 声音的工作。我试图运行此代码(答案中提供):Python openAL 3D sound类似,两次都收到: ModuleNotFoundError: No
我一直认为 open 和 io.open 可以互换。 显然不是,如果我相信这个片段: import ctypes, io class POINT(ctypes.Structure): _fie
这个问题在这里已经有了答案: What's the difference between io.open() and os.open() on Python? (7 个答案) 关闭 9 年前。 我是
我正在尝试更好地了解 WCF 的一些内部工作原理。我已经做了相当多的环顾四周,但我无法找到关于 ChannelFactory.Open() 与 IClientChannel.Open() 相比的明确解
这个问题在这里已经有了答案: What is the python "with" statement designed for? (11 个答案) 关闭 7 年前。 我知道有很多关于在 python
CFSDN坚持开源创造价值,我们致力于搭建一个资源共享平台,让每一个IT人在这里找到属于你的精彩世界. 这篇CFSDN的博客文章adodb.recordset.open(rs.open)方法参数详解由
不久前我遇到了一个interesting security hole Link 看起来足够无害,但有一个漏洞,因为默认情况下,正在打开的页面允许打开的页面通过 window.opener 回调到它。有
这在我的应用程序上运行良好,但由于某种原因我无法让它在这里正常工作。无论如何,我的问题是,当我单击列表标题时,我想关闭之前打开的列表标题并仅保留事件的列表标题打开。目前它会打开我点击的所有内容,但也会
我是一名优秀的程序员,十分优秀!