当前位置: 首页 > ds >正文

.NET外挂系列:5. harmony 中补丁参数的有趣玩法(下)

一:背景

1. 讲故事

开局一张表,故事全靠编,为了能够承上启下,先把参数列表放出来。

参数名说明
__instance访问非静态方法的实例(类似 this)。
__result获取/修改返回值,要想修改用 ref
__resultRef修改返回引用(方法返回是 ref 返回 )。
__state在前缀和后缀间传递自定义数据 。
___fields读写私有字段(三下划线开头,修改需加 ref)。
__argsobject[] 形式访问所有参数(修改数组即修改参数)。
方法参数同名直接映射原参数。
__n__n 表示直接访问第 n 个参数,从 0 开始)。
__originalMethod获取原方法的 MethodBase
__runOriginal判断原方法是否被执行。

如果说上一篇聊到的参数是无害的,那这篇所聊到的参数就具有破坏性了,会让一些底层方法产生匪夷所思的输出结果。

二:补丁参数解读

1. __result

这个参数可以获取被注入方法的返回值,你可以对他进行查看和修改,为了让例子更有趣一点,我们对 DateTime.Now 进行注入,让它永远的丢失时分秒,是不是有点像黑客? 哈哈,参考代码如下:

internal class Program{static void Main(string[] args){var harmony = new Harmony("com.example.patch");harmony.PatchAll();var time = DateTime.Now;Console.WriteLine($"当前时间:{time}");Console.ReadLine();}}[HarmonyPatch(typeof(DateTime), "Now", MethodType.Getter)]public class DateTimeHook{public static void Postfix(ref DateTime __result){__result = __result.Date;}}

是不是让人很恼火,明明调的是 DateTime.Now ,怎么时分秒不见了。。。

2. __args

在 harmony 中有三种方式可以获取原方法的参数,分别为:

  • object[] __args 获取,支持读写。
  • __n 下标获取,支持读写。
  • parameter 同名法,默认只读,写的话要加 ref。

为了让例子更加有趣和黑客,我们对 HttpClient 的底层方法 SendAsync 进行拦截,然后纂改url,指向一个来历不明的网址,参考代码如下:

internal class Program{static async Task Main(string[] args){// 应用Harmony补丁var harmony = new Harmony("com.example.httpclient");harmony.PatchAll();var url = "https://www.cnblogs.com";var httpClient = new HttpClient();Console.WriteLine($"1.request:{url}");var response = await httpClient.GetAsync(url);var content = await response.Content.ReadAsStringAsync();Console.WriteLine($"2.response:\n{content.Substring(0, 500)}");Console.ReadKey();}}[HarmonyPatch(typeof(HttpClient), "SendAsync", new Type[] { typeof(HttpRequestMessage), typeof(HttpCompletionOption), typeof(CancellationToken) })]class HttpClientPatch{static void Prefix(object[] __args){HttpRequestMessage request = (HttpRequestMessage)__args[0];request.RequestUri = new Uri("http://www.baidu.com");}}

从卦中看,我明明请求的是 博客园,怎么给我返回 百度 的内容,是不是非常诡异。。。

可能有朋友看到了,这里有一个 (HttpRequestMessage)__args[0]; 强转的逻辑,能不能在 Prefix(object[] __args) 中直接接收 HttpRequestMessage 参数呢?可以的,这这就 harmony 的另外一种同名参数法,也就是参数名一定要和底层的 SendAsync 方法签名保持一致,截图如下:

修改后的代码如下,是不是非常的清爽。

[HarmonyPatch(typeof(HttpClient), "SendAsync", new Type[] { typeof(HttpRequestMessage), typeof(HttpCompletionOption), typeof(CancellationToken) })]class HttpClientPatch{static void Prefix(HttpRequestMessage request){request.RequestUri = new Uri("http://www.baidu.com");}}

可能有些人会遇到这样的情况,比如 SendAsync 方法的第一个参数是 internel 类型,由于是程序集可访问,所以你无法在另一个程序集的 Prefix 中声明此类型,这时候怎么办呢?可以借助 harmony 提供的 __n 索引法,下标是从0开始的。修改代码如下:

[HarmonyPatch(typeof(HttpClient), "SendAsync", new Type[] { typeof(HttpRequestMessage), typeof(HttpCompletionOption), typeof(CancellationToken) })]class HttpClientPatch{static void Prefix(object __0){Type requestType = __0.GetType();PropertyInfo requestUriProperty = requestType.GetProperty("RequestUri");Uri newUri = new Uri("http://www.baidu.com");requestUriProperty.SetValue(__0, newUri);}}

3. ___fields

这个参数也是一个非常简单粗暴的特性,它可以用三下划线___引出当前 this 实例上的私有字段,使用场景可以是这样的,我们知道 new Thread 默认是没有 ThreadName 的,这在高级调试中往往有所不便,所以可加这样的一段逻辑:一旦发现无名的 ThreadName 就给它赋一个默认的名字,参考代码如下:

internal class Program{static void Main(string[] args){var harmony = new Harmony("com.example.patch");harmony.PatchAll();var thread = new Thread(() => { });thread.Start();Console.WriteLine($"1.查看线程名:{thread.Name?.ToString()}");Console.ReadLine();}}[HarmonyPatch(typeof(Thread), "Name", MethodType.Getter)]public class ThreadStartHook{public static void Prefix(Thread __instance, ref string ____name){if (string.IsNullOrEmpty(____name)){____name = $"Default Threadid:{__instance.ManagedThreadId}";}}}

三:总结

这篇文章我们聊到的一些参数多多少少都带点黑客性质,建议大家不要乱用,这里声明一下,我所说的一切都是为.NET高级调试训练营服务的,也是给学员们提供的拓展资料。

http://www.xdnf.cn/news/7850.html

相关文章:

  • 野火鲁班猫(arrch64架构debian)从零实现用MobileFaceNet算法进行实时人脸识别(四)安装RKNN Toolkit Lite2
  • IP地址详解
  • vue调后台接口
  • 【5.19-5.26学习周报】
  • RPA浪潮来袭,职业竞争的新风口已至?
  • HOT100(二叉树)
  • 大语言模型 16 - Manus 超强智能体 Prompt分析 原理分析 包含工具列表分析
  • Python数据库编程案例
  • 2022CCPC吉林省赛长春邀请赛 Java 做题记录
  • 软考软件评测师—— 操作系统综合知识
  • RedissonClient主要功能概述
  • 黑马点评相关知识总结
  • 大模型会话窗口为什么对最新和最久记忆表现较好
  • 13 分钟讲解所有知名 Python 库/模块
  • 命名常量集合接口INamedConstantCollection<T>实现
  • 顶级流媒体服务商 Spotify 2025.04 故障复盘报告,吃他人的堑长自己的智
  • 4.8 加密模块
  • 无人机报警器360°检测技术分析!
  • 先验知识融合机器学习的几种方式
  • VentureBeat AI 最新资讯 (2025-05-19)
  • NVM安装使用及问题解决
  • Semaphore解决高并发场景下的有限资源的并发访问问题
  • 整型数相加的溢出
  • Python的蚁群优化算法实现与多维函数优化实战
  • 【Java高阶面经:微服务篇】1.微服务架构核心:服务注册与发现之AP vs CP选型全攻略
  • C语言指针深入详解(五):回调函数、qsort函数
  • 卡片布局自适应
  • c语言刷题之实际问题
  • 一文读懂|大模型智能体互操作协议:MCP/ACP/A2A/ANP
  • Redis学习专题(三)主从复制