反射提供了封装程序集、模块和类型的对象(Type类型)。可以使用反射动态创建类型的实例。将类型绑定到现有对象,或从现有对象获取类型并调用其方法或访问其字段和属性。如果代码中使用了属性,可以利用反射对它们进行访问。
一、概述
反射在下列情况下很有用:
- 需要访问程序元数据的属性。
- 检查和实例化程序集中的类型
- 在运行时构建新类型。
- 执行后期绑定,访问在运行时创建的类型的方法。
在程序集中可以直接引用dll文件,从而达到使用该dll文件中的类和方法,现在可以通过反射机制达到同样的效果。
二、用途
有了反射,即可对每一个类型了如指掌。如果获得了构造函数的信息,即可直接创建对象,即使这个对象的类型在编译时还不知道。
- 使用 Assembly 定义和加载程序集,加载在程序集清单中列出的模块,以及从此程序集中查找类型并创建该类型的实例。
- 使用 Module 发现以下信息:包含模块的程序集以及模块中的类等。 您还可以获取在模块上定义的所有全局方法或其他特定的非全局方法。
- 使用 ConstructorInfo 发现以下信息:构造函数的名称、参数、访问修饰符(如 public 或 private)和实现详细信息(如 abstract 或 virtual)等。 使用 Type 的 GetConstructors 或 GetConstructor 方法来调用特定的构造函数。
- 使用 MethodInfo 发现以下信息:方法的名称、返回类型、参数、访问修饰符(如 public 或 private)和实现详细信息(如 abstract 或 virtual)等。 使用 Type 的 GetMethods 或 GetMethod 方法来调用特定的方法。
- 使用 FieldInfo 发现以下信息:字段的名称、访问修饰符(如 public 或 private)和实现详细信息(如static)等;并获取或设置字段值。
- 使用 EventInfo 发现以下信息:事件的名称、事件处理程序数据类型、自定义特性、声明类型和反射类型等;并添加或移除事件处理程序。
- 使用 PropertyInfo 发现以下信息:属性的名称、数据类型、声明类型、反射类型和只读或可写状态等;并获取或设置属性值。
- 使用 ParameterInfo 发现以下信息:参数的名称、数据类型、参数是输入参数还是输出参数,以及参数在方法签名中的位置等。
在一个应用程序域的仅反射上下文中工作时,使用 CustomAttributeData 来发现有关自定义特性的信息。 通过使用 CustomAttributeData,不必创建特性的实例就可以检查它们。
三、实践
小团队做了个Demo,小编就拿这个反射Demo举例说明吧。
反射加载程序集的几种方法
1、在小编的Demo中运用的反射实例如下实现。
<span style="font-size:18px;"> //使用Assembly 定义和加载程序集 Assembly assembly = Assembly.Load("Test_DAL");</span>
此时无需在引用相应的dll文件,但需要注意的是 输出路径的修改
2、小编尝试过其他的方式 ,加载路绝对路径,找到相应的文件类
<span style="font-size:18px;">//通过绝对路径加载方式一 Assembly assembly = Assembly.LoadFile("E:\\技术分享——徐露\\Test_Reflection\\Test_Reflection\\bin\\Debug\\Test_DAL.dll");</span>
<span style="font-size:18px;"> //通过绝对路径加载方式二 Assembly assembly = Assembly.LoadFrom ("E:\\技术分享——徐露\\Test_Reflection\\Test_Reflection\\bin\\Debug\\Test_DAL.dll");</span>
<span style="font-size:18px;">//通过绝对路径加载方式三 Assembly assembly = Assembly.ReflectionOnlyLoadFrom ("E:\\技术分享——徐露\\Test_Reflection\\Test_Reflection\\bin\\Debug\\Test_DAL.dll");</span>
是不是感觉如出一辙呢,后来小编一想通过默认的路径加载,此种方法运用反射就没有意义了,因为已经做不到动态更换dll,在试想加载默认路径和引用相应的dll,不是同样的么。所以说团队中
Assembly assembly = Assembly.Load("Test_DAL");
这种方式灵活性更好。
四、总结
ITOO项目中,小编身在其中,不知起意。通过这次技术分享,小编明白了:
- 在程序的加载过程中
通过反射创建实例化每一个类,此时的容器相当一个大的hashtable,容纳这些创建的类
<!--创建容器--> <configSections> <sectionGroup name="spring"> <section name="context" type="Spring.Context.Support.ContextHandler, Spring.Core" /> <section name="objects" type="Spring.Context.Support.DefaultSectionHandler, Spring.Core" /> </sectionGroup> </configSections>
- 创建之后,可以有两种方法获得在hashtable中所创建的类
①在配置文件中标识id(key值)
和type(value),根据WCF层中的key值查找获得value,从而找到相应的类
<spring> <context> <!--<resource uri="assembly://SpringLover/SpringLover/Objects.xml"/>--> <!--<resource uri="~/Objects.xml"/> <resource uri="~/objects.xml"/> <resource uri="file://objects.xml"/>--> <resource uri="config://spring/objects" /> </context> <objects xmlns="http://www.springframework.net"> <object id="FreshEntities" type="ITOO.FreshStuNumConfig.Model.FreshEntities,ITOO.FreshStuNumConfig.Model" singleton="false" /> <!--DbSession层的的注解--> <object id="DBSession" type="ITOO.FreshStuNumConfig.DAL.DBSession,ITOO.FreshStuNumConfig.DAL" singleton="false"> <!--加入属性注入,指向D层的注入--> <property name="StudentManageDal" ref="StudentManageDal" /> <property name="FreshClassDal" ref="FreshClassDal" /> <property name="FreshMajorDal" ref="FreshMajorDal" /> <property name="FreshDepartmentDal" ref="FreshDepartmentDal" /> </object> <!--D层的的注解--> <object id="StudentManageDal" type="ITOO.FreshStuNumConfig.DAL.StudentManageDal,ITOO.FreshStuNumConfig.DAL" singleton="false" /> <object id="FreshDepartmentDal" type="ITOO.FreshStuNumConfig.DAL.FreshDepartmentDal,ITOO.FreshStuNumConfig.DAL" singleton="false" /> <object id="FreshMajorDal" type="ITOO.FreshStuNumConfig.DAL.FreshMajorDal,ITOO.FreshStuNumConfig.DAL" singleton="false" /> <object id="FreshClassDal" type="ITOO.FreshStuNumConfig.DAL.FreshClassDal,ITOO.FreshStuNumConfig.DAL" singleton="false" /> <!--B层的的注解--> <object id="StudentManageBll" type="ITOO.FreshStuNumConfig.BLL.StudentManageBll,ITOO.FreshStuNumConfig.BLL" singleton="false" /> <object id="DistributeClassBll" type="ITOO.FreshStuNumConfig.BLL.DistributeClassBll,ITOO.FreshStuNumConfig.BLL" singleton="false" /> <object id="DistributeStudentNumberBll" type="ITOO.FreshStuNumConfig.BLL.DistributeStudentNumberBll,ITOO.FreshStuNumConfig.BLL" singleton="false" /> <object id="DepartmentBll" type="ITOO.FreshStuNumConfig.BLL.DepartmentBll,ITOO.FreshStuNumConfig.BLL" singleton="false" /> <object id="FreshMajorBll" type="ITOO.FreshStuNumConfig.BLL.FreshMajorBll,ITOO.FreshStuNumConfig.BLL" singleton="false" /> </objects> </spring>
在WCF层 key值可根据配置文件的id ,想对应查找出来
IStudentManageBll StudentBll = SpringHelper.GetObject<IStudentManageBll>("StudentManageBll");
②依赖注入 ,ITOO框架中的DbSession中就是通过这种方式实现的
//声明DbSession管理的StudentManageDal public IStudentManageDal StudentManageDal { get; set; }
这只是小编浅显的理解,和大家分享一下
版权声明:本文为博主原创文章,未经博主允许不得转载。