【问题标题】:Nesting goto labels with the preprocessor in C99/C11?在 C99/C11 中使用预处理器嵌套 goto 标签?
【发布时间】:2012-06-24 11:03:18
【问题描述】:

是否可以使用 C 预处理器在 C11 或 C99 中嵌套 goto 标签?通过查看以下代码可能最好地说明我的情况。使用gcc -std=c99 -pedantic -Wall -Wextra 编译干净。

#include <stdio.h>

// Macro for mangling the identifier to avoid collisions
#define CTX_ID_(NAME) context_label_ ## NAME ## _
#define CTX_ID(NAME) CTX_ID_(NAME)

// The context keyword starts a block that can be exited with break (ID);
// Just syntactic sugar to keep it structured.
#define context(ID) \
    if (0) { CTX_ID(ID): ; } else

// Overloaded break keyword. Doesn't prevent using the plain break;    
#define break(ID) \
    do { goto CTX_ID(ID); } while (0)

// Example run
int main(void) {
    context (c) {
        while (1) {
            puts("Outer loop, visible.");
            while (1) {
                puts("Inner loop, visible.");
                break (c);
                puts("You won't see me.");
            }
        }
        puts("Nor me.");
    }
}

我正在尝试取消标识符(在本例中为 c)。但是,与变量不同,goto 标签不能嵌套/限定范围,因为它们在函数中必须是唯一的。是否可以在 C 预处理器中实现可用作 goto 标签的唯一范围标识符?

GCC 支持获取标签地址,但它不是 ISO 标准的一部分。此外,由于开销和易失性问题,我特别试图避免 setjmp。最后,如果您没有看到上述构造的用处,请考虑进一步使用,例如 try-catch 子句或 Python 样式的 with-expressions 来启用类似 RAII 的功能。

【问题讨论】:

  • 我不明白你为什么不想使用标识符。那么,您将如何确定“打破”的范围?另外,我认为重载关键字并不是一个好主意。对宏使用大写字母(除非它们真的只是通用函数接口)确实是一个很好的约定。
  • 我将进一步说,尝试在现有语言的基础上使用这样的宏开发一种新语言是一项您不应该轻率的任务。成本——对于不熟悉你的宠物语言的人来说难以理解——几乎总是比效用高得多。使用标准 goto 编写功能等效的代码的成本为零。

标签: c


【解决方案1】:

我确定__LINE__ 宏可能会派上用场。不会有作用域,但至少您可以通过这种方式生成唯一的标签名称。

但是,这也不能立即解决您的问题。我会大胆地声明这是无法解决的,尽管我确定可能有人会出现并证明我错了!

【讨论】:

  • 不,很遗憾__LINE__ 不适用于这种情况。该标准不清楚物理源代码行的含义,当您有多行宏定义时会出现问题。 gccclang 例如给你不同的扩展。
  • 我考虑过使用__LINE__。但是,我无法弄清楚如何在代码中稍后重用标签,以便可以使用 goto 调用它。在我上面的示例中,context (c) 处的第一个 __LINE__ 将计算为某个整数常量 n,break (c) 将计算为 n+5。
【解决方案2】:

你的想法看起来不错,我唯一想念的是你的break(c) 可以在函数的任何地方发出。我会在这两个宏中添加类似的内容:

#define CONTEXT(ID)                                     \
    if (0) { CTX_ID(ID): ; }                            \
    else for (register bool CTX_ID(ID ## ID) = true;    \
              CTX_ID(ID ## ID);                         \
              CTX_ID(ID ## ID) = false)

#define BREAK(ID)                \
    do {                         \
       CTX_ID(ID ## ID) = false; \
       goto CTX_ID(ID);          \
   } while (0)

如果在依赖块之外使用BREAK(c),这将导致语法错误。根据我的经验,这里使用的 for 变量很容易被现代编译器优化掉。

【讨论】:

  • 谢谢。我在提交之前清理了我的代码,在此之前我实际上使用了 (void) sizeof(CTX_ID(ID ## ID)); 之类的东西来检查控制变量是否存在。
  • @andyn,换句话说,类似于_Static_assert(sizeof(CTX_ID(ID ## ID));。是的,这可能比修改值更好。
猜你喜欢
  • 2014-04-25
  • 2014-06-10
  • 1970-01-01
  • 1970-01-01
  • 2017-06-28
  • 2011-03-09
  • 1970-01-01
  • 2015-10-19
  • 1970-01-01
相关资源
最近更新 更多