今天看到一个有意思的知识点,赶紧记录一下,总所周知,C语言没有泛型,但是C11新增了一种表达式,叫做泛型选择表达式(_ Generic),这个是干啥的呢?它的作用是根据表达式的类型选择一个值,让我们来看一看它具体的语法
泛型选择引入
先看一段代码:
1 2 3 4 5 6 7 8 9 10
| int main() { int x = 1; double y = 2.0; char z = 'c'; printf("%d\n", _Generic(x, int:0, double : 1, default:3)); printf("%d\n", _Generic(y, int:0, double : 1, default:3)); printf("%d\n",_Generic(z, int:0, double : 1, default:3)); return 0; }
|
_Generic是C11关键字,后面的圆括号中包含有多个用逗号分隔的项,第一个项是表达式,后面的每一个项都由一个类型、一个冒号和一个值组成,如double: 1。第一个项的类型匹配哪一个标签,整个表达式的值就是该标签后面的值
运行结果如下:

可以看到,根据传入的变量的类型,打印的结果也不同,第一个printf的第一项x是int,那么整个表达式的结果为0,第二个printf的第一项是double,表达式的结果就是1,第三个printf打印结果为3是因为char类型没有匹配,走了默认的default,结果为3
其实,这个东西很像switch语句,只是Generic用表达式类型匹配标签,switch用表达式的值匹配标签
与宏定义组合
我们可以看到啊,上边的用法还是比较恶心麻烦的,但是呢,我们可以结合宏定义组合去玩这个东西,这样就会非常的方便
直接看例子:
1 2 3 4 5 6 7 8 9 10 11 12
| #define MYTYPE(X) _Generic((X),int:"int", double:"double", default:"other")
int main() { int d = 2; printf("%s\n", MYTYPE(d)); printf("%s\n", MYTYPE(1.0*d)); printf("%s\n", MYTYPE("string"));
return 0; }
|
运行结果如下:

与宏定义结合是不是感觉好多了,甚至有点C++泛型编程的感觉了,但是其实还差点。
进阶玩法
_ Generic标签对应的值,可以是整形,也可以是字符串,当然也可以是函数指针。
我们来看一下下边的这段代码。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34
| void PrintInt(int x) { printf("%d\n", x); } void PrintDouble(double x) { printf("%lf\n", x); } void PrintSting(char* x) { printf("%s\n", x); }
void PrintOther(void x) { print("类型有点问题\n"); } #define PRINT(X) _Generic((X),\ int:PrintInt,\ double:PrintDouble,\ const char*:PrintSting,\ default:PrintOther)(X)
int main() { int x = 1; int y = 2.0; const char* str = "hello _Generic"; PRINT(x); PRINT(y); PRINT(str); return 0; }
|
运行结果如下:

这样是不是和C++的泛型编程很像了呢,是不是挺有意思的,当然还是C++的泛型好用一些。