反射奥秘

反射是一个程序集发现及运行的过程,通过反射可以得到*.exe或*.dll等程序集内部的信息。使用反射可以看到一个程序集内部的接口、类、方法、字段、属性、特性等等信息。在System.Reflection命名空间内包含多个反射常用的类,下面表格列出了常用的几个类。

类型 作用
Assembly 通过此类可以加载操纵一个程序集,并获取程序集内部信息
EventInfo 该类保存给定的事件信息
FieldInfo 该类保存给定的字段信息
MethodInfo 该类保存给定的方法信息
MemberInfo 该类是一个基类,它定义了EventInfo、FieldInfo、MethodInfo、PropertyInfo的多个公用行为
Module 该类可以使你能访问多个程序集中的给定模块
ParameterInfo 该类保存给定的参数信息      
PropertyInfo 该类保存给定的属性信息

一、System.Reflection.Assembly类

通过Assembly可以动态加载程序集,并查看程序集的内部信息,其中最常用的就是Load()这个方法。

Assembly assembly=Assembly.Load("MyAssembly");

利用Assembly的object CreateInstance(string) 方法可以反射创建一个对象,参数0为类名。

二、System.Type类

Type是最常用到的类,通过Type可以得到一个类的内部信息,也可以通过它反射创建一个对象。一般有三个常用的方法可得到Type对象。

  1. 利用typeof() 得到Type对象

    Type type=typeof(Example);

  2. 利用System.Object.GetType() 得到Type对象

    Example example=new Example();

    Type type=example.GetType();

  3. 利用System.Type.GetType() 得到Type对象

    Type type=Type.GetType("MyAssembly.Example",false,true);

    注意参数0是类名,参数1表示若找不到对应类时是否抛出异常,参数1表示类名是否区分大小写

 例子:

我们最常见的是利用反射与Activator结合来创建对象。

Assembly assembly= Assembly.Load("MyAssembly");

Type type=assembly.GetType("Example");

object obj=Activator.CreateInstance(type);

三、反射方法

1.通过 System.Reflection.MethodInfo能查找到类里面的方法

代码:

Type type=typeof(Example);

MethodInfo[] listMethodInfo=type.GetMethods();

foreach(MethodInfo methodInfo in listMethodInfo)

Cosole.WriteLine("Method name is "+methodInfo.Name);

2.我们也能通过反射方法执行类里面的方法

代码:

Assembly assembly= Assembly.Load("MyAssembly");

Type type=assembly.GetType("Example");

object obj=Activator.CreateInstance(type);

MethodInfo methodInfo=type.GetMethod("Hello World");  //根据方法名获取MethodInfo对象

methodInfo.Invoke(obj,null);  //参数1类型为object[],代表Hello World方法的对应参数,输入值为null代表没有参数

四、反射属性

1.通过 System.Reflection.PropertyInfo 能查找到类里面的属性

常用的方法有GetValue(object,object[]) 获取属性值和 SetValue(object,object,object[]) 设置属性值

代码:

Type type=typeof(Example);

PropertyInfo[] listPropertyInfo=type.GetProperties();

foreach(PropertyInfo propertyInfo in listPropertyInfo)

Cosole.WriteLine("Property name is "+ propertyInfo.Name);

2.我们也可以通过以下方法设置或者获取一个对象的属性值

代码:

Assembly assembly=Assembly.Load("MyAssembly");

Type type=assembly.GetType("Example");

object obj=Activator.CreateInstance(type);

PropertyInfo propertyInfo=obj.GetProperty("Name");    //获取Name属性对象

var name=propertyInfo.GetValue(obj,null);                //获取Name属性的值

PropertyInfo propertyInfo2=obj.GetProperty("Age");     //获取Age属性对象

propertyInfo.SetValue(obj,34,null);                              //把Age属性设置为34

五、反射字段

通过 System.Reflection.FieldInfo 能查找到类里面的字段

它包括有两个常用方法SetValue(object ,object )和GetValue(object)  因为使用方法与反射属性非常相似,在此不再多作介绍

(略)

六、反射特性

通过System.Reflection.MemberInfo的GetCustomAttributes(Type,bool)就可反射出一个类里面的特性,以下例子可以反射出一个类的所有特性

代码:

Type type=typeof("Example");

object[] typeAttributes=type.GetCustomAttributes(false);       //获取Example类的特性

foreach(object attribute in typeAttributes)

Console.WriteLine("Attributes description is "+attribute.ToString());

通过下面例子,可以获取Example类Name属性的所有特性

 代码:

public class Example

{

[DataMemberAttribute]

publics string Name

{get;set;}

..................

}

Type type = typeof(Example);        
    PropertyInfo propertyInfo=type.GetProperty("Name");    //获取Example类的Name属性
    foreach (object attribute in propertyInfo.GetCustomAttributes(false))        //遍历Name属性的所有特性
           Console.WriteLine(“Property attribute: "+attribute.ToString());

七、常用实例

虽然反射有很多奥妙之处,但要注意使用反射生成对象会耗费很多性能,所能必须了解反射的特性,在合适的地方使用。最常见例子就是利用单体模式与反射一并使用,  在BLL调用DAL的时候,通过一个反射工厂生成DAL实例。

namespace Project.Common{    public class Factory    {        //记录dal的对象        private static Hashtable dals;        //用assemblyString记录DAL程序集的全名称        private static string assemblyString = ConfigurationManager.AppSettings["LinqDAL"];        private static Assembly assembly;

        static Factory()        {            dals = new Hashtable();            assembly = Assembly.Load(assemblyString);        }

        private static object CreateInstance(string typeName)        {            //当第一次加载时,将反射对象保存于dals集合里            if (!dals.ContainsKey(typeName))            {                //创建反射对象                object object1 = assembly.CreateInstance(typeName);

                if (object1 == null)                    throw new Exception("未能创建此对象");                //把对象加入dals集合                dals["typeName"] = object1;            }            return dals["typeName"];        }

        public static IExampleDAL CreateExampleDAL()        {            return (IExampleDAL)CreateInstance(assemblyString + ".ExampleDAL");        }    }

     class Program    {        //利用工厂模式生成对象        static void Main(string[] args)        {            IExampleDAL iExampleDAL=Factory.CreateExampleDAL();            .................            Console.ReadKey();        }    }}

namespace Project.IDAL{    public interface IExampleDAL    {        ///<summary>        /// 插入Example行,若插入成功,则返回新增Example的行数        ///</summary>        ///<param name="example">Example对象</param>        ///<returns>返回新增Example行数,默认值为-1</returns>        int AddExample(Example example);

        ///<summary>        /// 更新Example表,Update成功返回已经更新的数据条数,失败返回-1        ///</summary>        ///<param name="example">Example对象</param>        ///<returns>Update成功返回已经更新的数据条数,失败返回-1</returns>        int UpdateExample(Example example);

        ///<summary>        /// 删除Example表中ID等于exampleID的行,返回已删除行数        ///</summary>        ///<param name="exampleID">Example对象的ID值</param>        ///<returns>返回删除行数</returns>        int DeleteExample(int exampleID);

        ///<summary>        /// 获取Example表的所有行        ///</summary>        ///<returns>返回Example表中的所有Example对象</returns>        IList<Example> GetList();

        ///<summary>        ///  根据ID获取对应Example对象        ///</summary>        ///<param name="id"></param>        ///<returns></returns>        Example GetExampleByID(int id);    }}

namespace Project.DAL{    public class ExampleDAL:IExampleDAL    {        public int AddExample(Example example)        {             //实现AddExample方法             ...........................        }        ..................................        ..................................    }}

上面只是介绍了反射的基本用法,当然反射还有很多其他用途,下面几篇文章将为大家一一介绍

时间: 2024-09-20 05:27:03

反射奥秘的相关文章

谈一谈:抽象工厂+反射+配置文件 实现数据访问程序

<大话设计模式>中第15章中<就不能不换DB吗?>引出了我今天要谈论的主题:抽象工厂+反射+配置文件 实现数据访问程序.当时也不甚理解啊!到了机房收费的亲身实践中,终于体会到了这对组合的奥秘. 抽象工厂模式(Abstract Factory) 提供一个创建一系列相关或相互依赖对象的接口,而无需制定它们具体的类.知道它是用来创建工厂的就OK了. 反射 提供了封装程序集.模块和类型的对象.这里仅仅用到反射的一部分功能.且记住它的格式: Assembly.Load("程序集名称

.NET简谈反射(动态调用)

我们继续C#基础知识的学习,这篇文章主要要讲的是我们C#程序员迈向高级C#程序员的关键性的一步. 有的朋友会说事实不是这样的,我不用反射就不能开发吗?当然可以,但是用与不用肯定是不一样的,任何复杂抽象的分层架构或者说是复杂的设计模式均是建立在这些基础之上的,比如我们要进行模块化.组件化开发,要严格的消除模块之间的耦合,要进行动态接口调用.这样的强大而灵活的系统开发,必须要用反射才行:任何技术都有它存在的价值和意义,只要我们把它用在合适的位置就能发挥出惊人的力量:能尽可能的减少我们编写的代码,更能

delphi 反射(原理)

关于反射的用途是『降低模块间的耦合度』这个倒未必尽然 单就delphi来说,从实现上看,它的所谓反射是基于RTTI,而RTTI的出现按照官方的说法是为了实现RAD中窗体文件DFM的持久化而产生的,其实也不是针对DFM文件或TForm啦,由于TPersistent在声明的时候加上了{$M+},所以从TPersistent派生的对象都在编译的时候添加了RTTI,而在TComponent中又增加了对TReader和TWriter支持,说的准确一点.时髦一点RTTI是为了实现对象的持久化和反持久化,在D

.NET Core跨平台的奥秘[上篇]:历史的枷锁

微软推出的第一个版本的.NET Framework是一个面向Windows桌面和服务器的基础框架,在此之后,为此微软根据设备自身的需求对.NET Framework进行裁剪,不断推出了针对具体设备类型的.NET Framework版本以实现针对移动.平板和嵌入式设备提供支持.除此之外,在Windows平台之外一致游荡着一只特立独行的猴子(Mono)..NET平台看起来欣欣向荣,而实际上却日薄西山,就在这个时候微软走了一条唯一正确的道路,那就是基于跨平台理念重新设计的.NET Core,以及由此驱

NET Core跨平台的奥秘[中篇]:复用之殇

从本质上讲,按照CLI规范设计的.NET从其出生的那一刻就具有跨平台的基因,这与Java别无二致.由于采用了统一的中间语言,微软只需要针对不同的平台设计不同的虚拟机(运行时)就能弥合不同操作系统与处理器架构之间的差异,但是"理想很丰满,现实很骨感".在过去十多年中,微软将.NET引入到了各个不同的应用领域,表面上看起来似乎欣欣向荣,但是由于采用完全独立的多目标框架的设计思路,导致针对多目标框架的代码平台只能通过PCL(参考<.NET Core跨平台的奥秘[中篇]:复用之殇>

Android小例子:使用反射机制来读取图片制作一个图片浏览器

效果图: 工程文件夹: 该例子可供于新手参考练习,如果有哪里不对的地方,望指正>-< <黑幕下的人> java代码(MainActivity.java): package com.example.imageswitchtest; import java.lang.reflect.Field; import android.app.Activity; import android.os.Bundle; import android.util.Log; import android.v

.Net 反射

反射是.NET中的重要机制,通过反射,可以在运行时获得程序或程序集中每一个类型(包括类.结构.委托.接口和枚举等)的成员和成员的信息.有了反射,即可对每一个类型了如指掌.另外我还可以直接创建对象,即使这个对象的类型在编译时还不知道.     反射的用途:    (1)使用Assembly定义和加载程序集,加载在程序集清单中列出模块,以及从此程序集中查找类型并创建该类型的实例.     (2)使用Module了解包含模块的程序集以及模块中的类等,还可以获取在模块上定义的所有全局方法或其他特定的非全

C#图解教程 第二十四章 反射和特性

反射和特性元数据和反射Type 类获取Type对象什么是特性应用特性预定义的保留的特性Obsolete(废弃)特性Conditional特性调用者信息特性DebuggerStepThrough 特性其他预定义特性有关应用特性的更多内容多个特性其他类型的目标全局特性自定义特性声明自定义特性使用特性的构造函数指定构造函数使用构造函数构造函数中的位置参数和命名参数限制特性的使用自定义特性的最佳实践访问特性使用IsDefined方法使用GetCustomAttributes方法 Note 类的元数据包含

Java反射

1. 介绍 反射是一种能够在程序运行时动态访问.修改某个类中任意属性和方法的机制. 具体:对于任意一个类,都能够知道这个类的所有属性和方法对于任意一个对象,都能够调用它的任意一个方法和属性 在运行时,当加载完类之后,JVM在堆内存中会自动产生一个Class类型的对象,这个对象包含了完整的类的结构信息 这个Class对象就像一面镜子,透过这个镜子看到类的结构 那么,如何得到这个Class对象呢?以下可否 Class c = new Class(); 答案是不行的,因为Class的构造函数定义为私有