同学们动态加载dll的时候是不是感觉挺麻烦的,每次都::LoadLibrary,::GetProcAddress,还要typedef一堆函数。最近闲来无聊,用宏封装了一下,可以少写不少代码,用来也挺方便的。

不过最多支持5个参数,同学们可以自己加。等着下次用flexbison写一个生成工具,我感觉还是不错的。

 #pragma once
/*! \brief 动态加载dll,并调用导出函数
author:12178761@qq.com
*/
#define DLLLOADER_BEGIN( NAME)\
class DLLLOADER_##NAME\
{\
public:\
static BOOL Init( LPCWSTR sDllPath){\
if ( RefHModule())\
{\
return TRUE;\
}\
RefHModule() = ::LoadLibrary( sDllPath);\
if ( !RefHModule())\
{\
return FALSE;\
}\
return TRUE;\
}\
static void UnInit(){\
if ( RefHModule())\
{\
FreeLibrary( RefHModule());\
RefHModule() = NULL;\
}\
}\
static HMODULE& RefHModule(){\
static HMODULE M = NULL;\
return M;\
}
#define DLLLOADER_ENTRY_0( FUNNAME, FUNDESC, FUNRET, FUNRETDEF)\
typedef FUNRET (FUNDESC *_##FUNNAME)();\
static _##FUNNAME Ref##FUNNAME(){\
_##FUNNAME F = (_##FUNNAME)::GetProcAddress( RefHModule(), #FUNNAME);\
return F;\
}\
static FUNRET FUNNAME(){\
if ( Ref##FUNNAME())\
{\
return Ref##FUNNAME()();\
}\
return FUNRETDEF;\
}
#define DLLLOADER_ENTRY_VOID0( FUNNAME, FUNDESC)\
typedef void (FUNDESC *_##FUNNAME)();\
static _##FUNNAME Ref##FUNNAME(){\
_##FUNNAME F = (_##FUNNAME)::GetProcAddress( RefHModule(), #FUNNAME);\
return F;\
}\
static void FUNNAME(){\
if ( Ref##FUNNAME())\
{\
Ref##FUNNAME()();\
}\
}
#define DLLLOADER_ENTRY_1( FUNNAME, FUNDESC, FUNRET, FUNRETDEF, PARAMTYPE1)\
typedef FUNRET (FUNDESC *_##FUNNAME)(PARAMTYPE1);\
static _##FUNNAME Ref##FUNNAME(){\
_##FUNNAME F = (_##FUNNAME)::GetProcAddress( RefHModule(), #FUNNAME);\
return F;\
}\
static FUNRET FUNNAME( PARAMTYPE1 PARAM1){\
if ( Ref##FUNNAME())\
{\
return Ref##FUNNAME()( PARAM1);\
}\
return FUNRETDEF;\
}
#define DLLLOADER_ENTRY_VOID1( FUNNAME, FUNDESC, PARAMTYPE1)\
typedef void (FUNDESC *_##FUNNAME)(PARAMTYPE1);\
static _##FUNNAME Ref##FUNNAME(){\
_##FUNNAME F = (_##FUNNAME)::GetProcAddress( RefHModule(), #FUNNAME);\
return F;\
}\
static void FUNNAME( PARAMTYPE1 PARAM1){\
if ( Ref##FUNNAME())\
{\
Ref##FUNNAME()( PARAM1);\
}\
}
#define DLLLOADER_ENTRY_2( FUNNAME, FUNDESC, FUNRET, FUNRETDEF, PARAMTYPE1, PARAMTYPE2)\
typedef FUNRET (FUNDESC *_##FUNNAME)(PARAMTYPE1, PARAMTYPE2);\
static _##FUNNAME Ref##FUNNAME(){\
_##FUNNAME F = (_##FUNNAME)::GetProcAddress( RefHModule(), #FUNNAME);\
return F;\
}\
static FUNRET FUNNAME( PARAMTYPE1 PARAM1, PARAMTYPE2 PARAM2){\
if ( Ref##FUNNAME())\
{\
return Ref##FUNNAME()( PARAM1, PARAM2);\
}\
return FUNRETDEF;\
}
#define DLLLOADER_ENTRY_VOID2( FUNNAME, FUNDESC, PARAMTYPE1, PARAMTYPE2)\
typedef void (FUNDESC *_##FUNNAME)(PARAMTYPE1, PARAMTYPE2);\
static _##FUNNAME Ref##FUNNAME(){\
_##FUNNAME F = (_##FUNNAME)::GetProcAddress( RefHModule(), #FUNNAME);\
return F;\
}\
static void FUNNAME( PARAMTYPE1 PARAM1, PARAMTYPE2 PARAM2){\
if ( Ref##FUNNAME())\
{\
Ref##FUNNAME()( PARAM1, PARAM2);\
}\
}
#define DLLLOADER_ENTRY_3( FUNNAME, FUNDESC, FUNRET, FUNRETDEF, PARAMTYPE1, PARAMTYPE2, PARAMTYPE3)\
typedef FUNRET (FUNDESC *_##FUNNAME)(PARAMTYPE1, PARAMTYPE2, PARAMTYPE3);\
static _##FUNNAME Ref##FUNNAME(){\
_##FUNNAME F = (_##FUNNAME)::GetProcAddress( RefHModule(), #FUNNAME);\
return F;\
}\
static FUNRET FUNNAME( PARAMTYPE1 PARAM1, PARAMTYPE2 PARAM2, PARAMTYPE3 PARAM3){\
if ( Ref##FUNNAME())\
{\
return Ref##FUNNAME()( PARAM1, PARAM2, PARAM3);\
}\
return FUNRETDEF;\
}
#define DLLLOADER_ENTRY_VOID3( FUNNAME, FUNDESC, PARAMTYPE1, PARAMTYPE2, PARAMTYPE3)\
typedef void (FUNDESC *_##FUNNAME)(PARAMTYPE1, PARAMTYPE2, PARAMTYPE3);\
static _##FUNNAME Ref##FUNNAME(){\
_##FUNNAME F = (_##FUNNAME)::GetProcAddress( RefHModule(), #FUNNAME);\
return F;\
}\
static void FUNNAME( PARAMTYPE1 PARAM1, PARAMTYPE2 PARAM2, PARAMTYPE3 PARAM3){\
if ( Ref##FUNNAME())\
{\
Ref##FUNNAME()( PARAM1, PARAM2, PARAM3);\
}\
}
#define DLLLOADER_ENTRY_4( FUNNAME, FUNDESC, FUNRET, FUNRETDEF, PARAMTYPE1,\
PARAMTYPE2, PARAMTYPE3, PARAMTYPE4)\
typedef FUNRET (FUNDESC *_##FUNNAME)(PARAMTYPE1, PARAMTYPE2, PARAMTYPE3, PARAMTYPE4);\
static _##FUNNAME Ref##FUNNAME(){\
_##FUNNAME F = (_##FUNNAME)::GetProcAddress( RefHModule(), #FUNNAME);\
return F;\
}\
static FUNRET FUNNAME( PARAMTYPE1 PARAM1, PARAMTYPE2 PARAM2, PARAMTYPE3 PARAM3, PARAMTYPE4 PARAM4){\
if ( Ref##FUNNAME())\
{\
return Ref##FUNNAME()( PARAM1, PARAM2, PARAM3, PARAM4);\
}\
return FUNRETDEF;\
}
#define DLLLOADER_ENTRY_VOID4( FUNNAME, FUNDESC, PARAMTYPE1,\
PARAMTYPE2, PARAMTYPE3, PARAMTYPE4)\
typedef void (FUNDESC *_##FUNNAME)(PARAMTYPE1, PARAMTYPE2, PARAMTYPE3, PARAMTYPE4);\
static _##FUNNAME Ref##FUNNAME(){\
_##FUNNAME F = (_##FUNNAME)::GetProcAddress( RefHModule(), #FUNNAME);\
return F;\
}\
static void FUNNAME( PARAMTYPE1 PARAM1, PARAMTYPE2 PARAM2, PARAMTYPE3 PARAM3, PARAMTYPE4 PARAM4){\
if ( Ref##FUNNAME())\
{\
Ref##FUNNAME()( PARAM1, PARAM2, PARAM3, PARAM4);\
}\
}
#define DLLLOADER_ENTRY_5( FUNNAME, FUNDESC, FUNRET, FUNRETDEF, PARAMTYPE1,\
PARAMTYPE2, PARAMTYPE3, PARAMTYPE4, PARAMTYPE5)\
typedef FUNRET (FUNDESC *_##FUNNAME)(PARAMTYPE1, \
PARAMTYPE2, PARAMTYPE3, PARAMTYPE4, PARAMTYPE5);\
static _##FUNNAME Ref##FUNNAME(){\
_##FUNNAME F = (_##FUNNAME)::GetProcAddress( RefHModule(), #FUNNAME);\
return F;\
}\
static FUNRET FUNNAME( PARAMTYPE1 PARAM1, PARAMTYPE2 PARAM2, \
PARAMTYPE3 PARAM3, PARAMTYPE4 PARAM4, PARAMTYPE5 PARAM5){\
if ( Ref##FUNNAME())\
{\
return Ref##FUNNAME()( PARAM1, PARAM2, PARAM3, PARAM4, PARAM5);\
}\
return FUNRETDEF;\
}
#define DLLLOADER_ENTRY_VOID5( FUNNAME, FUNDESC, PARAMTYPE1,\
PARAMTYPE2, PARAMTYPE3, PARAMTYPE4, PARAMTYPE5)\
typedef void (FUNDESC *_##FUNNAME)(PARAMTYPE1, \
PARAMTYPE2, PARAMTYPE3, PARAMTYPE4, PARAMTYPE5);\
static _##FUNNAME Ref##FUNNAME(){\
_##FUNNAME F = (_##FUNNAME)::GetProcAddress( RefHModule(), #FUNNAME);\
return F;\
}\
static void FUNNAME( PARAMTYPE1 PARAM1, PARAMTYPE2 PARAM2, \
PARAMTYPE3 PARAM3, PARAMTYPE4 PARAM4, PARAMTYPE5 PARAM5){\
if ( Ref##FUNNAME())\
{\
Ref##FUNNAME()( PARAM1, PARAM2, PARAM3, PARAM4, PARAM5);\
}\
}
#define DLLLOADER_END()\
}; /*
例子:
DLLLOADER_BEGIN( SAMPLE)
DLLLOADER_ENTRY_0( Function1, BOOL, FALSE, )
DLLLOADER_ENTRY_VOID0( Function2, )
DLLLOADER_ENTRY_1( Function3, BOOL, FALSE, , LPCWSTR)
DLLLOADER_END() DLLLOADER_BEGIN(KERNEL32)
DLLLOADER_ENTRY_1( LoadLibraryW, WINAPI, HMODULE, 0, LPCWSTR)
DLLLOADER_ENTRY_3( GetModuleFileNameW, WINAPI, DWORD, 0, HMODULE, LPWSTR, DWORD)
DLLLOADER_END() int _tmain(int argc, _TCHAR* argv[])
{
DLLLOADER_SAMPLE::Init( L"C:\\Users\\sosopop\\Documents\\Visual Studio 2005\\Projects\\testHandle\\debug\\testLoadDll.dll");
DLLLOADER_SAMPLE::Function1();
DLLLOADER_SAMPLE::Function2();
DLLLOADER_SAMPLE::Function3(L"Function3 Hello");
DLLLOADER_SAMPLE::Function4(L"Function4", L"Hello");
DLLLOADER_SAMPLE::UnInit(); DLLLOADER_KERNEL32::Init( L"c:\\windows\\system32\\kernel32.dll");
HMODULE h =DLLLOADER_KERNEL32::LoadLibraryW( L"user32.dll");
WCHAR sFileName[MAX_PATH]={0};
DWORD nSize = DLLLOADER_KERNEL32::GetModuleFileNameW( NULL, sFileName, _countof(sFileName)); return 0;
}
*/

用宏定义封装LoadLibrary,方便的动态加载dll的更多相关文章

  1. Delphi静态加载DLL和动态加载DLL示例

    下面以Delphi调用触摸屏动态库xtkutility.dll为例子,说明如何静态加载DLL和动态加载DLL. 直接上代码. 1.静态加载示例 unit Unit1; interface uses W ...

  2. c#实现动态加载Dll(转)

    c#实现动态加载Dll 分类: .net2009-12-28 13:54 3652人阅读 评论(1) 收藏 举报 dllc#assemblynullexceptionclass 原理如下: 1.利用反 ...

  3. 动态加载dll的实现+远线程注入

    1.在目标进程中申请内存 2.向目标进程内存中写入shellcode(没有特征,编码比较麻烦) 3.创建远线程执行shellcode 之前可以看到shellcode很难编写还要去依赖库,去字符串区等等 ...

  4. C# 利用反射动态加载dll

    笔者遇到的一个问题,dll文件在客户端可以加载成功,在web端引用程序报错.解决方法:利用反射动态加载dll 头部引用加: using System.Reflection; 主要代码: Assembl ...

  5. unity3d动态加载dll的API以及限制

    Unity3D的坑系列:动态加载dll 一.使用限制 现在参与的项目是做MMO手游,目标平台是Android和iOS,iOS平台不能动态加载dll(什么原因找乔布斯去),可以直接忽略,而在Androi ...

  6. Unity3D的坑系列:动态加载dll

    我现在参与的项目是做MMO手游,目标平台是Android和iOS,iOS平台不能动态加载dll(什么原因找乔布斯去),可以直接忽略,而在Android平台是可以动态加载dll的,有了这个就可以实现代码 ...

  7. C#,动态加载DLL,通过反射,调用参数,方法,窗体

    .net中常会用到动态加载DLL,而DLL中可能包含各种参数.方法.窗体,如何来调用动态加载这些参数.方法.窗体呢? 在C#中,我们要使用反射,首先要搞清楚以下命名空间中几个类的关系: System. ...

  8. 动态加载Dll时,通过Type生成类对象

    原文:动态加载Dll时,通过Type生成类对象 转:http://www.cnblogs.com/zfanlong1314/p/4197383.html "反射"其实就是利用程序集 ...

  9. c#实现动态加载Dll

    原文:c#实现动态加载Dll 原理如下: 1.利用反射进行动态加载和调用. Assembly assembly=Assembly.LoadFrom(DllPath); //利用dll的路径加载,同时将 ...

随机推荐

  1. Flexigrid在IE下不显示数据的处理

    文章总结自我的论坛提问: http://bbs.csdn.net/topics/390498434?page=1#post-394918028 解决方法: 网上的答案经我验证都是不靠谱的,以后大家就知 ...

  2. Linux基础教程之/dev/null和/dev/zero的区别及其用法

    在Linux操作系统中/dev/null和/dev/zero是两个相似却又很特殊的文件,特别是在shell脚本开发和系统运维过程中会经常用这两个文件,因此作为Linux系统工程师,必须了解这两个文件的 ...

  3. javascript异步延时载入及推断是否已载入js/css文件

    <html> <head> <script type="text/javascript"> /**======================= ...

  4. 龙书(Dragon book) +鲸书(Whale book)+虎书(Tiger book)

    1.龙书(Dragon book)书名是Compilers: Principles,Techniques,and Tools作者是:Alfred V.Aho,Ravi Sethi,Jeffrey D. ...

  5. 多条件判断语句case

    一.case语句的基本格式: case 变量 in 模式1) 语句块1 :: 模式2) 语句块2 :: ...... :: esac 上面的格式中,每个模式后面的两个分号"::"是 ...

  6. 创建MS Office 和 WPS 兼容插件

    在工作中我们要实现一个功能,需要创建MS Office 和 WPS 兼容插件,也就是创建一个DLL,可以同时兼容office和wps.这样带来的好处就是只需要维护同一份代码,大大降低维护的工作! 1. ...

  7. qregularexpression和qregexp的区别

    QRegularExpression 是Qt 5.0才引进的,相对于QRegExp,QRegularExpression class修复了很多bug,提高了效率,提供了对Perl的RegEx几乎全面兼 ...

  8. 分布式缓存技术redis学习(四)——redis高级应用(集群搭建、集群分区原理、集群操作)

    本文是redis学习系列的第四篇,前面我们学习了redis的数据结构和一些高级特性,点击下面链接可回看 <详细讲解redis数据结构(内存模型)以及常用命令> <redis高级应用( ...

  9. WPF 之 WPF应用程序事件

    当新建一个wpf应用程序,会自动生成一个App.xaml和MainWindow.xaml文件. 其中 App.xam 用来设置Application,应用程序的起始文件和资源及应用程序的一些属性和事件 ...

  10. 为Web Api 2认证服务器增加令牌刷新功能

    Refresh tokens can potentially improve the situation but also increase complexity. A refresh token i ...