一、反射概述
(能够通过非反射方式实现的功能,尽量不要用反射)
反射是.NET中的重要机制,通过反射可以得到*.exe或*.dll等程序集内部的接口、类、方法、字段、属性、特性等信息,还可以动态创建出类型实例并执行其中的方法。
反射的定义:审查元数据并收集关于它的类型信息的能力。
元数据(编译以后的最基本数据单元)就是一大堆的表,当编译程序集或者模块时,编译器会创建一个类定义表,一个字段定义表,和一个方法定义表等。
1)什么是反射
Reflection,中文翻译为反射。
这是.Net中获取运行时类型信息的方式,.Net的应用程序由几个部分:‘程序集(Assembly)’、‘模块(Module)’、‘类型(class)’组成,而 反射提供一种编程的方式,让程序员可以在程序运行期获得这几个组成部分的相关信息(反射指程序可以访问、检测和修改它本身状态或行为的一种能力),例如:
Assembly类可以获得正在运行的装配件信息,也可以动态的加载装配件,以及在装配件中查找类型信息,并创建该类型的实例。
Type类可以获得对象的类型信息,此信息包含对象的所有要素:方法、构造器、属性等等,通过Type类可以得到这些要素的信息,并且调用之。
MethodInfo包含方法的信息,通过这个类可以得到方法的名称、参数、返回值等,并且可以调用之。
诸如此类,还有FieldInfo、EventInfo等等,这些类都包含在System.Reflection命名空间下。
2)命名空间与装配件
1.介绍
命名空间 :类库的逻辑组织形式 (命名空间类似于Java的包,但又不完全等同,因为Java的包必须按照目录结构来放置,命名空间则不需要)
程序集(装配件):类库的物理组织形式 (程序集是.Net应用程序执行的最小单位,.Net应用程序包含一个或多个程序集,编译出来的.dll、.exe都是程序集,.NET程序集包含元数据,这些元数据描述了程序集中定义的所有类型及其成员的信息,即方法、属性、事件和字段。【CLI程序集可分为两类:进程程序集(EXE)、库程序集(DLL),其中.exe文件是一个自己执行的程序集,而.dll将被其他程序集加载后运行。】)
2.联系
命名空间与程序集(在一个程序集中可以有不同的名称空间,同一个名称空间也可以分布在多个程序集上)
举个例子:
程序集A
namespace N1
{
public class AC1 {…}
public class AC2 {…}
}
namespace N2
{
public class AC3 {…}
public class AC4{…}
}
程序集B
namespace N1
{
public class BC1 {…}
public class BC2 {…}
}
namespace N2
{
public class BC3 {…}
public class BC4{…}
}
理解:
这两个装配件中都有N1和N2两个命名空间,而且各声明了两个类,这样是完全可以的,然后我们在一个应用程序中引用装配件A,那么在这个应用程序中,我们能看到N1下面的类为AC1和AC2,N2下面的类为AC3和AC4。
接着我们去掉对A的引用,加上对B的引用,那么我们在这个应用程序下能看到的N1下面的类变成了BC1和BC2,N2下面也一样。
如果我们同时引用这两个装配件,那么N1下面我们就能看到四个类:AC1、AC2、BC1和BC2。
到这里,我们可以清楚一个概念了, 命名空间只是说明一个类型是那个族的,比如有人是汉族、有人是回族;而装配件表明一个类型住在哪里,比如有人住在北京、有人住在上海;那么北京有汉族人,也有回族人,上海有汉族人,也有回族人,这是不矛盾的。
上面我们说了,装配件是一个类型居住的地方,那么在一个程序中要使用一个类,就必须告诉编译器这个类住在哪儿,编译器才能找到它,也就是说必须引用该装配件。
那么如果在编写程序的时候,也许不确定这个类在哪里,仅仅只是知道它的名称,就不能使用了吗?答案是可以,这就是反射了, 就是在程序运行的时候提供该类型的地址,而去找到它。
3.应用
只有同时指定类型所在的命名空间及实现该类型的程序集,才能完全限定该类型。
(摘抄自《精通.NET核心技术--原来与架构》 电子工业出版社)
也就是说,你要创建一个类的实例,必须知道该类的 命名空间(这个一般都知道)+程序集(这个很容易被我们忽略,因为一般我们不需要引用外部的程序集,如果你用C#做过Excel文件的导入导出,就会知道必须添加一个Excel相关的程序集引用)
3)System.reflection命名空间
官方文档:链接地址
System.reflection包含的几个类,允许你反射(解析)这些元数据表的代码。
System.Reflection.Assembly System.Reflection.MemberInfo System.Reflection.EventInfo System.Reflection.FieldInfo System.Reflection.MethodBase System.Reflection.ConstructorInfo System.Reflection.MethodInfo System.Reflection.PropertyInfo | Assembly定义和加载程序集,加载在程序集清单中列出模块,以及从此程序集中查找类型并创建该类型的实例。 |
MemberInfo获取有关成员属性的信息并提供对成员元数据的访问权限。 | |
EventInfo了解事件的名称、事件处理程序数据类型、自定义属性、声明类型和反射类型等,添加或移除事件处理程序。 | |
FieldInfo了解字段的名称、访问修饰符(如public或private)和实现详细信息(如static)等,并获取或设置字段值。 | |
MethodBase提供有关方法和构造函数的信息。 | |
ConstructorInfo了解构造函数的名称、参数、访问修饰符(如pulic 或private)和实现详细信息(如abstract或virtual)等。使用Type的GetConstructors或 GetConstructor方法来调用特定的构造函数。 | |
MethodInfo了解方法的名称、返回类型、参数、访问修饰符(如pulic 或private)和实现详细信息(如abstract或virtual)等。使用Type的GetMethods或GetMethod方法来调用特定的方法。 | |
PropertyInfo发现属性 (Property) 的属性 (Attribute) 并提供对属性 (Property) 元数据的访问。了解属性的名称、数据类型、声明类型、反射类型和只读或可写状态等,获取或设置属性值。 | |
Type访问元数据的主要方式,类型。 |
基础知识 另一文章:C#反射相关基础
4)相关知识
1.反射的作用
1、可以使用反射动态地创建类型的实例,将类型绑定到现有对象,或从现有对象中获取类型
2、应用程序需要在运行时从某个特定的程序集中载入一个特定的类型,以便实现某个任务时可以用到反射。
3、反射主要应用于类库,这些类库需要知道一个类型的定义,以便提供更多的功能。
2.应用要点
1、现实应用程序中很少有应用程序需要使用反射类型
2、使用反射动态绑定需要牺牲性能
3、有些元数据信息是不能通过反射获取的
4、某些反射类型是专门为那些clr 开发编译器的开发使用的,所以你要意识到不是所有的反射类型都是适合每个人的。
二、反射应用
1)创建对象
1.根据类型来动态创建对象
System.Activator提供了方法来根据类型动态创建对象,比如创建一个DataTable:
// 1.获取类型
Type t = Type.GetType("System.Data.DataTable,System.Data,Version=1.0.3300.0, Culture=neutral, PublicKeyToken=b77a5c561934e089");
// 2.创建对象
DataTable table = (DataTable)Activator.CreateInstance(t);
2.根据有参数的构造器创建对象
using System.Reflection;
using Test;
namespace ConsoleApp1
{
public class Program
{
static void Main(String[] args)
{
//1.创建Test程序集下TestC类的Type对象
Type t = Type.GetType("Test.TestC");
//2.把参数按照顺序放入一个Object数组中
object[] constructParms = new object[] { "hello" };
//3.创建TestC对象 强制转换
TestC obj = (TestC)Activator.CreateInstance(t,constructParms);
//4.打印值
Console.WriteLine(obj._value);
}
}
}
//另一个命名空间
namespace Test
{
public class TestC
{
public string _value;
public TestC(string value) //带参的构造函数
{
_value = value;
}
}
}
结果:
2)获取及动态调用方法
using System.Reflection;
using Test;
namespace ConsoleApp1
{
public class Program
{
static void Main(String[] args)
{
// 获取类型
Type t = Type.GetType("Test.TestC");
// 创建对象
TestC tc = (TestC)Activator.CreateInstance(t,new object[] {"cml"});
// 获取类型t的方法AddValue
MethodInfo mi = t.GetMethod("AddValue");
// 调用方法
object retu = mi.Invoke(tc, new object[] { "20230112" });
// 打印方法返回值
Console.WriteLine(retu);
}
}
}
//另一个命名空间
namespace Test
{
public class TestC
{
//字段
public string value { get; set; }
//构造函数
public TestC(){ }
public TestC(string v){ value = v; }
//自定义函数
public string AddValue(string prefix)
{
return value + "+" + prefix; //将value与该方法传入参数拼接
}
}
}
结果: