反射的初识

       对于反射的第一次接触是在做机房的时候,那个时候感觉就是在配置文件中写入连接数据库的类型以及字符串,这样对于更换数据库就比较灵活,感觉反射的其他东西也没接触过,今天再次学习反射。简单记录一下学习。

       概念:在计算机科学领域,反射是指一类应用,它们能够自描述和自控制。也就是说,这类应用通过采用某种机制来实现对自己行为的描述(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)享元模式:采用反射技术实例化享元可以简化享元工厂。






你可能感兴趣的:(反射)