<学习>.NET的反射基础

时间:2023-03-09 02:16:25
<学习>.NET的反射基础

关键词

Assembly 使用Assembly定义和加载程序集,加载在程序集清单中列出模块,以及从此程序集中查找类型并创建该类型的实例。
Module 通过它可以获取包含模块的程序集以及模块中的类等,还可以获取在模块上定义的所有全局方法或其他特定的非全局方法。 
MemberInfo 这是一个基类,它定义了EventInfo、FieldInfo、MethodInfo、PropertyInfo的多个公用行为。
Type 是System命名空间下的一个类,一般用于装载反射得到的类对象,通过Type可以得到一个类的内部信息,也可以通过它反射创建一个对象。
EventInfo 通过它可以获取事件的相关信息。如事件的名称、事件处理程序数据类型、声明类型等,还可以添加或移除事件处理程序。
FieldInfo 可以通过它获取字段的相关信息。如字段的名称、访问修饰符等,还可以获取或设置字段值。 
MethodInfo 通过它可以获取方法的相关信息。例如方法的名称、返回类型、参数、访问修饰符等。使用Type的GetMethods或GetMethod方法来调用特定的方法。
PropertyInfo 通过它可以获取属性的相关信息。例如属性的名称、数据类型、声明类型和只读或可写状态等,还可以获取或设置属性值。
ConstructorInfo 通过它可以了解构造函数的属性及调用构造函数。可以由Type对象的GetConstructors或GetConstructor方法返回的。
ParameterInfo 通过它可以了解参数相关信息。例如参数的名称、数据类型、是输入参数还是输出参数,以及参数在方法签名中的位置等。

什么是反射?

Reflection,中文翻译为反射,是.Net中获取运行时类型信息的方式。
.Net的应用程序由几个部分:程序集(Assembly)、模块(Module)、类型 (class)组成,而反射提供一种编程的方式,让程序员可以在程序运行期获得这几个组成部分的相关信息。

如何使用反射获取程序集?

通过Assembly类对象的Load方法、LoadFrom方法和LoadFile方法可以获取程序集的相关信息。

三者之间的区别:
LoadFrom和Load差不多,只是LoadFrom是多了先获取其中的程序集版本,语言文化,公钥标记等信息,把他们传递给 Load方法这一步。
LoadFrom不能用于加载标识相同但路径不同的程序集。
LoadFile 方法用来来加载和检查具有相同标识但位于不同路径中的程序集.但不会加载程序的依赖项。
1、Assembly.LoadFile只载入相应的dll文件,比如Assembly.LoadFile("abc.dll"),则载入abc.dll,假如abc.dll中引用了def.dll的话,def.dll并不会被载入。
Assembly.LoadFrom则不一样,它会载入dll文件及其引用的其他dll,比如上面的例子,def.dll也会被载入。
2、用Assembly.LoadFrom载入一个Assembly时,会先检查前面是否已经载入过相同名字的Assembly,比如abc.dll有两个版本(版本1在目录1下,版本2放在目录2下),程序一开始时载入了版本1,当使用Assembly.LoadFrom("2//abc.dll")载入版本2时,不能载入,而是返回版本1。

Assembly.LoadFile的话则不会做这样的检查,比如上面的例子换成Assembly.LoadFile的话,则能正确载入版本2。

如何使用反射获取模块?

1、用Object.GetType().Module来获取。
2、用Assembly类对象的GetModules方法来获取。

如何使用反射获取Type对象?

1、用typeof运算符获取。
2、用静态方法Type.GetType来获取。
3、用Assembly类对象的GetType或者GetTypes方法来获取。

如何根据类型来动态创建对象?

1、用System.Reflection空间下的Assembly类对象的CreateInstance方法来创建对象。

2、用System.Activator类提供的静态方法CreateInstance来创建对象。

如何获取方法以及动态调用特定的方法?

通过Type类对象的GetMethod或者GetMethods方法获取到类里面的方法,然后用MethodInfo对象的Invoke方法去执行。

如何获取字段以及动态设置获取它?

通过Type类对象的GetField方法来获取字段,用FieldInfo对象存储获取到的字段,可以通过FieldInfo对象的GetValue和SetValue方法读取和设置字段。

如何获取属性以及动态设置获取它?

