在检查 C/C 代码时,我们经常会发现从未使用过的变量。出现此类未使用变量的原因有多种。在这篇文章中,我们将探讨 C/C 代码中出现未使用变量的一些最常见原因。我们回顾如何让编译器警告未使用的变量,以及如何抑制特定未使用变量的警告。
未使用的变量可能保留在代码库中的原因有很多。其中包括:
缺陷和错误:未使用变量的最明显原因是有缺陷的代码。要么该变量确实根本不需要并且可以删除,要么该变量是必要的,但我们忘记在某些关键点使用它。
重构:随着软件的编写和重写,整个代码部分可能会被删除。曾经对代码至关重要的变量(例如辅助计算的结果)可能会被留下、未使用。
面向未来:未使用的变量不仅可能作为过去代码的遗留而出现,而且还可能作为未来代码的遗留而出现。您可以在尚未编写的代码中声明变量。
条件编译: 变量可能会保持未使用状态,具体取决于预处理器阶段。标准示例是仅为调试目的而定义的变量。您的代码可能包含以下形式的内容
const auto value = compute_some_value(); const auto value_for_comparison_only = compute_same_value_differently(); assert( value == value_for_comparison_only );
如果您使用 -DNDEBUG 进行编译,那么编译器可能会警告您 value_for_comparison_only 从未使用过:事实上,assert 语句已被替换为...。
不同的编译器和警告级别设置可能会影响变量在编译过程中是否报告为未使用。
例如,GCC 和 Clang 具有 -Wunused-variable 标志,该标志会触发有关未使用变量的警告。 -Wall 警告选项已经暗示了该标志,并且可以通过 -Wno-unused-variable 将其关闭。
我的建议是始终使用 -Wall 进行编译,然后在允许的情况下有选择地关闭警告。这将是未使用变量的所有实例。
虽然我们应该始终在编译时启用尽可能多的警告,但在某些情况下我们希望有选择地关闭有关特定未使用变量的警告。一种流行的方法是强制转换为 void:
Object unused_object; (void)unused_object;
强制转换为 void 算作变量的(形式上)使用,因此不会发出警告。
虽然这消除了used_object未使用的警告,但正如预期的那样,有一些方法可以改进这一点。我们希望有明确的语义,表明这个 void 转换表示一个未使用的对象。常见的方法是定义宏:
const auto value = compute_some_value(); const auto value_for_comparison_only = compute_same_value_differently(); assert( value == value_for_comparison_only );
一个优点是我们现在可以明确传达此变量的意图(或缺乏意图)。此外,如果我们决定清除代码中未使用的变量,那么搜索它们就会容易得多。
除了宏之外,我们还有可变属性:要么是 C 语言本机的,要么是 C/C 编译器提供的语言扩展。例如,Clang 和 GCC 允许使用变量属性 __attribute__((unused))。 C 17 支持 [[maybe_unused]] 属性:
Object unused_object; (void)unused_object;
这些属性向编译器(以及我们)传达这些变量可能未被使用的信息,我们对此表示同意。
从历史上看,GCC 属性首先出现,是 C 和 C 中编译器特定的语言扩展。从 C 17 开始,属性是语言标准的一部分。然而,不仅拼写不同,而且标准和 GCC 扩展对于属性的放置位置也不一致。
[[maybe_unused]] 属性会发现它的大部分应用都带有条件编译。例如,它是仅调试变量的自然属性。纯粹出于美观原因,我个人更喜欢定义一个宏 #define MAYBE_UNUSED [[maybe_unused]] .
__attribute__((unused)) 的一个优点是,如果代码中曾经使用过该变量,它实际上会警告您。它不是可能未使用,而是绝对没有使用过,并且使用该变量现在会产生警告。
显然,未使用的变量非常频繁且重要,甚至足以保证它们自己的语言扩展。
相比之下,注释掉未使用的变量是一个好的策略吗?并非总是如此!在整个开发代码和调试阶段中保留未使用的变量是有原因的。假设该变量在过去版本的代码中使用过,也许您还没有决定是否应该搁置旧代码或重新集成;意思是:你不知道是否可能再次需要未使用的变量。
保留如下代码可能有助于调试目的:
#define UNUSED(x) (void)(x); // ... Object unused_object; UNUSED(unused_object);
即使复杂计算的结果从未被使用,保留它也会构成另一个失败点……而这正是您在调试过程中想要的。即使最初不是为了调试,如果您决定(再次)需要它,让程序初始化该变量也会很有帮助。
我希望这篇关于未使用变量的帖子对您有用。
以上是C/C 中未使用的变量:为什么以及如何使用?的详细内容。更多信息请关注PHP中文网其他相关文章!