最近使用了c++模板,觉得非常强大,只是写起来需要掌握一点技巧。大部分模板都是直接把定义写在.h头文件,并且有些人还说这样做的原因是模板不支持分编译,可是以前的编译器对模板的支持不够好吧,但是现在完全可以。
环境:
win7 32旗舰版、VS2010 sp1
1、普通函数模板
define.h文件
template<typename T>
T GetString(int value);
define.cpp文件
#include "define.h" template<typename T>
T GetString<T>(int value)
{
return T(value);
} //偏特化
template<>
string GetString<string>(int value)
{
char psz[32] = {0};
itoa(value, psz, 10);
return string(psz);
} //偏特化
template<>
wstring GetString<wstring>(int value)
{
wchar_t psz[32] = {0};
_itow(value, psz, 10);
return wstring(psz);
} /*
如果我们直接包含define.h后,使用GetString<int>(100)函数调用,是使用不了的,会产生链接错误,为什么呢?我们上面定了函数模板的通用
实现,但是并没有实例化为一个真正的函数,既然不是函数,编译器当然不会生成代码,所以会出现链接错误;要解决这个问题的最主要就是实例化函
数模板使之成为一个真正的函数,第一种办法是:使用上面偏特化的方法,例如string和wstring类型的偏特化,因为上面已经写了,所以不多说了;
下面说第二种办法
*/
#define TEMPLATE_GET_STRING(T) template T GetString<T>(int value);
TEMPLATE_GET_STRING(int)//函数模板实例化
// 如果要在dll中导出此函数,我们可以这样
template __declspec(dllexport) int GetString<int>(int value);
2、类模板
define.h文件
template<typename T>
class A
{
public:
A()
{
InitValue();
} void InitValue(); int m_nValue;
};
define.cpp文件
template<typename T>
A<T>::A()
{
InitValue();
} template<>
void A<string>::InitValue()
{
m_nValue = 100;
} template<>
void A<int>::InitValue()
{
m_nValue = 200;
} // 类模板实例化
template
A<int>;
3、成员函数模板
define.h文件
class B
{
public:
template<typename T>
T Get(int index);
};
define.cpp文件
//成员函数模板偏特化
template<>
double B::Get<double>(int index)
{
return 2.1;
} //成员函数模板偏特化
template<>
int B::Get<int><int>(int index)
{
return 1;
}
</int>
//如果类B是一个数据库封装类,我们就可以利用这样的调用方式来获取数据库一条记录中的字段
B b;
b.Get<int>(1);//获取当前记录中第一个字段的值,并转化为int类型
b.Get<double>(2)//获取当前记录中第二个字段的值,并转化为double类型
这样是不是非常方便^^。