tin*_*man 11 c++ visual-c++ c-preprocessor
我最近发现,当__FILE__在 MSVC(特别是 2013)中使用预定义宏时,默认情况下它将打印源文件的相对路径和头文件的绝对路径。
作为示例,我有一个 VS 项目,其中包含以下内容:
Solution
Project
Headers
foo.h
Sources
main.cpp
Run Code Online (Sandbox Code Playgroud)
main.cpp 和 foo.h 都位于磁盘上的同一目录中。
主要.cpp:
#include <iostream>
#include <string>
#include "foo.h"
int main(int, char*[])
{
std::cout << __FILE__ << std::endl;
foo::bar();
std::cout << "Press enter to exit";
std::string str;
std::getline(std::cin, str);
return 0;
}
Run Code Online (Sandbox Code Playgroud)
foo.h:
#ifndef FOO_H
#define FOO_H
#include <iosfwd>
class foo
{
public:
static void bar()
{
std::cout << __FILE__ << std::endl;
}
};
#endif
Run Code Online (Sandbox Code Playgroud)
当我运行应用程序时(在发布模式下,使用默认设置 - 编译并未/Zi定义/FC),则输出为:
main.cpp
c:\users\<user>\documents\dev\solution\project\foo.h
Press enter to exit
Run Code Online (Sandbox Code Playgroud)
我知道我可能可以传入一个基本路径并在运行时将其删除,但我想知道是否有任何方法可以在编译时更改此行为?显然,定义/FC会产生相反的结果,并且我在 MSVC 手册中看不到任何其他内容来控制标头路径的显示。我认为这可能是一种硬编码行为,这样如果编译器能够从不同的包含路径中选取两个名为 foo.h 的文件,您仍然可以区分它们,或者因为可能有一个与来源的基础和相对显示会很混乱。
我将发布我自己的解决方案,该解决方案并不完美,但却是我所知道的最接近的解决方法。
包含此代码的每个文件都会在调用时删除PROJECT_DIR前缀,因此二进制文件将不包含该前缀。__FILE__assert
更好的解决方案可能希望根据..\需要添加前缀、处理/,并可能保留不同卷上的任何绝对路径,但我在这里不做任何这些。
// Add this to your .vcxproj:
// <PreprocessorDefinitions>PROJECT_DIR="$(ProjectDir.Replace('\', '\\').TrimEnd('\'))";%(PreprocessorDefinitions)</PreprocessorDefinitions>
#include <cassert>
#include <type_traits>
#ifdef PROJECT_DIR
template<class Char1, class Char2>
constexpr size_t path_imismatch(Char1 const a[], Char2 const b[])
{
size_t r = 0;
size_t i = 0;
for (;;)
{
Char1 ch1 = a[i];
Char2 ch2 = b[i];
bool const finished = ch1 == Char1() || ch2 == Char2() || !(ch1 == ch2);
if (ch2 == Char2()) { ch2 = '\\'; }
bool const isdirsep = ch1 == '\\' && ch1 == ch2;
if (isdirsep && ch1 == ch2) { r = i + 1; }
if (finished) { break; }
++i;
}
return r;
}
#define FILENAME(File, Root) ([]() { typedef std::decay_t<decltype(*(File))> _FileNameChar; constexpr _FileNameChar const f[] = File; enum : size_t { O = path_imismatch(f, Root), N = sizeof(f) / sizeof(*f) - O }; constexpr std::array<_FileNameChar, N> const result = []() { std::array<_FileNameChar, N> r = {}; for (size_t i = 0; i < N; ++i) { r[i] = f[O + i]; } return r; }(); return std::integral_constant<std::remove_const_t<decltype(result)>, result>(); }().value.data())
#define _wassert(Source, File, Line) (_wassert)(Source, FILENAME(File, _CRT_WIDE(PROJECT_DIR)), Line)
#endif
Run Code Online (Sandbox Code Playgroud)