程序集与反射技术(C#)
转载:https://www.cnblogs.com/czx1/p/201413137070-com.html
首先我们来看程序集,程序集是代码进行编译是的一个逻辑单元,把相关的代码和类型进行组合,然后生成PE文件(例如可执行文件.exe和类库文件.dll)。由于程序集在编译后并不一定会生成单个文件,而可能会生成多个物理文件,甚至可能会生成分布在不同位置的多个物理文件,所以程序集是一个逻辑单元,而不是一个物理单元。即程序集在逻辑上是一个编译单元,但在物理储存上可以有多种存在形式。对于静态程序集可以生成单个或多个文件,而动态程序集是存在于内存中的。在C#中程序集处处可见,因为任何基于.NET的代码在编译时都至少存在一个程序集(所有.NET项目都会默认引用mscorlib程序集)。
基于.NET框架的.dll库是一个完整的程序集,需要事先引用对应的类库。从代码的结构上看,一个程序集可以包含一个或多个命名空间,而每个命名空间中又可以包含子命名空间或类型列表。由于程序集在编译后可以生成多个模块文件,因此一个物理文件并不代表它就是一个程序集,一个程序集并不一定只有一个文件。不过,在VS开发环境中,一个解决方案可以包含多个项目,而每个项目就是一个程序集。
程序集信息其实是通过在程序集上应用各种Attribute来设置的,并且这些特性都位于System.Reflection命名空间,也就是这一系列Attribute提供给反射技术,可用于获取程序集的基本信息.
应用程序结构分为应用程序域—程序集—模块—类型—成员几个层次,公共语言运行库加载器管理应用程序域,这种管理包括将每个程序集加载到相应的应用程序域以及控制每个程序集中类型层次结构的内存布局。
程序集包含模块,而模块包含类型,类型又包含成员,反射则提供了封装程序集、模块和类型的对象。我们可以使用反射动态地创建类型的实例,将类型绑定到现有对象或从现有对象中获取类型,然后调用类型的方法或访问其字段和属性。反射通常具有以下用途。
(1)使用Assembly定义和加载程序集,加载在程序集清单中列出模块,以及从此程序集中查找类型并创建该类型的实例。
(2)使用Module了解包含模块的程序集以及模块中的类等,还可以获取在模块上定义的所有全局方法或其他特定的非全局方法。
(3)使用ConstructorInfo了解构造函数的名称、参数、访问修饰符(如pulic
或private)和实现详细信息(如abstract或virtual)等。使用Type的GetConstructors或GetConstructor方法来调用特定的构造函数。
(4)使用MethodInfo了解方法的名称、返回类型、参数、访问修饰符(如pulic 或private)和实现详细信息(如abstract或virtual)等。使用Type的GetMethods或GetMethod方法来调用特定的方法。
(5)使用FiedInfo了解字段的名称、访问修饰符(如public或private)和实现详细信息(如static)等,并获取或设置字段值。
(6)使用EventInfo了解事件的名称、事件处理程序数据类型、自定义属性、声明类型和反射类型等,添加或移除事件处理程序。
(7)使用PropertyInfo了解属性的名称、数据类型、声明类型、反射类型和只读或可写状态等,获取或设置属性值。
(8)使用ParameterInfo了解参数的名称、数据类型、是输入参数还是输出参数,以及参数在方法签名中的位置等。
动态加载程序集:
(1)通过程序集的长名称加载,一般用于加载位于全局程序集缓存中的程序集。(.NET框架的类库都位于全局程序集缓存中)(使用GacUtil命令行工具可以获取System程序集的长名称);
(2)通过指定的文件加载程序集,通常是.dll或.exe文件。
例如:本人的计算机上的全局程序集缓存包含下列程序集:
system, Version=2.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089, processorArchitecture=MSIL
system, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089, processorArchitecture=MSIL
项目数 = 2
反射(Reflection)是.NET中的重要机制,通过反射,可以在运行时获得.NET中每一个类型(包括类、结构、委托、接口和枚举等)的成员,包括方法、属性、事件,以及构造函数等。还可以获得每个成员的名称、限定符和参数等。有了反射,即可对每一个类型了如指掌。如果获得了构造函数的信息,即可直接创建对象,即使这个对象的类型在编译时还不知道。
1)反射用于获取类型信息。下面一个例子:
1 using System;
2 using System.Collections.Generic;
3 using System.Linq;
4 using System.Text;
5 using System.Threading.Tasks;
6 using System.Reflection;
7
8 namespace My
9 {
10 class Program
11 {
12 static void Main(string[] args)
13 {
14 // 加载程序集
15 Assembly ass = Assembly.Load("system.windows.forms, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089");
16 // 获取Button类的Type,要写命名空间.类名;例外的情况.net6.0 新写法 在没有写命名空间时候,默认代码存储在Program类的main函数中,而Program没有命名空间,因此反射Program类时候不用写命名空间
17 Type typeofButton = ass.GetType("System.Windows.Forms.Button", false);
18 if (typeofButton != null)
19 {
20 // 获得Button类型的公共属性列表
21 PropertyInfo[] props = typeofButton.GetProperties();
22 // 输出属性信息
23 Console.WriteLine("\n Button类的公共属性列表:");
24 foreach (PropertyInfo p in props)
25 {
26 Console.WriteLine(" 属性名:{0},属性类型:{1}", p.Name, p.PropertyType.Name);
27 }
28 // 获取Button类型的公共事件列表
29 EventInfo[] events = typeofButton.GetEvents();
30 // 输出事件信息
31 Console.WriteLine("\n Button类的公共事件列表:");
32 foreach (EventInfo ev in events)
33 {
34 Console.WriteLine(" 事件名:{0},事件类型:{1}", ev.Name, ev.EventHandlerType.Name);
35 }
36 }
37 Console.Read();
38 }
39 }
40 }
2)反射还可以在运行阶段动态创建类型实例,然后就可以动态调用实例的成员。
动态创建实例有两种方法可以选择:
第一种:使用Activator类的CreateInstance静态方法,其中方法返回的对象就是对应的实例引用。
看个例子:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
using System.Reflection;
namespace Me
{
class Program
{
static void Main(string[] args)
{
// 从当前执行的程序集中获取Test类型的Type
Assembly ass = Assembly.GetExecutingAssembly();
Type tp = ass.GetType("Me.Test", false);
if (tp != null)
{
// 创建实例
object instance = Activator.CreateInstance(tp, 105);
if (instance != null)
{
// 获取实例的公共属性
PropertyInfo[] props = tp.GetProperties(BindingFlags.Public | BindingFlags.Instance);
// 显示属性信息
foreach (var p in props)
{
Console.WriteLine("{0} : {1}", p.Name, p.GetValue(instance));
}
}
}
Console.Read();
}
}
public class Test
{
public int NumValue { get; private set; }
public Test(int sead)
{
Random rand = new Random(sead);
NumValue = rand.Next();
}
}
}
第二种:通过反射找出类型的构造函数,然后调用构造函数来获取实例引用。
代码如下:
1 using System;
2 using System.Collections.Generic;
3 using System.Linq;
4 using System.Text;
5 using System.Threading.Tasks;
6 using System.Reflection;
7
8 namespace My
9 {
10 class Program
11 {
12 static void Main(string[] args)
13 {
14 // 从mscorlib程序集中获取DateTime结构的Type
15 Type tpdatetime = Type.GetType("System.DateTime", false);
16 if (tpdatetime != null)
17 {
18 // 获取具有六个参数的构造函数
19 ConstructorInfo constructor = tpdatetime.GetConstructor(new Type[] { typeof(int), typeof(int), typeof(int), typeof(int), typeof(int), typeof(int) });
20 if (constructor != null)
21 {
22 // 调用构造函数创建实例
23 object instance = constructor.Invoke(new object[] { 2015, 9, 27, 13, 25, 27 });
24 // 获取实例的属性列表
25 if (instance != null)
26 {
27 PropertyInfo[] props = tpdatetime.GetProperties(BindingFlags.Public | BindingFlags.Instance);
28 Console.WriteLine("以下是DateTime实例的属性值列表:");
29 foreach (PropertyInfo p in props)
30 {
31 // 获取属性值
32 object objval = p.GetValue(instance);
33 // 显示属性名称与属性值
34 Console.WriteLine("{0,-15} :{1}", p.Name, objval ?? string.Empty);
35 }
36 }
37 }
38 }
39
40 Console.Read();
41 }
42 }
43 }
学习到这里,相信你已经对这部分知识有啦相当一部分了解,由于是第一次写博客,或许写的不是很好,若有错,请联系我,下期再见。。。。