如何避免来自嵌套已弃用函数调用的警告?
我支持 C++ 库,并希望将许多遗留函数声明为已弃用。不幸的是,这些函数相互调用,我从它们的编译中收到警告。例如:
[[deprecated]] void foo();
[[deprecated]] void bar() { foo(); }
我想避免有关从 bar() 主体编译中调用已弃用函数的警告,但如果某些外部函数(未标记为已弃用)调用 foo() 或 bar(),仍然会发出警告。它以某种方式可能吗?
回答
虽然这不适用于 OP 发布的情况,但由于从库的头文件中bar()调用foo(),有一个简单的解决方案适用于任何面临相同问题的人,而没有该特定约束。所以它可能对登陆这里的其他人有用。
实际上,您需要两组不同的标题。一种是图书馆的用户使用它,另一种是图书馆本身使用它。在这种情况下我们可以这样做,因为[[deprecated]]它不会对生成的代码产生任何影响。
您“可以”单独维护它们,但这显然非常脆弱。值得庆幸的是,该语言确实为我们提供了一种方法,可以在一个文件中拥有同一头文件的两个“版本”:经常被诽谤,公平地说,也经常被误用的宏。
作为奖励,如果[[deprecated]]恰好是强制用户使用 C++14 或更高版本的原因,您可以通过检查__cplusplus或适当的功能宏同时提供对旧版本标准的支持。
//mylib.h
#if !defined(MY_LIB_NO_DEPRECATE) && __has_cpp_attribute(deprecated)
#define MY_LIB_DEPRECATED [[deprecated]]
#else
#define MY_LIB_DEPRECATED
#endif
// ...
MY_LIB_DEPRECATED void foo();
MY_LIB_DEPRECATED void bar();
// ...
#undef MY_LIB_DEPRECATED
使用 编译库-DMY_LIB_NO_DEPRECATE,就像在该过程的特定阶段不存在弃用警告一样。用户仍然会收到所有弃用警告,除非他们还通过定义MY_LIB_NO_DEPRECATE.
不要被这里的宏的使用吓到。使用它们来区分同一标头的内部/外部版本是一种常见且完善的做法。如果没有它,Windows dll 几乎不可能编写。
在 OP 的情况下,如果将bar()的定义从头文件移动到库的实现中是可能的,那么它们也应该很好。
回答
您可以忽略弃用警告。它在技术上不是可移植的,但它适用于所有 4 个主要编译器,至少:
#if defined(__GNUC__)
#define MY_LIB_IGNORE_DEPRECATED_BEGIN
_Pragma("GCC diagnostic push")
_Pragma("GCC diagnostic ignored "-Wdeprecated-declarations"")
#define MY_LIB_IGNORE_DEPRECATED_END
_Pragma("GCC diagnostic pop")
#elif defined(_MSC_VER)
#define MY_LIB_IGNORE_DEPRECATED_BEGIN
_Pragma("warning(push)")
_Pragma("warning(disable : 4996)")
#define MY_LIB_IGNORE_DEPRECATED_END
_Pragma("warning(pop)")
#else
#define MY_LIB_IGNORE_DEPRECATED_BEGIN
#define MY_LIB_IGNORE_DEPRECATED_END
#endif
你可以在图书馆范围内做到这一点:
MY_LIB_IGNORE_DEPRECATED_BEGIN
[[deprecated]] void foo();
[[deprecated]] void bar() { foo(); }
MY_LIB_IGNORE_DEPRECATED_END
或者你可以只保护违规电话:
[[deprecated]] void foo();
[[deprecated]] void bar()
{
MY_LIB_IGNORE_DEPRECATED_BEGIN
foo();
MY_LIB_IGNORE_DEPRECATED_END
}