DLL(Dynamic Link Library)是一段特殊的代码,它能够被外部程序在程序运行的时候调用。在DLL里面的代码可以同时被许多外部程序共享,而且不会引起手机内存的重复分配。
DLL根据接口的类型Symbian系统支持两种类型的DLL:静态接口DLL
和多态接口DLL
。
静态接口DLL在主调程序启动的时候被系统自动载入到手机内存里面(唯一的例外是如果该DLL是在ROM只读存储器里面,那么可以直接通过它的地址进行调用)。并且如果没有外部程序使用它了,它将被自动的从内存中卸载。静态接口DLL在系统内提供了一系列唯一的函数(例如,两个具有相同导出函数的DLL在系统中也不会混淆)。静态接口DLL具有.dll文件后缀,通常用于在Symbian系统中实现应用程序引擎(例如,与UI无关的代码)。
多态接口DLL是只带一个单独导出函数的普通DLL,一般都是通过调用RLibrary::Load()函数才能被载入到内存中,而且在卸载的时候,也最好通过RLibrary::Close()函数进行卸载。多个多态接口DLL可以为外部程序导出多个相同函数名的函数。所以,这种DLL通常情况下被广泛应用于实现某个应用程序框架功能扩展时候的插件。在Symbian系统中,多态接口DLL可以有多种不同的文件后缀。其中最著名的一个就是.app结尾的(应用程序),.ldd(逻辑设备驱动),.tsy和.csy(电话和通信服务模块)等等。。。
在本文中,我们将只关注静态接口DLL技术,它是你我在开发中最最常用的DLL类型。我们将以通用DLL的继续我们的内容。
静态接口的DLL
从DLL调用者的角度
来看,DLL有三个文件组成:
(1)头文件:.h的文件后缀,可以#include到主调程序的代码中去,只有在编译
的时候有用。
(2)导出文件:.lib的文件后缀,可以由主调程序进行链接,这个文件记载了DLL的提供的所有接口函数名称和地址。(在arm版本中应该是dso后缀)
(3)DLL文件本身:.dll的文件后缀包括所有.lib文件记载的接口函数的具体实现,主调函数在运行的时候实际调用和执行的部分。
从DLL编写者的角度
来看,DLL可以被看成是一个完整的Symbian项目。它由以下几部分组成:
(1)项目自己的MMP文件(在bld.inf文件中列出的)
(2)一个头文件,指定了该DLL需要导出的接口
(3)源代码文件,具体导出函数的实现
头文件
DLL的头文件与其它类的头文件的头文件非常类似。不同的地方在于使用了IMPORT_C
宏来定义所有导出函数:
class CMyEngine : public CBase
{
public:
// These functions are visible by the
// clients of the DLL and needs to have
// the IMPORT_C tag
IMPORT_C static CMyEngine* NewL();
IMPORT_C static CMyEngine* NewLC();
IMPORT_C void MyPublicMethod();
IMPORT_C void AnotherPublicMethod();
...
private:
// These functions are not visible by the
// clients of the DLL and then do not need
// the IMPORT_C tag
CMyEngine();
void ConstructL();
void SomePrivateMethod();
}
实现文件
编写DLL工作本身没有什么需要复杂的地方,但是有两个重要的地方需要留意:
(1)E32Dll()
函数必须实现
(2)另外一个特殊的宏,EXPORT_C
,应该加在每个导出函数实现的前面。
例如:
// This function is mandatory for all DLLs
EXPORT_C TInt E32Dll(TDllReason)
{
return KErrNone;
}
// This function is exported: The EXPORT_C tag shall be used.
EXPORT_C void MyPublicMethod()
{
...
}
// This one is not: The EXPORT_C tag shall not be used.
void SomePrivateMethod()
{
// Do Something
}
MMP文件
DLL的MMP文件应该有以下特点
(1)定义项目(project)的类型是dll
(2)使用正确的UID2的值(0x1000008d)
在开发过程中,你也应该通过EXPORTUNFROZEN
告诉编译环境,DLL接口还没有定稿,还可以随时修改。
例如:
TARGET MyEngine.dll
TARGETTYPE dll
UID 0x1000008d
...
EXPORTUNFROZEN
把DLL接口定稿(Freezing DLL)
一旦你完成了DLL的开发,在你发行你的DLL版本之前,应该把接口定稿(Freezing)这样可以确定将来发行的DLL可以向下兼容。
通过将项目的MMP文件中的EXPORTUNFROZEN关键字去掉,用常规的方法重新建立DLL,来实现DLL库的定稿。这时候会有“.def文件不存在”的编译警告信息出现,不要紧,继续就建立当前项目,在项目编译完毕以后,你可以用如下命令定稿:
abld freeze [platform]
例如
1、bldmake bldfiles;
2、abld build gcce;
3、abld freeze gcce;
注意:
这里通常有三个定稿的def文件夹:在二版本中编译armi时,用abld freeze armi产生的是BMARM文件夹和对应的Def;
在三版本中编译gcce和armV5时会产生EABI文件夹和对应的Def;
而用wins时则编译产生BWINS文件夹和对应的Def。
(个人总结)
所有的ARM平台共享一个.def文件,但是对于wins模拟器和winscw CodeWarrior环境来说,他们有不同的.def文件。
一旦项目已经定稿,重新生成makefile,这时候导入的lib会直接通过定稿的.def文件生成。
IMPORT_C
是一个宏,在 e32def.h 定义,意思是从 DLL 中导入函数。
#define IMPORT_C __declspec(dllexport)
如果一个 DLL 函数要被导入到应用程序那么这个函数必须首先以 IMPORT_C 声明,告诉编译器,这个函数可以在 DLL 中找到。
EXPORT_C
也是一个在 e32def.h 定义的宏,意思是从 DLL 中导出函数。
#define EXPORT_C __declspec(dllexport)
如果一个函数是 DLL API 的一部分, 并且要在应用程序或其他 DLL 中被调用,那么这个函数在C++源文件中必须实现为导出。
需要注意的是在 WINS 下, IMPORT_C 和 EXPORT_C 同时被定义为 __declspec(export)。
分享到:
相关推荐
生成的DLL文件和对应的.lib静态导入库文件是链接到使用DLL的程序的关键。 4. **调用DLL**:有两种方式调用DLL中的函数,隐式链接和显式链接。 - **隐式链接**(implicit):在使用DLL的程序中,通过链接器直接...
为了实现Qt静态加载DLL,你需要在项目文件(.pro)中指定DLL的路径,并在代码中使用`Q_DECL_EXPORT`和`Q_DECL_IMPORT`宏来控制导出和导入函数。例如: ```cpp // 在DLL的头文件中 #ifdef MYDLL_EXPORTS #define MYDLL...
在DLL的源文件中使用`Q_DECL_EXPORT`,在使用DLL的客户端代码中使用`Q_DECL_IMPORT`。 5. **构建项目**:在Qt Creator中配置项目的构建步骤,确保选择了生成DLL的选项。这将生成一个.dll文件,以及对应的.lib(导...
然后,如同使用静态库一样,直接包含DLL的头文件并使用其功能。 ```cpp #include "mydll.h" // 使用DLL中的类和函数 ``` 在实际应用中,DLL的使用还涉及到版本控制、线程安全、异常处理等多个方面。注意,如果DLL...
创建MFC DLL时,通常需要在MFC库中定义类和接口,使用MFC的宏来导出函数,然后在主应用程序中通过`#import`语句或者`LoadLibrary`和`GetProcAddress`函数来加载并使用这些函数。此外,还需要考虑版本控制和兼容性...
- 在源代码中使用`Q_DECL_EXPORT`或`Q_DECL_IMPORT`宏进行导出和导入声明。 - 编写实现DLL功能的C++类和函数。 - 在主工程中,将DLL作为外部库引入,通过`#include`指令引用头文件,并使用`LoadLibrary`和`...
在创建完 DLL 工程之后,下一步是编写 DLL 的对外接口。这一步非常重要,因为正确的接口设计可以提高 DLL 的可重用性和灵活性。 1. **确定接口需求**: - 明确 DLL 将要提供的功能和服务。 - 考虑到不同场景的...
这里,`CDLL`是`ctypes`模块中的一个类,用于加载动态链接库并提供了一个接口去调用其中的函数。 ### 知识点四:调用C函数 一旦DLL被加载到Python环境中,就可以像调用普通Python函数一样调用C函数了。但在调用前...
extern "C" __declspec(dllexport) void MyFunction(); ``` - **实现导出函数**:在对应的源文件中实现这些函数,提供功能实现。 - **编译DLL**:使用VS2005编译该工程,生成.dll和.lib文件。.lib文件是导入库,...
使用`#import`指令可以自动处理DLL的导入过程,或者手动使用`LoadLibrary`和`GetProcAddress`函数加载和调用DLL中的函数。 7. **错误处理和调试** 在开发过程中,确保正确处理DLL中的异常和错误,使用`try-catch`...
静态链接库(Static Library)在编译时会被合并到目标程序中,形成一个独立的可执行文件,而导入库(Import Library)主要用于动态链接,它包含指向DLL中实际函数地址的引用。 在VS2015中创建DLL项目,你需要: 1. ...
3. **导出符号**:在C++中,为了使DLL的函数和类可供外部使用,需要使用`__declspec(dllexport)`或`Q_DECL_EXPORT`宏来导出符号。如果你是在DLL的使用者端,你需要使用`__declspec(dllimport)`或`Q_DECL_IMPORT`宏。...
- Non-MFC DLL不依赖MFC(Microsoft Foundation Classes),导出标准C接口,可用于非MFC或MFC程序。 - MFC Regular DLL包含CWinApp类但没有消息循环,适合扩展MFC应用程序功能。 - MFC Extension DLL使用MFC的...
- **静态导入**:需要`.lib`文件,通过`#pragma comment(lib,"xxx.lib")`或项目设置导入,然后使用`extern "C" _declspec(import)`声明导入函数。 - **动态导入**:在运行时使用API `LoadLibrary`加载DLL,`...
在对应的实现文件中,实现这些函数和类,并使用`Q_DECL_IMPORT`宏在使用DLL的地方导入。例如: ```cpp #include "mydll.h" QT_BEGIN_NAMESPACE MyDll::MyDll(QObject *parent) : QObject(parent) {} MyDll::~...
6. **接口设计**:为了使类在DLL和客户端之间安全地交互,通常会设计一个接口类,该接口只包含纯虚函数。类的实现细节(如数据成员)留在DLL内部,而客户端通过接口与DLL进行通信。这样可以更好地隐藏实现细节,降低...
这些函数或类需要使用`__declspec(dllexport)`修饰符标记为可导出。 3. **配置项目**:确保项目的配置设置正确,使其能够生成一个DLL文件而非可执行文件。 4. **编译项目**:编译项目生成DLL文件。 #### 在C#项目中...
1. **导出函数和类**:在DLL中,你需要使用`__declspec(dllexport)`来导出函数和类,而在主程序中则使用`__declspec(dllimport)`来导入它们。正确的导出和导入是实现DLL功能的关键。 2. **链接问题**:处理好静态...
3. **导入库**:在Qt代码中,使用Q_DECL_IMPORT和Q_DECL_EXPORT宏来正确声明DLL中的接口。在DLL的头文件中,将导出的函数和类用Q_DECL_EXPORT标记;在使用DLL的Qt项目中,使用Q_DECL_IMPORT。 4. **加载和调用DLL**...