0

我有一段代码:

// some code, which only do sanity check
expensive checks
// sanity check end

现在我如何告诉编译器强制它选择退出这块?基本上这意味着当我使用 -O2 或 O3 编译时,我不希望它在那里......

谢谢!

4

4 回答 4

3

您可以使用常量和单个 if/def 对来完成此操作。这允许代码仍然被编译和检查错误,但在优化期间被忽略。这可以防止可能破坏检查代码的更改未被检测到。

#if defined(USE_EXPENSIVE_CHECKS) || defined(DEBUG)
#define USE_EXPENSIVE_CHECKS_VALUE true
#else
#define USE_EXPENSIVE_CHECKS_VALUE false
#endif

namespace {
 const bool useExpensiveChecks = USE_EXPENSIVE_CHECKS_VALUE;
};

void function()
{
    if(useExpensiveChecks == true)
    {
        // expensive checks
    }
}
于 2013-05-03T21:18:49.673 回答
2

在预处理器中使用宏和条件确实是避免编译器生成代码的唯一方法。

所以,我会这样做:

#ifdef NEED_EXPENSIVE_CHECKS
inline expensive_checking(params...)
{
   ... do expensive checking here ... 
}
#else
inline expensive_checking(params...)
{
}
#endif

然后只需调用:

some code
expensive_checking(some_parameters...)
some other code

在任何体面的现代编译器中,空的内联函数将导致“无代码”。在您的调试构建设置中使用-DNEED_EXPENSIVE_CHECKS,不要在发布构建中使用它。

我也知道使用宏和函数的组合,例如:

#ifdef NEED_EXPENSIVE_CHECKS
#define EXPENSIVE_CHECKS(stuff...) expensive_checks(__FILE__, __LINE__, stuff...)
inline expensive_checks(const char *file, int line, stuff ...)
{
   if (some_checking)
   {
      cerr << "Error, some_checking failed at " << file << ":" << line << endl;
   }
}
#else
#define EXPENSIVE_CHECKS(stuff...)
#endif

现在,当某事失败时,您可以获得有关哪个文件和哪一行的信息,如果在许多地方进行检查,这将非常有用(如果您愿意,您也可以使用__function__or__pretty_function__来获取函数名称)。

显然,assert()宏基本上会做我的宏解决方案所做的,除了它通常不提供文件名和行号。

于 2013-05-03T21:26:55.187 回答
2

您可以仅在希望代码运行时向编译器传递额外的符号定义,而不是依赖编译器来优化代码:

// some code, which only do sanity check
#ifdef my_symbol
expensive checks
#endif
// sanity check end
于 2013-05-03T21:13:11.263 回答
1

将您的支票移动到不同的函数中,然后导入cassert和 write assert(expensive_check())。如果要禁用检查,请在包含 cassert#define NDEBUG 之前使用。

于 2013-05-03T21:14:45.150 回答