C#反射(Reflection)

一、反射概述

(能够通过非反射方式实现的功能,尽量不要用反射)

反射是.NET中的重要机制,通过反射可以得到*.exe或*.dll等程序集内部的接口、类、方法、字段、属性、特性等信息,还可以动态创建出类型实例并执行其中的方法。

反射的定义:审查元数据并收集关于它的类型信息的能力。
元数据(编译以后的最基本数据单元)就是一大堆的表,当编译程序集或者模块时,编译器会创建一个类定义表,一个字段定义表,和一个方法定义表等。

1)什么是反射

Reflection,中文翻译为反射。
这是.Net中获取运行时类型信息的方式,.Net的应用程序由几个部分:‘程序集(Assembly)’、‘模块(Module)’、‘类型(class)’组成,而 反射提供一种编程的方式,让程序员可以在程序运行期获得这几个组成部分的相关信息(反射指程序可以访问、检测和修改它本身状态或行为的一种能力),例如:
Assembly类可以获得正在运行的装配件信息,也可以动态的加载装配件,以及在装配件中查找类型信息,并创建该类型的实例。
Type类可以获得对象的类型信息,此信息包含对象的所有要素:方法、构造器、属性等等,通过Type类可以得到这些要素的信息,并且调用之。
MethodInfo包含方法的信息,通过这个类可以得到方法的名称、参数、返回值等,并且可以调用之。
诸如此类,还有FieldInfo、EventInfo等等,这些类都包含在System.Reflection命名空间下。

2)命名空间与装配件

1.介绍

  • 命名空间 :类库的逻辑组织形式 (命名空间类似于Java的包,但又不完全等同,因为Java的包必须按照目录结构来放置,命名空间则不需要)

  • 程序集(装配件):类库的物理组织形式 (程序集是.Net应用程序执行的最小单位,.Net应用程序包含一个或多个程序集,编译出来的.dll、.exe都是程序集,.NET程序集包含元数据,这些元数据描述了程序集中定义的所有类型及其成员的信息,即方法、属性、事件和字段。【CLI程序集可分为两类:进程程序集(EXE)、库程序集(DLL),其中.exe文件是一个自己执行的程序集,而.dll将被其他程序集加载后运行。】)

2.联系

命名空间与程序集(在一个程序集中可以有不同的名称空间,同一个名称空间也可以分布在多个程序集上)

举个例子:

  • 程序集A

namespace  N1
{
      public  class  AC1  {…}
      public  class  AC2  {…}
}
namespace  N2
{
      public  class  AC3  {…}
      public  class  AC4{…}
}
  • 程序集B

namespace  N1
{
      public  class  BC1  {…}
      public  class  BC2  {…}
}
namespace  N2
{
      public  class  BC3  {…}
      public  class  BC4{…}
}
  • 理解:

这两个装配件中都有N1和N2两个命名空间,而且各声明了两个类,这样是完全可以的,然后我们在一个应用程序中引用装配件A,那么在这个应用程序中,我们能看到N1下面的类为AC1和AC2,N2下面的类为AC3和AC4。
接着我们去掉对A的引用,加上对B的引用,那么我们在这个应用程序下能看到的N1下面的类变成了BC1和BC2,N2下面也一样。
如果我们同时引用这两个装配件,那么N1下面我们就能看到四个类:AC1、AC2、BC1和BC2。
到这里,我们可以清楚一个概念了, 命名空间只是说明一个类型是那个族的,比如有人是汉族、有人是回族;而装配件表明一个类型住在哪里,比如有人住在北京、有人住在上海;那么北京有汉族人,也有回族人,上海有汉族人,也有回族人,这是不矛盾的。
上面我们说了,装配件是一个类型居住的地方,那么在一个程序中要使用一个类,就必须告诉编译器这个类住在哪儿,编译器才能找到它,也就是说必须引用该装配件。
那么如果在编写程序的时候,也许不确定这个类在哪里,仅仅只是知道它的名称,就不能使用了吗?答案是可以,这就是反射了, 就是在程序运行的时候提供该类型的地址,而去找到它

3.应用

只有同时指定类型所在的命名空间及实现该类型的程序集,才能完全限定该类型
(摘抄自《精通.NET核心技术--原来与架构》 电子工业出版社)

也就是说,你要创建一个类的实例,必须知道该类的 命名空间(这个一般都知道)+程序集(这个很容易被我们忽略,因为一般我们不需要引用外部的程序集,如果你用C#做过Excel文件的导入导出,就会知道必须添加一个Excel相关的程序集引用)

3)System.reflection命名空间

官方文档:链接地址

