编程学习笔记之c++相关::模板元的学习

      考虑这种情况,您的程序需要比较两个数值,然后返回较大的那个,数值的类型包括int、float和char,您要如何写代码?当然您可以选择传统的做法,就是乖乖写三个函数,分别是int类型、float类型和char类型。但既然我们的c++犀利在功能丰富上,所以我们当然要使用高大上的技巧,这就是此篇要讲的函数模板。先看看函数模板是如何解决上述要求的:

#include <iostream>
#include <string>
using namespace std;

template <typename T>
T max(T a, T b)
{
	return (a > b) ? a : b;
}


int main(void)
{
	int i1, i2;
	float f1, f2;
	char c1, c2;

	i1 = 2;
	i2 = 8;
	cout << "较大整数是:" << max<int>(i1, i2) << endl;

	f1 = 1.5;
	f2 = 3.14;
	cout << "较大浮点数是:" << max<float>(1.5, 3.14) << endl;

	c1 = 'b';
	c2 = 't';
	cout << "较大字符是:" << max<char>(c1, c2) << endl;
	return 0;
}

里面的template <typename T>就是我们所使用的函数模板,函数模板本身并不占用内存或生成可执行代码等,但是一旦当我们在程序中使用时,就会生成一个模板函数,而模板函数是实例存在的。以上面为例,我们写的那个max的函数模板,本身是不生成任何代码的,可是我们在cout中使用【max<int>(i1, i2)】时,max中所有的T就会被替换为尖括号中的参数int,然后程序按照max的文本在代码区中为max生成一个int版的二进制可执行代码。至于另外两个float和char,和int都是一样的道理,所以在上面的梨子中,程序一共生成了3个可调用函数,分别是max的int版本、float版本和char版本,它们位于不同的代码区地址。

      也许有朋友会问,我可以省事点,把上面的【max<int>(i1, i2)】写成【max(i1, i2)】形式吗?答案是也可以,编译器会把i1与i2本身的类型int拿给程序看。但强烈不建议这么做,因为这是比较冒险的。比如下面这段代码就无法通过编译了:

#include <iostream>
#include <string>
using namespace std;

template<typename T1,typename T2>
T1 numadd(T1 a, T2 b)
{
	b = 3;
	return a + b;
}

int main(void)
{
	int i = 9;
	numadd(i); //无法通过编译的错误代码
	numadd<int,int>(i,i); //正确的代码
	return 0;
}
这是因为在numadd(i),虽然把i的类型int拿过去了,但T2程序就不知道该怎么办了。

      此外,能给函数模板拿来当参数的,除了类型意外,“变量”也可以。上一句的变量二字为何要加引号,因为此处变量不是真正意义上的变量,而是常量伪装的变量,看代码:

#include <iostream>
#include <string>
using namespace std;

#include <iostream>
#include <string>
using namespace std;

template <typename T, int size>
void forprin(T a)
{
	T i;
	for (i = 0; i < size; i++)
		cout << a;
}

int main()
{
	int k = 5;
	forprin<int, k>(k); //错误的写法!
	forprin<int, 5>(k); //正确的写法.

	return 0;
}
其中第一个forprin的用法是无法通过的,这是因为函数模板生成模板函数时,是一个编译期的行为,而编译期是无法得到k的具体数值的,所以第二个模板参数必须要使用常量而非变量

      最后还要说明一点,函数模板是可以重载的,比如下面三个函数模板:

#include <iostream>
#include <string>
using namespace std;

#include <iostream>
#include <string>
using namespace std;

template <typename T>
void fun(T a)
{
	cout << "第一个函数模板" << endl;
}
/**********************************************/
template <typename T,typename T2>
void fun(T a)
{
	cout << "尖括号模板参数不同" << endl;
}
/********************************************/
template <typename T>
void fun(T a, T b)
{
	cout << "圆括号参数不同" << endl;
}

int main()
{
	fun<int>(1);
	fun<int ,char>(2);
	fun<int>(1, 2);
	return 0;
}
其中的函数模板fun名字都是一样的,但到了main函数中,程序会根据fun调用的具体情况【有参数个数不同的,有模板参数不同的等等】,再来决定具体生成哪个模板函数。

你可能感兴趣的:(template,模板)