通过Type类对象的GetPropertie或者GetProperties方法来获取属性,用PropertyInfo对象存储获取到的属性,然后通过PropertyInfo对象的GetValue和SetValue方法读取和设置属性值。

动手尝试

1、创建个类库工程,在这工程里面定义一个类。(我这里工程名为:Study__Reflection,类名为:StudyReflection)

 namespace Study__Reflection
{
public class StudyReflection
{
/// <summary>
/// 学习
/// </summary>
/// <param name="name">姓名</param>
/// <returns>字符串</returns>
public string Study(string name)
{
return name + "正在学习反射";
}
/// <summary>
/// 无参构造函数
/// </summary>
public StudyReflection() { } /// <summary>
/// 有参构造函数
/// </summary>
/// <param name="a"></param>
/// <param name="b"></param>
public StudyReflection(int a, int b)
{
SetValue(a, b);
} #region 私有成员变量
private int x;
private int y;
public string Status;
#endregion #region 属性
public int X
{
get { return x; }
set { x = value; }
} public int Y
{
get { return y; }
set { y = value; }
} #endregion
/// <summary>
/// 计算x和y的和
/// </summary>
/// <returns>x+y</returns>
public int sum()
{
return x + y;
} /// <summary>
/// 取2个数之间的商的整数
/// </summary>
/// <param name="z"></param>
/// <param name="t"></param>
/// <returns></returns>
public int div(int z, int t)
{
return z/t;
} /// <summary>
/// 计算x和y的积
/// </summary>
/// <returns>x*y</returns>
private int product()
{
return x * y;
}
/// <summary>
/// 给x和y赋值
/// </summary>
/// <param name="a">整形a</param>
/// <param name="b">整形b</param>
public void SetValue(int a,int b)
{
x = a;
y = b;
}
}
}

把这工程指定好路径编译好。生成的dll文件就是准备用于学习反射相关知识的。

下面是反射的基本用法:

 //获取程序集
Assembly asem = Assembly.LoadFrom("../lib/Study_ Reflection.dll");
//Assembly asem2 = Assembly.Load("Study_ Reflection, Version=1.0.0.0, Culture=neutral, PublicKeyToken=null");
//Assembly asem3 = Assembly.LoadFile(@"D:\mythings\Study\lib\Study_ Reflection.dll"); //注意是全路径 //用Object.GetType().Module来获取。
Class1 cc = new Class1();
Module m = cc.GetType().Module;
//用Assembly类对象的GetModules方法来获取。
if (asem == null)
{
asem = Assembly.LoadFrom("../lib/Study_ Reflection.dll");
}
Module[] ms = asem.GetModules(); //通过Assembly.GetType获取Type
Type t = asem.GetType("Study__Reflection.StudyReflection"); //通过Assembly.CreateInstance进行无参数实例化
object awc = asem.CreateInstance("Study__Reflection.StudyReflection", true);
//通过Activator.CreateInstance进行无参数实例化
object aa = Activator.CreateInstance(asem.FullName, "Study__Reflection.StudyReflection").Unwrap();
//通过Activator.CreateInstance方法传入Type对象执行默认的无参数构造函数来创建实例
object o = Activator.CreateInstance(t);
//通过Assembly.CreateInstance进行有参数实例化
object[] te = new object[] { , };
object ayc = asem.CreateInstance("Study__Reflection.StudyReflection", true, BindingFlags.Default, null, te, null, null); //属性
PropertyInfo[] ps = t.GetProperties(); //通过Type.GetProperties获取属性
PropertyInfo p = t.GetProperty("X"); //通过Type.GetPropertie来获取指定名称的属性
p.SetValue(o, ); //给X属性赋值
p.GetValue(o); //读取X属性
//获取字段
FieldInfo[] fs = t.GetFields(); //通过Type.GetFields获取字段
FieldInfo f = t.GetField("Status"); //通过Type.GetField来获取指定名称的字段
f.SetValue(o, "奋发图强"); //给Status字段赋值
f.GetValue(o); //读取Status字段
//获取方法
MethodInfo[] methods = t.GetMethods();//通过Type.GetMethods获取方法
MethodInfo method_div = t.GetMethod("div"); //通过Type.GetMethod来获取指定名称的方法
method_div.Invoke(o, new object[] { , });//调用有2个参数的div方法