对于反射的第一次接触是在做机房的时候,那个时候感觉就是在配置文件中写入连接数据库的类型以及字符串,这样对于更换数据库就比较灵活,感觉反射的其他东西也没接触过,今天再次学习反射。简单记录一下学习。
概念:在计算机科学领域,反射是指一类应用,它们能够自描述和自控制。也就是说,这类应用通过采用某种机制来实现对自己行为的描述(self-representation)和监测(examination),并能根据自身行为的状态和结果,调整或修改应用所描述行为的状态和相关的语义。
扩展:Java中,反射是一种强大的工具。它使您能够创建灵活的代码,这些代码可以在运行时装配,无需在组件之间进行源代表链接。反射允许我们在编写与执行时,使我们的程序代码能够接入装载到JVM中的类的内部信息,而不是源代码中选定的类协作的代码。这使反射成为构建灵活的应用的主要工具。但需注意的是:如果使用不当,反射的成本很高。
优缺点:提高了系统的灵活性,但增加了系统的开销,系统的性能会有损耗。
实例讲解:
下面是普通的一个类:
#region Author & Version /* ********************************************************************************** *作者: * 小组: * 说明: *创建日期: * 版本号: ********************************************************************************* */ #endregion using System; using System.Collections.Generic; using System.Linq; using System.Text; //using System.Threading.Tasks; namespace DAL { public class User { public string Field = "Hello World"; public string Name { get; set; } public User() { this.Name = "无参构造"; } public User(string name) { this.Name = name; } public void PublishMethod() { Console.WriteLine(string.Format("反射调用一个public方法")); } private void PrivateMethod() { Console.WriteLine(string.Format("反射调用一个Private方法")); } //静态方法 public static string StaticMethod() { return "反射调用一个Static方法"; } //public 带有参数返回值函数 public string PublishMethodReturn(string name) { return string.Format("反射调用一个public带参带返回值函数,参数为:{0}", name); } } }
这个类里面没有实例化也没有添加上面这个类的引用,通过反射机制来实现上面这个类的调用。
using System; using System.Collections.Generic; using System.Linq; using System.Text; using System.Threading.Tasks; using System.Reflection; namespace Reflection { class Program { static void Main(string[] args) { //使用 Assembly 定义和加载程序集 //加载在程序集清单中列出的模块 //以及在此程序集中查找类型并创建给类型的实例 //获取程序集 Assembly assembly = Assembly.Load("DAL"); //重程序集中获取指定对象类型 Type type = assembly.GetType("DAL.User"); //使用Activator创建实例(无参数构造函数) var userNoParameter = Activator.CreateInstance(type ); //使用Activator创建实例(带参数构造函数) var userParameter = Activator.CreateInstance(type,"wangpeng"); //使用无参构造函数创建类(先反射创建构造函数,再使用构造函数创建类) ConstructorInfo constructorNoParameter = type.GetConstructor(new Type[] { typeof(string) }); var userNoParameterByConstructorInfo = constructorNoParameter.Invoke(new object[]{"wangpeng"}); //调用public 函数(无参数) type.InvokeMember("PublishMethod", BindingFlags.InvokeMethod|BindingFlags.Public| BindingFlags.Instance,null, userNoParameter, null); //调用public函数(有参数) string returnValuePublic = type.InvokeMember("PublishMethodReturn", BindingFlags.InvokeMethod|BindingFlags.OptionalParamBinding,null,userNoParameter, new object[]{"wangpeng"}) as string; Console.WriteLine(returnValuePublic); //调用静态方法 string returnValueStatic = type.InvokeMember("StaticMethod",BindingFlags.InvokeMethod|BindingFlags.Public |BindingFlags.Static, null,null,new object[]{}) as string; Console.WriteLine(returnValueStatic); //反射属性 var Name = type.InvokeMember("Name",BindingFlags.GetProperty| BindingFlags.Public |BindingFlags.Instance,null, userNoParameter,new object [] {} ) as string; Console.WriteLine(Name); //设置属性(设置Name属性为"NewName") type.InvokeMember("Name",BindingFlags.SetProperty|BindingFlags.Public|BindingFlags.Instance,null, userNoParameter,new object[]{"NewName"}); //反射字段 string Field = type.InvokeMember("Field",BindingFlags.GetField|BindingFlags.Public| BindingFlags.Instance,null, userNoParameter,new object[]{}) as string; Console.WriteLine(Field); //设置字段(设置Field字段为"NewField") type.InvokeMember("Field",BindingFlags.SetField|BindingFlags.Public |BindingFlags.Instance,null, userNoParameter,new object[]{"NewField"}); } } }运行结果如下:
总结:
反射提高了系统的灵活性,但增加了系统的开销降低了系统的性能
(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命名空间中的类通过序列化来间接地使用反射。
采用反射技术可以简化工厂的实现。
(1)工厂方法:通过反射可以将需要实现的子类名称传递给工厂方法,这样无须在子类中实现类的实例化。
(2)抽象工厂:使用反射可以减少抽象工厂的子类。
采用反射技术可以简化工厂代码的复杂程度,在.NET项目中,采用反射技术的工厂已经基本代替了工厂方法。
采用反射技术可以极大地简化对象的生成,对以下设计模式的实现也有很大影响。
(1)命令模式:可以采用命令的类型名称作为参数直接获得命令的实例,并且可以动态执行命令。
(2)享元模式:采用反射技术实例化享元可以简化享元工厂。