C# 反射以及实际场景使用
1 什么是反射
首先要复习一下C#的编译过程,可以解释为下图

其中dll/exe中,包括元数据(metadata)和IL(中间语言Intermediate Language)
另外还出现的其他名词:CLR(公共语言运行时,Common Language Runtime)和JIT(实时编译器 Just in Time)
总结: 一个运行的程序查看本身的元数据或其他程序元数据的行为称之为反射
再配合 C# 命名空间和程序集 小记 中的图,来一层层的获取里面的数据

2 读取和使用Assembly
可以使用以下方式获取到Assembly
// See https://aka.ms/new-console-template for more information
using System.Reflection;
var assembly = Assembly.Load("Reflection");
//Assembly.LoadFile();
//Assembly.LoadFrom();
Console.Read();
2.1 Assmbly实际使用
个人使用Assmbly有以下场景
2.1.1 当程序集存在资源文件时
项目因为存在贴牌,是通过XML文件配置的,因为平时对配置文件管控不足,所以需要和XML嵌入到项目内,一来防止丢失,二来是为了以后通过更新变更配置
var assembly = Assembly.Load("DesktopFramework.Start");
var file = $"DesktopFramework.Start.TemplateFiles.Config.{filename}.xml";
using (var stream = assembly.GetManifestResourceStream(file))
{
if (stream != null)
{
var doc = new XmlDocument();
doc.Load(stream);
//处理文件
}
}
2.1.2 动态加载dll文件
相信很多人都用过Cefsharp,其中对x64和x86的加载由以下代码实现
AppDomain.CurrentDomain.AssemblyResolve += Resolver;
private static Assembly Resolver(object sender, ResolveEventArgs args)
{
if (args.Name.StartsWith("CefSharp"))
{
var assemblyName = args.Name.Split(new[] { ',' }, 2)[0] + ".dll";
var archSpecificPath = Path.Combine(AppDomain.CurrentDomain.SetupInformation.ApplicationBase,
Environment.Is64BitOperatingSystem ? "x64" : "x86",
assemblyName);
return File.Exists(archSpecificPath)
? Assembly.LoadFile(archSpecificPath)
: null;
}
return null;
}
2.1.3 获取版本号
烂大街的写法:Assembly.GetExecutingAssembly().GetName().Version.ToString();
3 Type
Type是抽象类,使用这个对象能让我们获取程序的类型信息
- 对于程序的每一个类型,CLR都会创建一个包含这个类型信息的Type类型的对象
- 每一个类型都会关联到独立的Type类的对象
- 不管创建的类型有多少个实例,都只有一个Type对象会关联到所有这些实例
3.1 获取Type
获取Type有两种办法 GetTypes()和typeof
假设存在类Test1 Test2
namespace Reflection
{
public class Test1
{
}
public class Test2
{
}
}
using Reflection;
using System.Reflection;
namespace AssemblyReflection
{
internal static class GetReflection
{
public static void GetTypes()
{
var assembly = Assembly.Load("Reflection");
foreach (var value in assembly.GetTypes())
{
Console.WriteLine(value.Name);
}
/*
* 输出
*Test1
*Test2
*/
}
public static void GetTypeName()
{
Type type = typeof(Test1);
Console.WriteLine(type.Name);
}
}
}
4 构造函数和方法
4.1 构造函数和创建实例
在说构造函数之前,先看看怎么通过type构造实例。首先是通过 Activator.CreateInstance(type),然后通过 type.GetConstructor寻找构造方法,再通过Invoke调用。
public static void CreateInstance()
{
//对默认构造函数
Type type = typeof(Test3);
var instance1 = Activator.CreateInstance(type) as Test3;
Console.WriteLine(instance1?.PublicValue);
}
public static void GetConstructInfo()
{
Type type = typeof(Test3);
ConstructorInfo? publicDefaultConstructor = type.GetConstructor(Type.EmptyTypes);
var instance = publicDefaultConstructor?.Invoke(null);
}
4.2 方法调用和重载方法
有了上面获取构造方法的例子,那么去获取方法也是比较容易。存在一种场景,比如和调用方约定方法名和函数签名,那么调用方只需要使用方法名和参数就可以调用。
首先给出一个类,仅仅函数签名不一样却存在多个方法。
public class Test3
{
private string PrivateValue { get; set; }
public string PublicValue { get; set; } = "1";
public bool PublicBoolValue { get; set; }
public Test3()
{
}
static Test3()
{
}
public Test3(string value)
{
PublicValue = value;
}
public Test3(string value,bool boolValue)
{
PublicValue = value;
PublicBoolValue = boolValue;
}
private Test3(string privateValue,string publicValue)
{
PrivateValue = privateValue;
PublicValue = publicValue;
}
public void OutPut()
{
Console.WriteLine(PublicValue);
}
public void OutPut(int param)
{
PublicValue += param;
Console.WriteLine(PublicValue);
}
public void OutPut(bool param)
{
PublicBoolValue = param;
Console.WriteLine(PublicValue);
}
}
按照调用构造方法的模式,很快就可以得到以下代码,很可惜,这段代码无法使用,原因是上述给的方法存在重载,无法通过普通的方式反射得到对应的方法。
public static void Invoke()
{
Type type = typeof(Test3);
var instance1 = Activator.CreateInstance(type) as Test3;
var method = type.GetMethod("OutPut");
method?.Invoke(instance1, null);
}
我们对寻找方法的代码稍作修改:var method = type.GetMethod("OutPut",new []{typeof(bool)});,这样就可以得到一个指定的方法。
还可以通过 GetParameters获取方法参数。
5 GetFields
有些时候,某些字段是通过 const static保存下来的,需要通过反射得到它们。假设存在一个类
public class GetFields
{
public static string Key1 = "1";
public const string Key2 = "2";
}
5.1 BindingFlags
反射当中很多都用到了BindingFlags。可以通过命名知道我们要获取Public的字段,那么
public static void GetClassFields()
{
FieldInfo[] fields = typeof(Test4).GetFields(BindingFlags.Public |
BindingFlags.Instance |
BindingFlags.Static);
foreach (FieldInfo item in fields)
{
string name = item.Name; //名称
object? value = item.GetValue(typeof(Test4)); //值
}
}
参考链接和文件代码
C#反射中的GetConstructor与GetConstructors构造函数参数的获取
https://github.com/yinghualuowu/blogsCodeSimple/tree/main/AssemblyReflection
C# 反射以及实际场景使用的更多相关文章
- java反射的运用场景
1.反射的好处是:可以在运行时确认对象以及方法. 2.下面举个简单的例子来说下反射的运用场景: 假如一款游戏有一个配置文件,配置文件里有个renderType设置了游戏启动时调用哪个RenderHan ...
- 简单模拟Java中反射的应用场景
有人说Java是一门静态语言.那么何为静态语言,动态语言又是什么? 1.动态语言 是一类在运行时可以改变其结构的语言:例如新的函数.对象.甚至代码可以 被引进,已有的函数可以被删除或是其他结构上的变化 ...
- Java反射机制使用场景
import java.io.*; import java.util.Properties; /*问题描述:存在一个主板--已经定义好,不想修改其代码,还想在主板上面增加一些其他功能? *问题解决方法 ...
- C# 反射、使用场景
创建一个 Console 控制台应用程序, 1. 创建一个 Project 类 public class Project { public int ID { get; set; } public st ...
- 03.反射--01【反射机制】【反射的应用场景】【Tomcat服务器】
https://blog.csdn.net/benjaminzhang666/article/details/9408611 https://blog.csdn.net/benjaminzhang66 ...
- Java反射机制的适用场景及其利与弊 ***
一.反射的适用场景是什么? 1).Java的反射机制在做基础框架的时候非常有用,有一句话这么说来着:反射机制是很多Java框架的基石.而一般应用层面很少用,不过这种东西,现在很多开源框架基本都已经给你 ...
- Java反射详解:入门+使用+原理+应用场景
反射非常强大和有用,现在市面上绝大部分框架(spring.mybatis.rocketmq等等)中都有反射的影子,反射机制在框架设计中占有举足轻重的作用. 所以,在你Java进阶的道路上,你需要掌握好 ...
- 长篇图解java反射机制及其应用场景
一.什么是java反射? 在java的面向对象编程过程中,通常我们需要先知道一个Class类,然后new 类名()方式来获取该类的对象.也就是说我们需要在写代码的时候(编译期或者编译期之前)就知道我们 ...
- 在 .NET 4.5 中反射机制的变更
反射机制(Reflection)通常会涉及到3中场景: 运行时反射 场景:可以检索已加载程序集.类型.对象.实例和方法调用的元数据(Metadata). .NET 支持情况:支持 仅供静态分析的反射 ...
- Unity3D ShaderLab 简单的立方体图反射
Unity3D ShaderLab 简单的立方体图反射 反射是着色器模拟现实环境的一个关键因素,它能使我们的着色器渲染效果更加具备视觉冲击,因为他利用了我们周围的环境, 让着色器反射外界的场景信息并将 ...
随机推荐
- Postman使用记录,通过表格动态赋值循环调用接口 - Using CSV and JSON data files in the Postman Collection Runner
1.GET请求,通过导入csv文件来处理 GET http://localhost:8080/web/addstudent?sno={{sno}}&name={{name}}&sex= ...
- 随机二次元图片API第三弹
Tips:当你看到这个提示的时候,说明当前的文章是由原emlog博客系统搬迁至此的,文章发布时间已过于久远,编排和内容不一定完整,还请谅解` 随机二次元图片API第三弹 日期:2020-3-10 阿珏 ...
- hbck2的一些用法
一.执行 hbase org.apache.hbase.HBCK2 可以看到下面一些选择项 **示例: -d 打印debug日志 -s 跳过客户端与服务端一致性的版本检测 hbase org.apac ...
- 11-DNS域名解析服务
背景 我们都知道,用ip可以唯一标识互联网上的主机. 从前,互联网的主机非常的少.我们都可以记住每台Server的ip. 就像是大哥大时期,电话非常少,电话号码也就非常少,我们都能记住某个人的电话. ...
- Ubuntu 18.04安装xdrp以使用远程桌面
背景 开发环境有一台服务器默认没有屏幕(被我拿走用来拓展屏幕了),有时候需要使用到界面但嫌弃拆显示器太麻烦,因此使用远程桌面来解决这个需求. 做法 安装xrdp sudo apt install -y ...
- Linux 更新网络时间
下载包 yum install -y ntpdate 同步网络时间 ntpdate 0.asia.pool.ntp.org 若上面的时间服务器不可用,也可以改用如下服务器进行同步: time.nist ...
- redis基本数据结构-列表
redis基本数据结构-列表list 特性 每个列表键最多存储 2^32 - 1个字符串元素 元素在列表中有序 元素在列表中不唯一 向列表左侧添加元素 lpush key value lpush nu ...
- 国内版Unity 2023编辑器无法切换到DX12模式的解决方法
2024.7.4更新:至6000.0.5f1c1,此问题已修复 在{安装目录}\Editor里(也就是Unity安装的根目录)建立一个名叫D3D12的文件夹. 去{安装目录}\Editor\Data\ ...
- 原创软件 | 第3期:PDF合并分割助手V1.0(个人免费)
这是一个短的"发布会". 01 基本介绍 近期开发了一个[PDF合并分割助手]. 它是一个实现pdf快速合并.分割的免费软件. 你拥有以下7种选项设置. >>合并选项& ...
- UE5 射线检测排除隐藏的Actor
0x00 Unreal Engine 5(UE5)以其卓越的性能和直观的开发工具在游戏开发领域占据了重要地位.本系列将深入探讨UE5中射线检测的关键概念,着重介绍处理隐藏Actor的技巧. 0x01. ...