考虑这个程序:
#include <stdio.h>
int main() {
printf("%s\n", __FILE__);
return 0;
}
根据文件的名称,此程序是否有效。我面临的问题是我想以编码安全的方式打印当前文件的名称。但是,如果文件包含无法在当前代码页中表示的有趣字符,编译器会发出警告(理所当然):
?????????.c(3) : warning C4566: character represented by universal-character-name '\u043F' cannot be represented in the current code page (1252)
我该如何解决这个问题?我想将__FILE__
给出的字符串存储在例如UTF-16,以便我可以在运行时在任何其他系统上正确打印它(通过将存储的UTF-16表示转换为运行时系统使用的任何表示)。为此,我需要知道:
__FILE__
给出的字符串使用了什么编码?看来,至少在Windows上,使用了当前的系统代码页(在我的例子中是Windows-1252) - 但这只是猜测。这是真的吗?我的真实用例:我有一个跟踪当前程序执行的宏,将当前源代码/行号信息写入文件。它看起来像这样:
struct LogFile {
// Write message to file. The file should contain the UTF-8 encoded data!
void writeMessage( const std::string &msg );
};
// Global function which returns a pointer to the 'active' log file.
LogFile *activeLogFile();
#define TRACE_BEACON activeLogFile()->write( __FILE__ );
如果当前源文件的名称包含当前代码页无法表示的字符,则会中断。
答案 0 :(得分:11)
使用可以使用令牌粘贴操作符,如下所示:
#define WIDEN2(x) L ## x
#define WIDEN(x) WIDEN2(x)
#define WFILE WIDEN(__FILE__)
int main() {
wprintf("%s\n", WFILE);
return 0;
}
答案 1 :(得分:1)
__FILE__
将始终扩展为字符串文字,因此本质上它将与char const*
兼容。这意味着编译器实现除了使用源文件名的原始字节表示之外别无选择,因为它在编译时呈现。
这是否在当前语言环境中是否合理无关紧要,只要您的运行时系统和编译器将其作为有效文件名接受,您就可以拥有基本上包含垃圾的源文件名。
如果您作为用户使用的语言环境与文件系统中使用的编码不同,您会看到很多????或者一样。
但如果你的语言环境都同意编码,那么普通的printf
应该足够了,你的终端(或者你用来查看输出的任何东西)应该能够打印字符正确。
所以简短的回答是,它只有在你的系统与w.r.t编码一致的情况下才有效。否则你运气不好,因为猜测编码是一项非常困难的任务。
答案 2 :(得分:-1)
至于编码,我猜它是文件系统使用的,可能是Unicode。
至于处理它,如何改变你的代码如下:
#define TRACE_BEACON activeLogFile()->write( FixThisString(__FILE__ ));
std::string FixThisString(wchar_t* bad_string) { .....}
(FixThisString的实现留给学生练习。)
答案 3 :(得分:-1)
最佳解决方案是在可移植文件名字符集[A-Za-z0-9._-]
中使用源文件名。由于Windows不支持UTF-8,因此无法在普通字符串中表示任意非ASCII字符,而不依赖于您配置的本地语言。
L
添加到__FILE__
的扩展中。并使用%ls
对其进行格式化。
答案 4 :(得分:-1)
在MSVC中,您可以打开Unicode并获取UTF-16编码的字符串。它位于某个项目属性中。另外,你应该只使用wcout / cout而不是printf / wprintf。在Unicode存在之前,Windows需要Unicode,因此它们具有自定义多字节字符编码,这是默认设置。但是,Windows确实支持UTF16-例如,C#。
#include <iostream>
int main() {
std::wcout << __WFILE__;
}