动态导入库和静态库: 他们实质是不一样的东西。静态库本身就包含了实际执行代码、符号表等等 ,而对于导入库而言,其实际的执行代码位于动态库中,导入库只包含了地址符号表等,确保程序找到对应函数的一些基本地址信息
问题分析: 模板函数和模板类在库中使用的时候 非常困难,但也不是不可能,因为只有相应参数类型的模板函数在库内部有实例,就能成功导出这个模板函数。对于模板类的公有成员函数们(包括构造/析构函数)全部都必须有实例存在。
为什么要将模板实例化: 只有将导出库里面的模板函数或者模板类实例化,才能将实例化后模板函数的地址信息和模板类实例化的地址信息保存在导出库中。
示范:我还是以一个例子来说明吧,这是个动态库(dynamic libraries)的例子。
//<span style="font-family:'Microsoft YaHei';">TemplateLib.h</span><span style="font-family:'Microsoft YaHei';">使用动态库</span>
#ifdefTEST_DLL_EXPORTS
#define TEST_API__declspec(dllexport)
#else
#define TEST_API__declspec(dllimport)
#endif
// 导出模板函数
template<typename T1>
TEST_APIvoidfun1(T1);
template<typename T1,typename T2>
TEST_APIvoidfun2(T1 , T2);
// 模板类
template<typename T,int size>
class TEST_APICTest
{
public:
CTest() {};
~CTest(){};
T*GetDataBuff() { return m_data;}
private:
T m_data[size];
};
// TemplateLib.cpp : 定义 DLL 应用程序的导出函数。
#include "stdafx.h"
#include "TemplateLib.h"
// 1.利用重载来实例化不同类型的模板,代码量大不说,基本上是重复的代码
// 2.库的设计者不知道用户会传入什么类型,也就是说设计者不可能实例化每一种类型的模板。
TEST_APIvoid fun1(int var1) {}
TEST_APIvoid fun1(char var1){}
template<typename T1,typename T2>
TEST_APIvoid fun2( T1 var1, T2 var2){}
// 这个名字空间不作为导出使用,唯一作用是用来例化函数模板和类模板.
namespace implement_template_private
{
voidimplement_template()
{
int idata = 10;
charchr = 'x';
float fdata = 20.f;
UINTundata= 9;
char* str = "hello";
// 这种方式的实例化,代码量比重载方式少许多,但需运行一次该模板函数
// 也许在某些时候凭空运行这个函数是不合理的。
fun2(idata,chr);// int,char
fun2(undata,str); // UINT,char*
fun2<float,char*>(fdata,str); // float,char* 显示参数
// 导出类的实例化。
// 1.除了要实例化提供给用户使用的公有成员函数外,这里面还隐含的实例化了构造函数和析构函数.
// 2.注意这里每一个模板的实例化都是唯一的。
// 3.假如客户如果在项目中使用了CTest<char,30> impl_obj; 将会连接错误, 模板的参数列表必须完全匹配。
// 4.假如该模板类非常大,功能非常多,那么实例化工作可以想象是不堪忍受的。
// 5.库的设计者不知道用户会传入什么类型,也就是说设计者不可能实例化每一种类型的模板。
CTest<char,20>impl_obj;
impl_obj.GetDataBuff();
CTest<int,5>impl_obj2;
impl_obj.GetDataBuff();
}
};
//
在另外一个项目中使用我们刚刚创建的动态库
// TemplateExport.cpp : 定义控制台应用程序的入口点。
//
#include "stdafx.h"
#include "..\TemplateLib\TemplateLib.h"
#pragma comment(lib,"TemplateLib.lib")
int _tmain(int argc, _TCHAR* argv[])
{
fun1<int>(10);
fun1<char>('x');
//fun1<float>(20.f);// 连接错误
fun2<float,char*>(20,"hello");
//fun2<int,int>(20,30); // 连接错误
CTest<char,20> test;
char* ret = test.GetDataBuff();
//CTest<char,30> test2; // 连接错误
return 0;
}
总结与建议:不建议在导出库中使用模板相关的技术,假如你能够确定用户在使用你设计的模板函数时,将传入哪些类型(type),设计者要将这些类型的模板一 一实例化。
以上就是在动态库和静态库中使用模板解决方法的全部内容和代码,希望我们的整理能够帮助到你。
您可能感兴趣的文章:Linux动态库函数的详解Linux静态库与动态库实例详解详解dll动态库的开发与调用及文件的读写小程序android studio2.3如何编译动态库的过程详解Linux下g++编译与使用静态库和动态库的方法Linux环境g++编译GDAL动态库操作方法解决Linux程序编译链接动态库版本的相关问题xcode 详解创建静态库和动态库的方法浅谈Linux C语言动态库及静态库c++实现加载so动态库中的资源linux 程序、动态库、静态库内部添加版本号和编译时间详解从源码编译Android系统的Java类库和JNI动态库的方法C#调用非托管动态库中的函数方法Visual Studio中根据系统区分引用64位、32位DLL动态库文件的配置方法动态库调用静态库示例讲解linux生成(加载)动态库静态库和加载示例方法深入探讨Linux静态库与动态库的详解(一看就懂)分析Windows和Linux动态库