在 C# 中,可以使用不同的方法调用内部或私有方法。下面分别介绍通过反射、MethodInfo.CreateDelegate、表达式(树)、动态方法(call)、动态方法(calli)这五种方法。

1. 通过反射方法

使用反射可以访问和调用内部或私有方法。

using System;
using System.Reflection; public class MyClass
{
private void MyPrivateMethod()
{
Console.WriteLine("调用了私有方法");
}
} class Program
{
static void Main()
{
MyClass myObject = new MyClass(); // 通过反射获取私有方法
MethodInfo methodInfo = typeof(MyClass).GetMethod("MyPrivateMethod", BindingFlags.NonPublic | BindingFlags.Instance); // 调用私有方法
methodInfo.Invoke(myObject, null);
}
}

2. 使用 MethodInfo.CreateDelegate 方法

通过 MethodInfo.CreateDelegate 方法可以创建委托,然后调用私有方法。

using System;
using System.Reflection; public class MyClass
{
private void MyPrivateMethod()
{
Console.WriteLine("调用了私有方法");
}
} class Program
{
static void Main()
{
MyClass myObject = new MyClass(); // 通过反射获取私有方法
MethodInfo methodInfo = typeof(MyClass).GetMethod("MyPrivateMethod", BindingFlags.NonPublic | BindingFlags.Instance); // 创建委托
Action action = (Action)Delegate.CreateDelegate(typeof(Action), myObject, methodInfo); // 调用私有方法
action();
}
}

3. 使用表达式(树)方法

通过表达式(树)可以创建动态方法,然后调用私有方法。

using System;
using System.Linq.Expressions; public class MyClass
{
private void MyPrivateMethod()
{
Console.WriteLine("调用了私有方法");
}
} class Program
{
static void Main()
{
MyClass myObject = new MyClass(); // 使用表达式创建动态方法
Action action = CreateDelegate<Action>(myObject, "MyPrivateMethod"); // 调用私有方法
action();
} // 使用表达式创建动态方法的通用方法
static TDelegate CreateDelegate<TDelegate>(object target, string methodName)
{
var methodInfo = target.GetType().GetMethod(methodName, BindingFlags.NonPublic | BindingFlags.Instance);
var parameter = Expression.Parameter(typeof(object), "instance");
var call = Expression.Call(Expression.Convert(parameter, target.GetType()), methodInfo);
var lambda = Expression.Lambda<TDelegate>(call, parameter);
return lambda.Compile();
}
}

4. 使用动态方法(call)方法

使用动态方法可以调用私有方法。

using System;
using System.Reflection;
using System.Reflection.Emit; public class MyClass
{
private void MyPrivateMethod()
{
Console.WriteLine("调用了私有方法");
}
} class Program
{
static void Main()
{
MyClass myObject = new MyClass(); // 使用动态方法调用私有方法
CallPrivateMethod(myObject, "MyPrivateMethod");
} // 使用动态方法调用私有方法的通用方法
static void CallPrivateMethod(object target, string methodName)
{
var methodInfo = target.GetType().GetMethod(methodName, BindingFlags.NonPublic | BindingFlags.Instance); // 使用动态方法
var dynamicMethod = new DynamicMethod("CallMethod", null, new[] { typeof(object) }, target.GetType());
var ilGenerator = dynamicMethod.GetILGenerator();
ilGenerator.Emit(OpCodes.Ldarg_0); // 加载第一个参数,即目标实例
ilGenerator.EmitCall(OpCodes.Call, methodInfo, null); // 调用私有方法
ilGenerator.Emit(OpCodes.Ret); // 返回
var action = (Action<object>)dynamicMethod.CreateDelegate(typeof(Action<object>)); // 调用私有方法
action(target);
}
}

5. 使用动态方法(calli)方法

使用动态方法(calli)可以调用私有方法。

using System;
using System.Reflection.Emit; public class MyClass
{
private void MyPrivateMethod()
{
Console.WriteLine("调用了私有方法");
}
} class Program
{
static void Main()
{
MyClass myObject = new MyClass(); // 使用动态方法(calli)调用私有方法
CallPrivateMethod(myObject, "MyPrivateMethod");
} // 使用动态方法(calli)调用私有方法的通用方法
static void CallPrivateMethod(object target, string methodName)
{
var methodInfo = target.GetType().GetMethod(methodName, BindingFlags.NonPublic | BindingFlags.Instance); // 使用动态方法(calli)
var dynamicMethod = new DynamicMethod("CallMethod", typeof(void), new[] { typeof(object) }, target.GetType());
var ilGenerator = dynamicMethod.GetILGenerator();
ilGenerator.Emit(OpCodes.Ldarg_0); // 加载第一个参数,即目标实例
ilGenerator.EmitCalli(OpCodes.Call, methodInfo.CallingConvention, methodInfo.ReturnType, methodInfo.GetParameters().Select(p => p.ParameterType).ToArray(), null); // 调用私有方法
ilGenerator.Emit(OpCodes.Ret); // 返回
var action = (Action<object>)dynamicMethod.CreateDelegate(typeof(Action<object>)); // 调用私有方法
action(target);
}
}

以上五种方法都可以用于调用内部或私有方法,具体使用哪种方法取决于具体的场景和需求。

