简介:
反射是.NET中的重要机制,通过反射可以得到*.exe或*.dll等程序集内部的接口、类、方法、字段、属性、特性等信息,还可以动态创建出类型实例并执行其中的方法。
反射指程序可以访问、检测和修改它本身状态或行为的一种能力。
通俗一点:我们在获取其他实体类的字段名或实列,只能获取公有的,而有了反射之后可以获取私有的,可以获取他的基类等等,可以说把家底查得清清楚楚
案例
创建一个学生类,里面有公有姓名,私有年龄但提供get方法,以及Home住址不提供get方法
namespace Project; public class Student { //公开信息id public int id { get; set; } //公开信息Name public String Name; //私有信息年龄 private int Age; //保密信息家庭住址 private String Home; //有参无参构造方法 public Student(){} public Student(String name, int age) { Name = name; Age = age; } //获取年龄方法 public int Getage() { return this.Age; } }
利用反射获取类型
在用反射对类进行操作之前,我们要先获取到类,下面是反射获取类的两种方式👇
[Test] public void TestGetType() { //通过typeof运算符获取type类型 Type type = typeof(Student); Console.WriteLine("type="+type); //通过实例对象获取 StuDdentInfo obj = new StuDdentInfo(); Type type2 = obj.GetType(); Console.WriteLine("type2="+type2); }
控制台输出,成功获取到了类型
获取Properties
首先变量类型声明方式创建一个 Type 类型的变量 type,表示 Student 类型。接着,使用反射 API 的 GetProperties() 方法获取 Student 类型的所有公共属性(包括只读、读写)、静态属性和实例属性等信息,然后存储在Students 变量中。
最后,使用 foreach 遍历每个属性信息对象
[Test] public void TestProperty() { Type type = typeof(Student); var Students = type.GetProperties(); foreach (var student in Students) { Console.WriteLine($"名称:{student.Name},类型:{student.GetMethod}"); } }
还可以通过属性名称获取
[Test] public void TestProperty() { Type type = typeof(Student); var property = type.GetProperty("id"); Console.WriteLine($"名称:{property.Name},类型:{property.PropertyType}"); } //控制台返回:名称:id,类型:System.Int32
获取私有实例
反射最牛的地方就是他可以获取私有方法与实列
BindingFlags.Instance 表示只获取实例方法
BindingFlags.NonPublic 表示只获取非公共方法
首先使用 C# 变量类型声明方式创建一个表示 Student 类型的 Type 类型变量 type。接着,利用 type 变量使用反射 API 的 GetFields() 方法找到所有 Student 类型的非公共实例字段,存储在 fieldInfos 变量中。
[Test] public void TestField() { Type type = typeof(Student); var fieldInfos = type.GetFields(BindingFlags.Instance | BindingFlags.NonPublic); foreach (var fieldInfo in fieldInfos) { Console.WriteLine($"名称:{fieldInfo.Name},类型:{fieldInfo.FieldType}"); } }
获取方法
使用反射 API 获取 Student 类型的所有实例方法(包括公共和非公共的),然后赋值给 methods 变量
BindingFlags.Instance 表示只获取实例方法
BindingFlags.Public 表示只获取公共方法
BindingFlags.NonPublic 表示只获取非公共方法
[Test] public void TestMethod() { var type = typeof(Student); var methods = type.GetMethods( BindingFlags.Instance |BindingFlags.Public| BindingFlags.NonPublic ); foreach (var methodInfo in methods) { Console.WriteLine($"名称:{methodInfo.Name},类型:{methodInfo.ReturnType}"); } }
控制台打印中有Student类中的方法,还有GetType、Finalize、ToString等方法,因为这六个方法继承Object方法。
- ToString() :将对象转换为字符串
- Equals(Object):确定该对象是否等于另一个对象
- GetHashCode():获取对象的哈希代码
- GetType():获取对象的类型
- Finalize():在垃圾回收器回收对象之前执行清理操作
- MemberwiseClone():创建当前对象的浅表副本
给属性赋值
[Test] public void TestOpenProperty() { Type type = typeof(Student); var propertyInfo = type.GetProperty("id"); var instance = Activator.CreateInstance(type); propertyInfo.SetValue(instance,1011); var value = propertyInfo.GetValue(instance); Console.WriteLine(value); } //输出1011
字段赋值
一般来说,方法的私有字段都是在方法内部使用的,哪怕我们new一个方法,用方法也是点不出来私有字段的,比如student的Home是私有字段,下面new一个Student类,提示里并没有Home字段。
有了反射,我们可以获取到类里的私有字段,还可以对其进行赋值
[Test] public void TestOperationField() { Type type = typeof(Student); var fieldInfo = type.GetField("Home", BindingFlags.Instance | BindingFlags.NonPublic); var instance = Activator.CreateInstance(type); fieldInfo.SetValue(instance,"爱情公寓");//给Home赋值 var obj = fieldInfo.GetValue(instance);//获取Home的值 Console.WriteLine("myHome:"+obj); } //输出:myHome:爱情公寓
获取程序集
程序集是C#程序构建后生成的二进制文件,其中包含了C#代码编译后产生的中间语言(Intermediate Language,简称IL)以及元数据信息。程序集可以是可执行文件(.exe文件)或动态链接库文件(.dll文件),它们被用于将C#源代码编译成.NET平台上的可执行文件。程序集包含了类、方法、属性、事件等编译生成的元素,同时也可能包含其他资源文件,如图像、声音、文本等。
通过反射获取程序集的方法👇
[Test] public void TestAssembly() { //通过项目名称加载程序集 var assembly = Assembly.Load("Project"); //通过dll的路径加载程序集 var loadFile = Assembly.LoadFile(@"D:\project\c#\Solution\Project\bin\Debug\net6.0\Project.dll"); }
加载程序集后可以看见此程序集的dll路径,版本号,解决方案,类名等等信息
然后我们可以通过Assembly.GetExecutingAssembly()来获取程序集的信息
如Assembly.GetExecutingAssembly().GetName().Name;可以获取程序集的解决方案名称project
同时还可以通过反射获取的程序集来实例化我们的类
//解决方案.类名称 , false:忽略大小写,可以不写var instance = loadFile.CreateInstance("Project.Student",false);