c - 不兼容的指针类型传入 _Generic 宏

标签 c macros printf clang c11

以下代码生成问题标题中描述的 2 个警告。

#include <stdio.h>

static void _print_f(float *f){printf("float : %f\n", *f);}
static void _print_i(int *i)  {printf("int   : %d\n", *i);}

#define print(num) _Generic((num), \
    int*   : _print_i(num),        \
    float* : _print_f(num))


int main(void)
{
    print((&(int){10}));
    print((&(float){10.f}));

    return 0;
}

输出:

int   : 10
float : 10.000000

我知道,这个宏可以这样写:

#define print(num) _Generic((num), \
    int*   : _print_i,             \
    float* : _print_f)(num)

在这种情况下,不会有任何警告,但我的示例是我编写的用于演示问题的虚拟代码段。在我的真实代码库中,我选择了前一种解决方案,因为需要将其他一些“默认”但类型特定的参数传递给所选函数。

所以问题是:即使宏正常工作,并且输出正是我所期望的,为什么会生成警告?


标志和环境:

/* Mac OS X 10.9.4
   Apple LLVM version 5.1 (clang-503.0.40) (based on LLVM 3.4svn) */
cc -Wall -v -g -std=c11 -fmacro-backtrace-limit=0 -I/usr/local/include
   -c -o build/tmp/main.o main.c

更新 1:

我忘了粘贴完整的回溯!这是第一个:

main.c:39:11: warning: incompatible pointer types passing 'int *'
to parameter of type 'float *' [-Wincompatible-pointer-types]
    print((&(int){10}));
          ^~~~~~~~~~~~
main.c:31:23: note: expanded from macro 'print'
    float* : _print_f(num))
                      ^
main.c:26:29: note: passing argument to parameter 'f' here
static void _print_f(float *f){printf("float : %f\n", *f);}
                            ^

这是第二个:

main.c:40:11: warning: incompatible pointer types passing 'float *'
to parameter of type 'int *' [-Wincompatible-pointer-types]
    print((&(float){10.f}));
          ^~~~~~~~~~~~~~~~
main.c:30:23: note: expanded from macro 'print'
    int*   : _print_i(num),        \
                      ^
main.c:27:27: note: passing argument to parameter 'i' here
static void _print_i(int *i)  {printf("int   : %d\n", *i);}
                          ^

更新 2:

clang 的开发者修复这个 bug 之前,这里有一个丑陋的解决方法来消除警告,如果 assoc-list 中的所有键都是类型,或者都是指向的指针,它就会起作用类型;如果类型和指向类型的指针也在键中,将会失败:

/* HACK: re-casting pointers to mute warnings */
#define print(num) _Generic((num), \
    int*   : _print_i((int*)num),  \
    float* : _print_f((float*)num))

最佳答案

这不是 clang 中的错误,但不幸的是,这是 C11 标准所要求的。 _Generic 主表达式的所有分支都必须是有效表达式,因此在所有情况下都有效。只有一个分支会被评估的事实与此无关。

您的替代版本是 C11 为这种情况预见的:选择函数(而不是求值调用)作为泛型表达式的结果,并将该函数应用于参数。

关于c - 不兼容的指针类型传入 _Generic 宏,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/24743520/

相关文章:

c - 如何将 C 宏 (#define) 分离到 .h 和 .c 文件中

java - 在java中将word文件另存为html

c++ - C++ 中带有换行符的 printf 的奇怪行为

C - 用户输入被跳过?

c - ld 使我的所有函数都链接到头文件中的最后一个函数

c - 数组点0不重置内容

macros - SICP:可以或在 lisp 中定义为没有 gensym 的句法转换吗?

c - 如何在 C 中不打印整数值

在 C 中使用 strcpy 和 strcat 的冲突?

c - 在 Union 中存储函数指针 (C)