网站建设售后服务,重网站建设,营销网站定制公司,无极网站在回答标题问题之前#xff0c;先了解下什么是泛型编程。泛型编程#xff08;generic programming#xff09;是程序设计语言的一种风格或范式。泛型允许程序员在强类型程序设计语言中编写代码时使用一些以后才指定的类型#xff0c;在实例化时作为参数指明这些类型。C支持… 在回答标题问题之前先了解下什么是泛型编程。泛型编程generic programming是程序设计语言的一种风格或范式。泛型允许程序员在强类型程序设计语言中编写代码时使用一些以后才指定的类型在实例化时作为参数指明这些类型。C支持泛型编程也就是模板比如#include
template class T
T add(T a,T b){T ret a b;std::cout a b ret std::endl;return ret;
}
int main(){add(1,2); // 整数相加add(1.2,2.3); // 浮点数相加return 0;
}运行结果1 2 3
1.2 2.3 3.5从上面的结果可以看到对于调用add函数如果传入的是整型则按照整型加法计算如果是浮点数则按照浮点数进行加法计算。也就是说add函数没有针对特定类型泛型。你同样可以使用重载实现上面的功能但是存在大量重复代码。C语言支持泛型编程吗很遗憾C语言本身不支持真正意义上的泛型编程但是却在一定程度上可以“实现泛型编程”。_Generic关键字_Generic是C11的关键字通过该关键字可以有一个泛型表达式_Generic((value). int:int, float:float,char*:char*,default:other type)什么意思呢如果value是int类型那么表达式的值就是“int”其他的以此类推。看起来是不是和switch语句有点类似呢根据这个示例我们来实现一个功能打印变量或常量到底是什么类型#include
#define TYPE(v) _Generic((v), \int:int, \char:char, \float:float, \double:double, \char*:char*, \default:other type)
int main(void)
{printf(1 2 type: %s\n,TYPE(1 2));printf(1/3 type: %s\n,TYPE(1/3));printf(2/3 type: %s\n,TYPE((float)2/3));printf(xxx type: %s\n,TYPE(xxx));return 0;
}这里为了方便使用我们通过define关键字将泛型表达式简化。运行结果1 2 type: int
1/3 type: int
2/3 type: float
xxx type: char*可以看到通过TYPE就可以获得表达式的结果类型这对初学者来说可真是福音了。泛型算法既然C语言有_Generic关键字了那么我们尝试实现开头C示例代码中的加法。看过上面的例子后相信你已经会了#include
// int类型加法
int addI(int a, int b)
{printf(%d %d %d\n,a,b, a b );return (a b);
}
// double类型加法
double addF(double a, double b)
{printf(%f %f %f\n,a,b, a b );return (a b);
}
void unsupport(int a,int b)
{printf(unsupport type\n);
}
#define ADD(a,b) _Generic((a), \int:addI(a,b),\double:addF(a,b), \default:unsupport(a,b))
int main(void)
{ADD(1 , 2);ADD(1.1,2.2);return 0;
}观察上面的代码我们注意到在这里我们需要定义两种类型的加法实际上通过C的模板由编译器帮我们完成了这件事由于C语言中并不支持重载因此两个加法的函数名不一样。由于涉及参数有两个在做类型判断时如果两个参数不一致可能仍然存在编译问题调用者无需区分被加对象是什么类型都可以统一使用ADDC99的tgmath.h前面说到_Generic关键字在C11中才有那么C99怎么办呢实际上tgmath.h中提供了一些泛型类型宏如果math.h的函数中定义了floatdouble和long double版本tgmath就会提供一个泛型类型宏。效果和前面的例子一样举个例子#include
#include
int main(void)
{float f 4.0f;long double d 1.44;printf(%f\n,sqrt(f)); // 实际上调用了sqrtfprintf(%Lf\n,sqrt(d)); // 实际上调用了sqrtlreturn 0;
}编译运行结果2.000000
1.200000但是不得不说tgmath中提供的泛型宏也是有限的。声明本文于网络整理版权归原作者所有如来源信息有误或侵犯权益请联系我们删除或授权事宜。