System.reflection包含的几个类,允许你反射(解析)这些元数据表的代码。

System.Reflection.Assembly

System.Reflection.MemberInfo

System.Reflection.EventInfo

System.Reflection.FieldInfo

System.Reflection.MethodBase

System.Reflection.ConstructorInfo

System.Reflection.MethodInfo

System.Reflection.PropertyInfo

System.Type

Assembly定义和加载程序集,加载在程序集清单中列出模块,以及从此程序集中查找类型并创建该类型的实例。

MemberInfo获取有关成员属性的信息并提供对成员元数据的访问权限。

EventInfo了解事件的名称、事件处理程序数据类型、自定义属性、声明类型和反射类型等,添加或移除事件处理程序。

FieldInfo了解字段的名称、访问修饰符(如public或private)和实现详细信息(如static)等,并获取或设置字段值。

MethodBase提供有关方法和构造函数的信息。

ConstructorInfo了解构造函数的名称、参数、访问修饰符(如pulic 或private)和实现详细信息(如abstract或virtual)等。使用Type的GetConstructors或 GetConstructor方法来调用特定的构造函数。

MethodInfo了解方法的名称、返回类型、参数、访问修饰符(如pulic 或private)和实现详细信息(如abstract或virtual)等。使用Type的GetMethods或GetMethod方法来调用特定的方法。

PropertyInfo发现属性 (Property) 的属性 (Attribute) 并提供对属性 (Property) 元数据的访问。了解属性的名称、数据类型、声明类型、反射类型和只读或可写状态等,获取或设置属性值。

Type访问元数据的主要方式,类型。

基础知识 另一文章:C#反射相关基础

4)相关知识

1.反射的作用

1、可以使用反射动态地创建类型的实例,将类型绑定到现有对象,或从现有对象中获取类型

2、应用程序需要在运行时从某个特定的程序集中载入一个特定的类型,以便实现某个任务时可以用到反射。

3、反射主要应用于类库,这些类库需要知道一个类型的定义,以便提供更多的功能。

2.应用要点

1、现实应用程序中很少有应用程序需要使用反射类型

2、使用反射动态绑定需要牺牲性能

3、有些元数据信息是不能通过反射获取的

4、某些反射类型是专门为那些clr 开发编译器的开发使用的,所以你要意识到不是所有的反射类型都是适合每个人的。

二、反射应用

1)创建对象

1.根据类型来动态创建对象

System.Activator提供了方法来根据类型动态创建对象,比如创建一个DataTable:

// 1.获取类型
Type t = Type.GetType("System.Data.DataTable,System.Data,Version=1.0.3300.0,  Culture=neutral,  PublicKeyToken=b77a5c561934e089");
// 2.创建对象
DataTable table = (DataTable)Activator.CreateInstance(t);

2.根据有参数的构造器创建对象

using System.Reflection;
using Test;
namespace ConsoleApp1
{
    public class Program
    {
        static void Main(String[] args)
        {
            //1.创建Test程序集下TestC类的Type对象
            Type t = Type.GetType("Test.TestC");    
            //2.把参数按照顺序放入一个Object数组中
            object[] constructParms = new object[] { "hello" };
            //3.创建TestC对象 强制转换
            TestC obj = (TestC)Activator.CreateInstance(t,constructParms);
            //4.打印值
            Console.WriteLine(obj._value);
        }
    }
}
//另一个命名空间
namespace Test
{
    public class TestC
    {
        public string _value;
        public TestC(string value)  //带参的构造函数
        {
            _value = value;
        }
    }
}

结果:

2)获取及动态调用方法

using System.Reflection;
using Test;
namespace ConsoleApp1
{
    public class Program
    {
        static void Main(String[] args)
        {
            // 获取类型
            Type t = Type.GetType("Test.TestC");
            // 创建对象
            TestC tc = (TestC)Activator.CreateInstance(t,new object[] {"cml"});
            // 获取类型t的方法AddValue
            MethodInfo mi = t.GetMethod("AddValue");
            // 调用方法
            object retu = mi.Invoke(tc, new object[] { "20230112" });
            // 打印方法返回值
            Console.WriteLine(retu);
        }
    }
}
//另一个命名空间
namespace Test
{
    public class TestC
    {
        //字段 
        public string value { get; set; }
        //构造函数
        public TestC(){ }
        public TestC(string v){     value = v;  }
        //自定义函数
        public string AddValue(string prefix)
        {
            return value + "+" + prefix;   //将value与该方法传入参数拼接
        }

    }
}

结果:

学习大佬文章https://www.cnblogs.com/vaevvaev/p/6995639.html

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值