Win32 的dll导入

dll 文件可以导入变量,函数,和C++类,但是导入变量会使执行程序与dll紧耦合,而C++类导入则需要两个文件的开发商所用的编译器相兼容,所以做好只导入函数;

创建dll : 
头文件:
#ifdef               MYLIBAPI                    //在dll源文件中必须定义这个宏为导出宏
#else                    //因为执行文件中必然没有定义这个宏所以
#define MYLIBAPI   __declspec(dllimport) //在执行文件中必定为导入宏
#endif
MYLIBAPI int g_nResult;                        //导入变量
MYLIBAPI int Add(int nLeft, int nRight);  //导入函数

源文件:
#include <windows.h>
#define MYLIBAPI   __declspec(dllexport)    //屏蔽掉头文件的导入宏为导出宏
#include "MyLib.h"
int g_nResult;
int Add(int nLeft, int nRight) {g_nResult = nLeft + nRight;   return(g_nResult);  }

隐式链接:
在mfc中调用win32 dll 的步骤:   (导出函数 /一般不导出c++类:可以只导出部分public函数)
1: 不要用extern "C",单独用_declspec (dllexport) ..... 生成dll 及 lib 文件;
2: 将lib 文件复制到当前程序目录,且将链接器的   输入->付加依赖项 中写入库文件;
3: 复制dll文件到debug 目录(必须的);
4:在调用函数时, 采用先声明外部函数 在使用;
 4.1: 用extern 来声明函数; 
 4.2: 用_declspec(dllimport) 来声明导入的函数 ;  (采用后者效率通常会更高)
对于带头文件的导入方式:
1: 其中第二三步不变, 采用文上的定义方式;
2:将头文件装置程序目录,并在原来声明外部函数的地方 #include<文件.h>,而不需要在显示声明函数原型了;

其中的 extern "C" 的作用是告诉c++编译器 不要改动所要导出的类型的名称,以防在别的模块中无法访问; (当在写c和c++的混合代码的时候,如果不加则会导致严重的链接问题)
当你用C++编译器写的dll,再用c语言的编译器来导入就会出错,而还是用同一种编译器便宜的话则不会出错;
可以通过dumpbin.exe 来查看所谓导出或者需要导入的dll 的信息;(exports /imports)
名字改编: 为了支持重载,按照自己的一套规则来改变函数的名字
这就导致了,要就要在客户端,和服务端同时用extern "C" 来用相同的规则来寻找同一个函数名;(要不就不要加)(用这个特性的话就不可以导出类函数;)

显示链接:(动态加载一个动态链接库) 这时候即使采用不同的调用约定(当然访问时,约定也要相同),其函数名也没有改变;
且不会dumpbin出需要动态加载的文件;   
1. 包含一个 .def文件  :文件中含有  EXPORTS    functionname  (这样就会只导出一个不会变的函数名称) 如果不用.def文件的话,可以在加载的时候,显示的使用变化后的函数名
将次dll文件的项目属性的   输入->模块定义文件-> 为 name.def                    就可以生成相应的.lib 文件了
(1)或者是                #pragma comment(linker, "/export:[email protected]")
2.HMODULE LoadLibrary("dll or exe name ");           导入模块                                       这里返回的HMODULE句柄就相当于HINSTANCE,其实是dll文件加载后所在的虚拟内存地址
3.FARPROC GetProAddress();   获得函数入口地址
例如: typedef  int (* addproc)(int a ,int b);
addproc Add=(addproc)GetProcAddress(hmoudle,"add");                     或者使第二个函数为 MAKEINTERESOURCE(序号);
Add(10,10);
FreeLibrary(hmoudle);    卸载动态链接库
注: 在非vc情况下调用一个c++(非改名的c++或者是C)的dll 通常会出现的问题:是不是标准调用,如果是的话则函数名即使指定extern也会改变
原因:由于:ms 的c编译器即使在你没有使用c++代码的时候也会改变c 函数的名字, 只要在你调用函数的时候使用了WINAPI 的_stdcall(绝大多数的情况), 这时候编译器会产生一个有下划线起始的,以@+参数所占有的字节数 结尾的函数; 而ms的编译器可以正确翻译出相应的名字;
在使用其它非ms编译器来调用可用dll的时候,必须要使用以下两种方法:(指示其不要将函数改名)
    
DllMain() dll 文件的入口函数 (是可选的)   可以为自己的函数提供一个本dll文件的句柄来供dll 函数来使用;

如果在dll文件中分配内存的话,那么一定要提供相应的释放函数在dll中释放内存,即一个模块提供了分配内存的函数,那么它就同时也要提供释放内存的函数.

时间: 2024-12-20 16:35:19

Win32 的dll导入的相关文章

在Delphi中静态调用DLL 引用外部Dll External Dll 导入Dll

  调用一个DLL比写一个DLL要容易一些.首先给大家介绍的是静态调用方法,稍后将介绍动态调用方法,并就两种方法做一个比较.同样的,我们先举一个静态调用的例子. unit Unit1; interface uses Windows, Messages, SysUtils, Classes, Graphics, Controls, Forms, Dialogs, StdCtrls; type TForm1 = class(TForm) Edit1: TEdit; Button1: TButton;

