admin 管理员组

文章数量: 887021


2024年1月18日发(作者:gb2312在线转换为汉字)

动态链接库是实现windows应用程序共享资源、节省内存空间、提高使用效率的一个重要手段。在应用程序能够调用DLL中的函数之前,DLL文件映像必须被映射到调用进程的地址空间中。

在windows版上动态库的加载有三种方式:

a 显示加载

就是在应用程序中调用LoadLibrary函数来加载指定的动态库,并用GetProcAddress得到动态库的导出函数地址以便调用,在应用程序退出前用FreeLibrary释放动态库。

b 隐式加载

应用程序编译link的时候,加上动态库的.lib文件,那末在程序运行的时候就会自动链接指定的动态库

c 延迟加载

在编译的时候指定/delayload编译选项,应用程序运行时只有用到了动态库的中的函数的时候,才加载这个动态库。

在动态库中存在全局变量,用_declspec(thread)声明只能在隐式加载的时候正常使用,在显示加载和延迟加载的时候都会出现非法操作。DLL定义的全局变量可以被调用进程访问;DLL可以访问调用进程的全局数据。使用同一DLL的每一个进程都有自己的DLL全局变量实例。如果多个线程并发访问同一变量,则需要使用同步机制;对一个DLL的变量,如果希望每个使用DLL的线程都有自己的值,则应该使用线程局部存储(TLS,Thread Local

Strorage)。

在动态库的入口函数DllMain中,当fdwReason值为DLL_PROCESS_ATTACH和DLL_THREAD_ATTACH时用TlsAlloc()、TlsSetValue、malloc给全局变量分配内存。

例:// : Defines the entry point for the DLL application.

//

#include "stdafx.h"

#include "mydll.h"

#include "stdlib.h"

#include "stdio.h"

extern "C"{

DWORD g_temp_index;

BOOL APIENTRY DllMain( HANDLE hModule,

DWORD ul_reason_for_call,

LPVOID lpReserved )

{

char * ptr=NULL;

switch (ul_reason_for_call)

{

case DLL_PROCESS_ATTACH:

//The Dll is being mapped into the process's address space.

printf("process attach event n");

g_temp_index=TlsAlloc();

if(TlsGetValue(g_temp_index)==0)

{

ptr=(char *)malloc(sizeof(g_temp));

if(!ptr) printf("malloc error");

TlsSetValue(g_temp_index,ptr);

}

break;

case DLL_THREAD_ATTACH:

//A thread is being created

printf("thread attach event n");

if(TlsGetValue(g_temp_index)==0)

{

ptr=(char *)malloc(sizeof(g_temp));

if(!ptr) printf("malloc error");

TlsSetValue(g_temp_index,ptr);

}

break;

case DLL_THREAD_DETACH:

printf("thread detach event n");

//A thread is exiting cleanly

if(TlsGetValue(g_temp_index) != NULL)

free(TlsGetValue(g_temp_index));

break;

case DLL_PROCESS_DETACH:

if(TlsGetValue(g_temp_index) != NULL)

free(TlsGetValue(g_temp_index));

//The Dll is being mapped into the process's address space.

if(g_temp_index != 0)

TlsFree(g_temp_index);

break;

printf("process detach event n");

}

return TRUE;

}

int _g_temp(){

return((int)TlsGetValue(g_temp_index));

}

MYDLL_API int MyFun(int value,char ch)

{

}

}

g_ue=value;

printf("g_ue=%dn",g_ue);

g_lue=ch;

printf("g_lue=%cn",g_lue);

return 0;


本文标签: 动态 加载 调用 使用 时候