3

我们有一个文件和一行(__file__, __line__),我们想打印它的内容。这样的事情可以通过预处理器实现吗?

4

2 回答 2

2

也许是这样的?

#include <stdio.h>

#define DBG(X) printf(__FILE__":%d DBG("#X")\n", __LINE__)

int main ()
{
    DBG(this is a test);
}

这将产生输出:

test.cc:7 DBG(this is a test)

编辑:您要么必须访问源文件,要么存储源文件。存储文件后,打印正确的行是微不足道的。单程:

#define PROGRAM_PRINT_INIT(X) \
    static ProgramPrint pp = ProgramPrint(X)
#define DBG(X) std::cerr << "[" << pp.name << ":" << X << "]" \
                         << pp.file[X] << std::endl

struct ProgramPrint {
    std::string name;
    std::vector<std::string> file;
    ProgramPrint (const char *filename) : name(filename) {
        std::ifstream in(filename);
        std::string line;
        while (getline(in, line)) file.push_back(line);
    }
};

请注意,vector索引从零开始,因此请DBG()相应地使用。

于 2012-08-01T23:03:51.050 回答
0

是的,C/C++ 不是为此而生的。但如果你真的想看看你能把东西弯曲多远,这是一个建议。

假设你想编译 test.cpp 来测试。在编译之前,我们需要将源代码行的 cstring 数组添加到源代码文件中。我们将在单独的文件 test_temp.cpp 中执行此操作,我们将调用数组 TEST_CODE。

将以下内容放入文件中:code.awk(这将使其更易于重用)。

BEGIN {
    for (i = 0; i < 128; i++) {
        table = sprintf("%s%c", table, i);
    }
    print("= {\"\"")
}
{
    printf(",\"")
    for(j=1; j<=length($1); j++) {
        c = substr($1,j,1)
        printf("\\x%x", index(table, c) - 1)
    }
    printf("\"\n")
}
END {
    printf("};\n")
}

通过引用 ASCII 字符的“表”,这将使用每个字符的(十六进制)ASCII 值(例如 '\x41' 表示 'A'),以防制表符等可能在您的源代码中但应该在 C/C++ 字符串文字中转义。

要编译,在命令行(或你的 makefile)中,输入

echo "char* TEST_CODE[]" > test_temp.cpp
awk -f code.awk test.cpp >> test_temp.cpp
cat test.cpp >> test_temp.cpp
g++ -o test test_temp.cpp
rm test_temp.cpp

如果你想获取源代码文件的第 80 行(基于 1),你会写

TEST_CODE[80]

注意这发生在预处理器命令之前。这可以通过在每种情况下从 TEST_CODE 更改数组名称来使用多个文件和头文件来完成。

这确实有漏洞,但它会在常见情况下工作。

注意:由于 char* / string 文字转换,g++ 可能会发出很多编译器警告,但您可以使用 -w 标志关闭这些警告。

于 2012-09-10T01:33:23.907 回答