探秘C#中的秘密通道:五种引人注目的方法调用内部或私有方法的更多相关文章

  1. js中声明Number的五种方式

    转载自:http://www.jb51.net/article/34191.htm <!DOCTYPE html> <html> <head> <meta c ...

  2. 五种I/O 模式,select、epoll方法的理解,BIO、NIO、AIO理解 相关文章

    一.io方式 Linux网络编程 五种I/O 模式及select.epoll方法的理解 web优化必须了解的原理之I/o的五种模型和web的三种工作模式 五种I/O 模式——阻塞(默认IO模式),非阻 ...

  3. 面试中常问的五种IO模型和BIO,NIO,AIO

    一,五种IO模型: 一个IO操作可以分为两个步骤:发起IO请求和实际的IO操作例如:1.操作系统的一次写操作分为两步:第一步,将数据从用户空间拷贝到系统空间:第二步,从系统空间往网卡写.2.一次读操作 ...

  4. iOS开发中数组常用的五种遍历方式

    随着iOS的不断发展,apple也不断推出性能更高的数组遍历方式,下面将对熟悉的五种遍历方式进行列举. 首先定义一个数组,并获取数组长度 NSArray *array=@[",]; NSIn ...

  5. JavaScript中常见的十五种设计模式

    在程序设计中有很多实用的设计模式,而其中大部分语言的实现都是基于“类”. 在JavaScript中并没有类这种概念,JS中的函数属于一等对象,在JS中定义一个对象非常简单(var obj = {}), ...

  6. webpack中,require的五种用法

    a.js: module.exports = function(x){ console.log(x); } 一,commonjs同步: var b = require('./a');b('你好')// ...

  7. Redis学习二 C#中如何进行这五种数据类型的操作

    我在网上找了好久,就是没有找到Redis和C#结合的书,都是和别的编程语言在一起鬼混. 简单的用C#实现向Redis中插入那我中类型的数据 首先需要到NuGet 里面下载 Redis IDatabas ...

  8. Redis 中可以存储的五种基本类型

    具体介绍 数字还是字符? String(字符串) 二进制安全 可以包含任何数据,比如jpg图片或者序列化的对象,一个键最大能存储512M --- Hash(字典) 键值对集合,即编程语言中的Map类型 ...

  9. Java程序栈信息文件中的秘密(五)

    最近发现在使用jstack工具在导出Java应用的线程栈时有一个小小的窍门,比如Linux环境上有一个用户为appuser,假如以这个用户启动了一个Java进程B,如果想要导出进程B的线程栈,则必须切 ...

  10. NSNotification,NSNotificationCenter的使用、iOS中五种对象间传值的方式

    学习内容 NSNitification与NotificationCenter(通知与通知中心) 通知的使用 [[NSNotificationCenter defaultCenter]addObserv ...

随机推荐

  1. Prism报错

    Rules.Default..WithoutFastExpressionCompiler()报错 说没有找到容器 1.查看Prism.Wpf源码 获取DryIoc容器规则 2.证明项目中出现了另外一个 ...

  2. Android RIL&IMS源码分析

    一.需求 1.了解IMS相关知识体系 2.RILD 与 RILJ.IMS回调消息的机制 二.相关概念 2.1 IMS IMS全称是IP Multimedia Subsystem,中文意义为IP多媒体子 ...

  3. react18-webchat网页聊天实例|React Hooks+Arco Design仿微信桌面端

    React18 Hooks+Arco-Design+Zustand仿微信客户端聊天ReactWebchat. react18-webchat基于react18+vite4.x+arco-design+ ...

  4. Seata AT和XA模式

    一.分布式事务产生得原因: 1.1.数据库分库分表 当数据库单表一年产生的数据超过1000W,那么就要考虑分库分表,具体分库分表的原理在此不做解释,以后有空详细说,简单的说就是原来的一个数据库变成了多 ...

  5. 「hackerrank - 101hack43」K-Inversion Permutations

    link. 原问题即:请你给出不同的序列 \(\{a_n\}\) 的数量,满足 \(0\leqslant a_i<i\),且 \(\sum a_i=k\). 那么写出 \({a_n}\) 的 o ...

  6. R3300L, Q7 ATV Android9固件

    R3300L, Q7 ATV Android9固件 固件来源 https://www.znds.com/tv-1239603-1-1.html 之前在恩山上发布过1080p安卓6固件 https:// ...

  7. Ubuntu更新软件的命令

    更新软件源 apt-get update 更新升级所有软件 apt-get upgrade 更新某个软件 apt-get upgrade 名 列出可更新的软件 apt list --upgradabl ...

  8. SpringBoot2.7升级到3.0的实践分享

    背景 最近把项目中的技术框架做一次升级,最重要的就是SpringBoot从2.7.x升级到3.0.x,当然还会有一些周边的框架也会连带着升级,比如Mybatis Plus,SpringCloud等,话 ...

  9. 11. 用Rust手把手编写一个wmproxy(代理,内网穿透等), 实现健康检查

    11. 用Rust手把手编写一个wmproxy(代理,内网穿透等), 实现健康检查 项目 ++wmproxy++ gite: https://gitee.com/tickbh/wmproxy gith ...

  10. 维修道路(repair)

    维修道路(repair) 时间限制: 1 Sec  内存限制: 128 MB 题目描述 由于在十多年前道路改建时的突出贡献, Bob 被任命为维修道路的承包商, 他可以任意 选择两条路径去修理. Bo ...