编写DLL所学所思(1)——导出函数

烛秋  http://www.cnblogs.com/cswuyg/archive/2011/09/30/dll.html 动态链接库的使用有两种方式,一种是显式调用.一种是隐式调用. (1)       显式调用:使用LoadLibrary载入动态链接库.使用GetProcAddress获取某函数地址. (2)       隐式调用:可以使用#pragma comment(lib, “XX.lib”)的方式,也可以直接将XX.lib加入到工程中. DLL的编写 编写dll时,有个重要的问题需要

Windows Dll Injection、Process Injection、API Hook

catalogue 1. 引言 2. 使用注册表注入DLL 3. 使用Windows挂钩来注入DLL 4. 使用远程线程来注入DLL 5. 使用木马DLL来注入DLL 6. 把DLL作为调试器来注入 7. 使用createprocess来注入代码 8. api拦截 9. Detours - Inline Hook 1.  引言 应用程序需要跨越进程边界来访问另一个进程的地址空间的情况如下 1. 我们想要从另一个进程创建的窗口派生子类窗口 2. 我们需要一些手段来辅助调试,例如我们需要确定另一个进

[转]windows下VS2010中lib与dll文件的生成与使用

原文地址:https://my.oschina.net/SysuHuyh5LoveHqq/blog/644622 近期在windows下开发了某个程序,需要将其生成静态文件(lib)以及动态库文件(dll),其中参考了不少帖子,有的讲得也不是很清楚明白,先将本人实践过的记录一下,供后期自己查询,也供各位大牛点评. 一.lib文件的生成与使用 1.lib的生成 相对来说,静态库文件还是比较容易生成和使用的,在代码上,貌似也不需要更改什么,举例说明: 头文件函数声明形式如下: extern bool

在VS2012中采用C++中调用DLL中的函数 (4)

这两天因为需要用到VS2012来生成一个DLL代码,但是之前并没有用过DLL相关的内容,从昨天开始尝试调试DLL的文件调用,起初笔者在网络上找到了3片采用VSXXX版本进行调试的例子,相关的内容见本人Blog在C++中调用DLL中的函数(1)(2)(3) 但是问题出现了,上面讲述的步骤很详细但是在自己运行的时候却会出现错误,于是今天在使用google“vs2012 c++调用dll”时,发现了一篇MSDN上的指南:http://msdn.microsoft.com/zh-cn/library/m

VC++:创建,调用Win32动态链接库

VC++:创建,调用Win32动态链接库 概述 DLL(Dynamic Linkable Library)动态链接库,Dll可以看作一种仓库,仓库中包含了可以直接使用的变量,函数或类.仓库的发展史经历了"无库" ---> "静态链接库"  ---> "动态链接库".静态链接库与动态链接库都能实现共享代码,如果使用静态链接库,编译后lib中的指令会被包含在生成的EXE文件中,如果使用动态链接库,则不会被包含到EXE文件中,EXE文件执行

C++ DLL导出类 知识大全

在公司使用C++ 做开发,公司的大拿搭了一个C++的跨平台开发框架.在C++开发领域我还是个新手,有很多知识要学,比如Dll库的开发. 参考了很多这方面的资料,对DLL有一个基本全面的了解.有一个问题让我有点困惑,普通的导入导出C++类的方式都是使用_declspec(dllexport) /_declspec(dllimport)来导入导出类,但是在公司的开发中我们没有导入导出,而是定义了一些只有纯虚函数的抽象类,然后定义了一个工厂类,将这个工厂类注册到框架的服务中心中,使用时从服务中心拿到这

深入理解 Win32 PE 文件格式

深入理解 Win32 PE 文件格式 Matt Pietrek 这篇文章假定你熟悉C++和Win32. 概述 理解可移植可执行文件格式(PE)可以更好地了解操作系统.如果你知道DLL和EXE中都有些什么东西,那么你就是一个知识渊博的程序员.这一系列文章的第一部分,讨论最近这几年PE格式所发生的变化. 这次更新后,作者讨论了PE格式如何适应于用.NET开发的应用程序,包括PE节,RVA,数据目录,以及导入函数.附录中包含了相关的映像头结构以及它们的描述. 很早以前,我为微软系统期刊(现在叫做MSD

在VS2012中采用C++中调用DLL中的函数(4)

转自:http://www.cnblogs.com/woshitianma/p/3683495.html 这两天因为需要用到VS2012来生成一个DLL代码,但是之前并没有用过DLL相关的内容,从昨天开始尝试调试DLL的文件调用,起初笔者在网络上找到了3片采用VSXXX版本进行调试的例子,相关的内容见本人Blog在C++中调用DLL中的函数(1)(2)(3) 但是问题出现了,上面讲述的步骤很详细但是在自己运行的时候却会出现错误,于是今天在使用google"vs2012 c++调用dll"