前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >C# —— 利用Marshal.GetDelegateForFunctionPointer 来转换一个函数指针为一个委托

C# —— 利用Marshal.GetDelegateForFunctionPointer 来转换一个函数指针为一个委托

作者头像
vv彭
发布2021-03-08 15:49:13
2.9K0
发布2021-03-08 15:49:13
举报
文章被收录于专栏:c#学习笔记

《转》

原来是为了在游戏外挂中发送键盘鼠标消息,自己写个sendmessage或者是postmessage又比较麻烦。于是google了一下,发现现在很多脚本工具都有这个功能,其中按键精灵的一个叫361度的插件已经有这个的实现,还验证过了。为什么不拿来己用呢? 首先分析一下按键精灵插件的接口,发现:

插件的功能函数没有直接暴露出来,而是通过一个GetCommand的函数返回一个函数描述结构。 接下来看看这个结构:

上面这个结构我已经是转换成C#的对应结构了,原结构可以查看按键精灵提供的插件C++接口源代码。 这个结构里面的 handlerFunction 实际上是指向函数的入口点,也就是一个函数指针,每个函数都一样是2个参数:

typedef int (*QMPLUGIN_HANDLER)(char *lpszParamList, char *lpszRetVal);

转换为C#中相应的委托为:

delegate void Invoker(string parameters, StringBuilder returnValue);

大家注意到,有两个参数,c++原型中都是char*类型,转换为C#的delegate后第一个为string,第二个为StringBuilder。这是因为parameters是in的,dll中不会对这个参数做修改,而returnValue是out的,dll返回时候要把返回值写入这个 StringBuilder的缓冲区。 原本的想法是用C++写一个桥来调用dll,不过在.net 2.0 中,框架直接提供了 Marshal.GetDelegateForFunctionPointer 来转换一个函数指针为一个委托,这就方便多拉。请看下面代码,注意看 BGKM_ExecuteCommand 这个函数里面的东西。

using System; using System.Collections.Generic; using System.Text; using System.Runtime.InteropServices; namespace WJsHome.Game.Utility { public class QMacro { [DllImport("BGKM5.dll", EntryPoint = "GetCommand")] static extern IntPtr BGKM_GetCommand(int commandNum); [StructLayout(LayoutKind.Sequential)] class QMPLUGIN_CMD_INFO { public string commandName; public string commandDescription; public IntPtr handlerFunction; public uint paramNumber; } delegate void Invoker(string parameters, StringBuilder returnValue); static string BuildParameters(params object[] parameters) { StringBuilder sb = new StringBuilder(); for (int i = 0; i < parameters.Length; i++) { sb.Append(parameters[i].ToString()); if (i != parameters.Length - 1) { sb.Append(','); } } return sb.ToString(); } static void BGKM_ExecuteCommand(int cmdNum, string parameters, StringBuilder retVal) { IntPtr pCmdInfo = BGKM_GetCommand(cmdNum); QMPLUGIN_CMD_INFO cmdInfo = new QMPLUGIN_CMD_INFO(); Marshal.PtrToStructure(pCmdInfo, cmdInfo); Invoker invoker = Marshal.GetDelegateForFunctionPointer(cmdInfo.handlerFunction, typeof(Invoker)) as Invoker; invoker(parameters, retVal); } public static void BGKM_KeyClick(IntPtr hWnd, int key) { BGKM_ExecuteCommand(0, BuildParameters(hWnd, key), null); } public static void BGKM_KeyDown(IntPtr hWnd, int key) { BGKM_ExecuteCommand(1, BuildParameters(hWnd, key), null); }

public static void BGKM_Mouse(IntPtr hWnd, int code, int x, int y) { BGKM_ExecuteCommand(15, BuildParameters(hWnd, code, x, y), null); } public static WinApi.POINT BGKM_ScrToCli(IntPtr hWnd, int x, int y) { StringBuilder retVal = new StringBuilder(); BGKM_ExecuteCommand(16, BuildParameters(hWnd, x, y), retVal); string[] tmp = retVal.ToString().Split('|'); return new WinApi.POINT(int.Parse(tmp[0]), int.Parse(tmp[1])); } } }


好了,方便哇?这样一来,我们可以在.net上面实现动态加载和卸载Win32 dll. 具体思路就是:(还是代码来得方便)

public delegate int MsgBox(int hwnd,string msg,string cpp,int ok); [DllImport("Kernel32")] public static extern int GetProcAddress(int handle, String funcname); [DllImport("Kernel32")] public static extern int LoadLibrary(String funcname); [DllImport("Kernel32")] public static extern int FreeLibrary(int handle); private static Delegate GetAddress(int dllModule, string functionname, Type t) {  int addr = GetProcAddress(dllModule, functionname);  if (addr == 0)   return null;  else   return Marshal.GetDelegateForFunctionPointer(new IntPtr(addr), t); } private void button1_Click(object sender, EventArgs e) {  int huser32 = 0;  huser32 = LoadLibrary("user32.dll");  MsgBox mymsg = (MsgBox)GetAddress(huser32, "MessageBoxA", typeof(MsgBox));  mymsg(this.Handle.ToInt32(), txtmsg.Text, txttitle.Text , 64);  FreeLibrary(huser32); }

本文参与 腾讯云自媒体同步曝光计划,分享自作者个人站点/博客。
原始发表:2021-02-25 ,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 作者个人站点/博客 前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体同步曝光计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
相关产品与服务
命令行工具
腾讯云命令行工具 TCCLI 是管理腾讯云资源的统一工具。使用腾讯云命令行工具,您可以快速调用腾讯云 API 来管理您的腾讯云资源。此外,您还可以基于腾讯云的命令行工具来做自动化和脚本处理,以更多样的方式进行组合和重用。
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档