反射<编程>
程序集包含模块,而模块包含类型,类型又包含成员。反射则提供了封装程序集、模块和类型的对象。您可以使用反射动态地创建类型的实例,将类型绑定到现有对象,或从现有对象中获取类型。然后,可以调用类型的方法或访问其字段和属性。反射通常具有以下用途:
1.使用 Assembly 定义和加载程序集,加载在程序集清单中列出的模块,以及从此程序集中查找类型并创建该类型的实例。
2.使用 Module 了解如下的类似信息:包含模块的程序集以及模块中的类等。您还可以获取在模块上定义的所有全局方法或其他特定的非全局方法。
3.使用 ConstructorInfo 了解如下的类似信息:构造函数的名称、参数、访问修饰符(如 public 或 private)和实现详细信息(如 abstract 或 virtual)等。
4.使用 Type 的 GetConstructors 或 GetConstructor 方法来调用特定的构造函数。
5.使用 MethodInfo 来了解如下的类似信息:方法的名称、返回类型、参数、访问修饰符(如 public 或 private)和实现详细信息(如 abstract 或 virtual)等。使用 Type 的 GetMethods 或 GetMethod 方法来调用特定的方法。
6.使用 FieldInfo 来了解如下的类似信息:字段的名称、访问修饰符(如 public 或 private)和实现详细信息(如 static)等;并获取或设置字段值。
7.使用 EventInfo 来了解如下的类似信息:事件的名称、事件处理程序数据类型、自定义属性、声明类型和反射类型等;并添加或移除事件处理程序。
8.使用 PropertyInfo 来了解如下的类似信息:属性的名称、数据类型、声明类型、反射类型和只读或可写状态等;并获取或设置属性值。
9.使用 ParameterInfo 来了解如下的类似信息:参数的名称、数据类型、参数是输入参数还是输出参数,以及参数在方法签名中的位置等。
概念
反射机制是使程序具有动态特性的非常关键的一种机制
需求1:根据指定的类名,类字段名和所对应的数据,得到该类的实例。
需求2:根据指定的装箱对象 获取其属性值或者调用其方法
需求3:获取未来会产生的程序集 并使用其中的类
让我们可以于运行时加载、探知、使用编译期间完全未知的classes。换句话说,程序可以加载一个运行时才得知名称的class,获悉其完整构造(但不包括methods定义),并生成其对象实体、或对其fields设值、或唤起其methods1。
反射需要的类
反射用到的命名空间:
System.Reflection
System.Type
System.Reflection.Assembly
(1)使用Assembly定义和加载程序集,加载在程序集清单中列出模块,以及从此程序集中查找类型并创建该类型的实例。
(2)使用Type这个类可以访问任何给定数据类型的信息。
(3)使用ConstructorInfo了解构造函数的名称、参数、访问修饰符(如pulic 或private)和实现详细信息(如abstract或virtual)等。
(4)使用MethodInfo了解方法的名称、返回类型、参数、访问修饰符(如pulic 或private)和实现详细信息(如abstract或virtual)等。
(5)使用FiedInfo了解字段的名称、访问修饰符(如public或private)和实现详细信息(如static)等,并获取或设置字段值。
(6)使用PropertyInfo了解属性的名称、数据类型、声明类型、反射类型和只读或可写状态等,获取或设置属性值。
System.Type类:
System.Type 类对于反射起着核心的作用。但它是一个抽象的基类,Type有与每种数据类型对应的派生类,我们使用这个派生类的对象的方法、字段、属性来查找有关该类型的所有信息。
获取给定类型的Type引用有3种常用方式:
●使用 C# typeof 运算符。已知类型
Type t = typeof(string)
●使用对象GetType()方法。 已知对象
string s = "grayworm"
Type t = s.GetType()
●还可以调用Type类的静态方法GetType()。已知类名
Type t = Type.GetType("System.String")
反射生成的对象
1.用Type构建对象
用构造函数动态生成对象:
既然有反射 那就有“正射”
对比一下:
有一个class Test 有无参构造方法
获取实例的比较:
“正” :由类名直接使用关键字new来获得 语法:new Test()
反射:未知类名 运行时获得类型名:例如:Type type = Type.GetType("Test")
Object o = type.GetConstructor (new Type[]{}).Invoke (null)
当然参数也可以由配置文件获得
以上调用无参构造 带参的构造怎么调用呢?
2.用Activator生成对象:
Type t = typeof(NewClassw)
//构造函数的参数
object[] obj = new object[2] { "实参1", "实参2" }
//用Activator的CreateInstance静态方法,生成新对象
object o = Activator.CreateInstance(t,obj)
3.Assembly生成对象:
Assembly assembly = Assembly.Load ("ReflectTest")
Object obj = assembly.CreateInstance ("ReflectTest.Wood")
反射获取字段值
一般用于已知对象时使用
同样用正反来做对比:
已知对象MyClass obj
对象内有name这个字段
正:
obj.name = "xxx"即可设置该对象的值
obj.name即可获得该对象的值
反射:
//获取类的描述对象
Type t = obj.GetType()
//取得字段描述对象
FieldInfo fi = t.GetField("name")
//获取或设置字段值
fi.SetValue(obj, "k001")
fi.GetValue(obj)
一般在C#中以属性替代字段
//获取类的描述对象
Type t = obj.GetType()
//取得属性描述对象
PropertyInfo pi1 = t.GetProperty("Name")
//获取或设置属性值
pi1.SetValue(obj, "k001",null)
pi1.GetValue(obj,null)
反射执行方法
一般用于对象、方法名、参数类型已知
对象:obj
方法名:methodName
参数类型 String int
正:
直接给予参数调用:obj.methodName("实参1",实参2(int))
反射:
//获取类的描述对象
Type t = obj.GetType()
//取得方法描述对象
MethodInfo mi = t.GetMethod("show")
//调用方法
mi.Invoke(obj, "实参1",实参2(int))
参数可以换成数组
动态载入程序集
System.Reflection.Assembly类
Assembly类可以获得程序集的信息,也可以动态的加载程序集,以及在程序集中查找类型信息,并创建该类型的实例。
使用Assembly类可以降低程序集之间的耦合,有利于软件结构的合理化。
通过程序集名称返回Assembly对象
Assembly ass = Assembly.Load("ClassLibrary831")
通过DLL文件名称返回Assembly对象
Assembly ass = Assembly.LoadFrom("ClassLibrary831.dll")
通过Assembly获取程序集中类
Type t = ass.GetType("ClassLibrary831.NewClass") //参数必须是类的全名
通过Assembly获取程序集中所有的类
Type[] t = ass.GetTypes()
//通过程序集的名称反射
Assembly ass = Assembly.Load("ClassLibrary831")
Type t = ass.GetType("ClassLibrary831.NewClass")
object o = Activator.CreateInstance(t, "", "")
MethodInfo mi = t.GetMethod("show")
mi.Invoke(o, null)
//通过DLL文件全名反射其中的所有类型
Assembly assembly = Assembly.LoadFrom("xxx.dll的路径")
Type[] aa = a.GetTypes()
foreach(Type t in aa)
{
if(t.FullName == "a.b.c")
{
object o = Activator.CreateInstance(t)
}
}
小结
反射属于面向对象的动态补充
让C#的面向对象做到"所见即所得"
反射(Reflection)是.NET中的重要机制,通过放射,可以在运行时获得.NET中每一个类型(包括类、结构、委托、接口和枚举等)的成员,包括方法、属性、事件,以及构造函数等。还可以获得每个成员的名称、限定符和参数等。有了反射,即可对每一个类型了如指掌。如果获得了构造函数的信息,即可直接创建对象,即使这个对象的类型在编译时还不知道。程序代码在编译后生成可执行的应用,我们首先要了解这种可执行应用程序的结构。
应用程序结构分为应用程序域—程序集—模块—类型—成员几个层次,公共语言运行库加载器管理应用程序域,这种管理包括将每个程序集加载到相应的应用程序域以及控制每个程序集中类型层次结构的内存布局。
程序集包含模块,而模块包含类型,类型又包含成员,反射则提供了封装程序集、模块和类型的对象。我们可以使用反射动态地创建类型的实例,将类型绑定到现有对象或从现有对象中获取类型,然后调用类型的方法或访问其字段和属性。反射通常具有以下用途。
(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了解参数的名称、数据类型、是输入参数还是输出参数,以及参数在方法签名中的位置等。
System.Reflection.Emit命名空间的类提供了一种特殊形式的反射,可以在运行时构造类型。
反射也可用于创建称为类型浏览器的应用程序,使用户能够选择类型,然后查看有关选定类型的信息。
此外,Jscript等语言编译器使用反射来构造符号表。System.Runtime.Serialization命名空间中的类使用反射来访问数据并确定要永久保存的字段,System.Runtime.Remoting命名空间中的类通过序列化来间接地使用反射。
欢迎分享,转载请注明来源:内存溢出
评论列表(0条)