C# 反射(Reflection)详解-Assembly

目录

介绍

反射中的运行时类型

Assembly案例

1.GetFiles

2.GetManifestResourceNames

3.GetReferencedAssemblies

4.GetTypes

5.Load

6.LoadFile

7.CreateInstance

8.LoadFrom

结束


介绍

微软关于反射文档:点击跳转

下面是微软官方的解释

System.Reflection 命名空间中的类与 System.Type 使你能够获取有关加载的程序集和其中定义的类型的信息,如接口和值类型(即结构枚举)。 可以使用反射在运行时创建、调用和访问类型实例。 有关反射的特定方面的主题,请参见本概述末的相关主题

公共语言运行时加载程序管理应用程序域,应用程序域构成具有相同应用程序范围的对象周围定义的边界。 此管理包括将每个程序集加载到相应的应用程序域中和控制每个程序集内的类型层次结构的内存布局。

程序集包含模块、模块包含类型,而类型包含成员。 反射提供封装程序集、模块和类型的对象。 可以使用反射动态地创建类型的实例,将类型绑定到现有对象,或从现有对象中获取类型。 然后,可以调用类型的方法或访问其字段和属性。 反射的典型用法如下所示:

  • 使用 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 使你能够检查特性,而无需创建它们的实例。 System.Reflection.Emit 命名空间的类提供一种专用形式的反射,使你能够在运行时生成类型。

还可以使用反射来创建称为类型浏览器的应用程序,它使用户能够选择类型,然后查看有关这些类型的信息。

反射还有其它用途。 JScript 等语言的编译器使用反射来构造符号表。 System.Runtime.Serialization 命名空间中的类使用反射来访问数据并确定要保存哪些字段。 System.Runtime.Remoting 命名空间中的类通过序列化间接使用反射。

反射中的运行时类型

反射提供类(如 Type 和 MethodInfo),用于表示类型、成员、参数和其它代码实体。 但使用反射时,你并不直接使用这些类,其中大部分类均是抽象的(Visual Basic 中为 MustInherit)。 相反,你使用由公共语言运行时 (CLR) 提供的类型。

例如,使用 C# typeof 运算符(Visual Basic 中为 GetType)获取 Type 对象时,该对象实际上是 RuntimeType。 RuntimeType 派生自 Type,并提供所有抽象方法的实现。

这些运行时类是 internal(Visual Basic 中为 Friend)。 它们没有与其基类分开记录,因为它们的行为由基类文档来描述。

Title说明
查看类型信息介绍 Type 类,并提供演示如何使用具有几个反射类的 Type 来获取有关构造函数、方法、字段、属性和事件的信息的代码示例。
反射类型和泛型类型说明反射如何处理泛型类型和泛型方法的类型参数和类型自变量。
反射的安全注意事项描述确定可以在何种程度上使用反射来发现类型信息和访问类型的规则。
动态加载和使用类型描述支持后期绑定的反射自定义绑定接口。
如何:将程序集加载到仅反射上下文中描述仅反射的加载上下文。 显示如何加载程序集、如何测试上下文以及如何检查应用到仅反射上下文中的程序集。
如何:使用 MetadataLoadContext 检查程序集内容使用 MetadataLoadContext 加载和检查程序集。
访问自定义特性演示如何使用反射来查询特性的存在和值。
指定完全限定的类型名称描述 Backus-Naur 形式 (BNF) 的完全限定类型名称的格式,以及指定特殊字符、程序集名称、指针、引用和数组所需的语法。
如何:使用反射挂接委托说明如何创建方法的委托并将委托挂钩到事件。 说明如何使用 DynamicMethod 在运行时创建事件处理方法。
发出动态方法和程序集说明如何生成动态程序集和动态方法。

在此,我将 Assembly,Module,ConstructorInfo,MethodInfo,FieldInfo,EventInfo,PropertyInfo,ParameterInfo,CustomAttributeData 这几个模块逐个写成帖子,每个帖子单独介绍其中的一个模块,其实每个模块的 API 接口非常多,我这里就不能一一介绍了,由于我们开发中反射用的并不多,我也不能真正理解每个接口的精髓之处,不足之处,还望各位大佬多多指点。

Assembly案例

先看一些简单的使用

namespace 计算3
{
    public class Program
    {
        static void Main(string[] args)
        {
            Assembly assem = Assembly.GetExecutingAssembly();
            Console.WriteLine("程序集的名称:" + assem.FullName);
            Console.WriteLine("版本:" + assem.GetName().Version);
            Console.WriteLine("程序集的位置:" + assem.CodeBase);
            Console.WriteLine("程序的完整路径:" + assem.Location);
            Console.WriteLine("此程序集的入口点:" + assem.EntryPoint);
            Type[] types = assem.GetTypes();
            foreach (var item in types)
            {
                Console.WriteLine("类:" + item.Name);
            }

            Console.ReadKey();
        }
    }
}

输出

1.GetFiles

获取此程序集清单的文件表中指定文件的 System.IO.FileStream。

官方文档:点击跳转

namespace 计算1
{
    class Program
    {
        static void Main(string[] args)
        {
            //获取默认应用程序域中的进程可执行文件
            Assembly assembly = Assembly.GetEntryAssembly();
            //获取程序集清单文件表中的文件,可以指定是否包括资源模块
            FileStream[] fileStreams = assembly.GetFiles();
            foreach (FileStream fileStream in fileStreams)
            {
                Console.WriteLine(fileStream.Name);
            }

            Console.ReadKey();
        }
    }
}

输出

2.GetManifestResourceNames

返回此程序集中的所有资源的名称。

官方文档:点击跳转

这个接口在 Winform 中比较合适,因为 Winform 中自带 Resources 文件

将程序的输出类型设置为控制台输出

namespace 计算2
{
    public partial class Form1 : Form
    {
        public Form1()
        {
            InitializeComponent();
        }

        private void Form1_Load(object sender, EventArgs e)
        {
            //获取默认应用程序域中的进程可执行文件
            Assembly assembly = Assembly.GetEntryAssembly();
            //此程序集中的所有资源的名称
            string[] names = assembly.GetManifestResourceNames();
            foreach (var name in names)
            {
                Console.WriteLine("资源:" + name);
            }
        }
    }
}

输出

3.GetReferencedAssemblies

 获取此程序集引用的所有程序集的 System.Reflection.AssemblyName 对象

官方文档:点击跳转

namespace 计算1
{
    class Program
    {
        static void Main(string[] args)
        {
            //获取默认应用程序域中的进程可执行文件
            Assembly assembly = Assembly.GetEntryAssembly();
            //此程序集引用的所有程序集
            AssemblyName[] assemblyNames = assembly.GetReferencedAssemblies();
            foreach (AssemblyName assemblyName in assemblyNames)
            {
                Console.WriteLine(assemblyName.FullName);//程序集全名
            }

            Console.ReadKey();
        }
    }
}

输出

4.GetTypes

获取此程序集中定义的类型。

官方文档:点击跳转

先看看当前程序集有那些类

namespace 计算1
{
    class Program
    {
        static void Main(string[] args)
        {
            //获取默认应用程序域中的进程可执行文件
            Assembly assembly = Assembly.GetEntryAssembly();
            //此程序集中定义的类型
            Type[] types = assembly.GetTypes();
            foreach (Type type in types)
            {
                Console.WriteLine(type.FullName);
            }

            Console.ReadKey();
        }
    }
}

 输出

5.Load

加载程序集

官方文档:点击跳转

namespace 计算1
{
    class Program
    {
        static void Main(string[] args)
        {
            Assembly assembly = Assembly.Load("计算1, Version=1.0.0.0, Culture=neutral, PublicKeyToken=null");
            Console.WriteLine(assembly.FullName);

            Console.ReadKey();
        }
    }
}

输出

6.LoadFile

加载指定路径上的程序集文件的内容。

官方文档:点击跳转

namespace 计算1
{
    class Program
    {
        static void Main(string[] args)
        {
            Assembly assembly = Assembly.LoadFile(Environment.CurrentDirectory + "\\ClassLibraryTest.dll");
            Console.WriteLine(assembly.FullName);

            Console.ReadKey();
        }
    }
}

输出:

7.CreateInstance

使用区分大小写的搜索,从此程序集中查找指定的类型,然后使用系统激活器创建它的实例。

官方文档:点击跳转

项目的结构,其中的 TList ,可以看我另一个帖子:点击跳转

namespace 计算1
{
    class Program
    {
        static void Main(string[] args)
        {
            Assembly assembly = typeof(TList<int>).Assembly;
            TList<int> list = (TList<int>)assembly.CreateInstance(typeof(TList<int>).FullName);
            list.Add(344);
            Console.WriteLine(list.Count);

            Console.ReadKey();
        }
    }
}

输出

8.LoadFrom

已知程序集的文件名或路径,加载程序集。

官方文档:点击跳转

namespace 计算1
{
    class Program
    {
        static void Main(string[] args)
        {
            Assembly assembly = Assembly.LoadFrom(Environment.CurrentDirectory + "\\ClassLibraryTest.dll");
            Console.WriteLine(assembly.FullName);

            Console.ReadKey();
        }
    }
}

输出

这里可以看出,LoadFrom 和 LoadFile 效果差不多,那么区别在哪里呢

1、Assembly.LoadFile只载入相应的dll文件,比如Assembly.LoadFile("abc.dll"),则载入abc.dll,假如abc.dll中引用了def.dll的话,def.dll并不会被载入。
Assembly.LoadFrom则不一样,它会载入dll文件及其引用的其他dll,比如上面的例子,def.dll也会被载入。

2、用Assembly.LoadFrom载入一个Assembly时,会先检查前面是否已经载入过相同名字的Assembly,比如abc.dll有两个版本(版本1在目录1下,版本2放在目录2下),程序一开始时载入了版本1,当使用Assembly.LoadFrom("2\\abc.dll")载入版本2时,不能载入,而是返回版本1。
Assembly.LoadFile的话则不会做这样的检查,比如上面的例子换成Assembly.LoadFile的话,则能正确载入版本2。
LoadFile:加载指定路径上的程序集文件的内容。LoadFrom: 根据程序集的文件名加载程序集文件的内容。

区别:

LoadFile 方法用来来加载和检查具有相同标识但位于不同路径中的程序集.但不会加载程序的依赖项。
LoadFrom 不能用于加载标识相同但路径不同的程序集。

结束

如果这个帖子对你有用,欢迎 关注 + 点赞 + 留言,谢谢

end

  • 3
    点赞
  • 11
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
两个现实中的例子: 1、B超:大家体检的时候大概都做过B超吧,B超可以透过肚皮探测到你内脏的生理情况。这是如何做到的呢?B超是B型超声波,它可以透过肚皮通过向你体内发射B型超声波,当超声波遇到内脏壁的时候就会产生一定的“回音”反射,然后把“回音”进行处理就可以显示出内脏的情况了(我不是医生也不是声学专家,不知说得是否准确^_^)。 2、地球内部结构:地球的内部结构大体可以分为三层:地壳、地幔和地核。地壳是固体,地核是液体,地幔则是半液半固的结构(中学地理的内容,大家还记得吧?)。如何在地球表面不用深入地球内部就知道其内部的构造呢?对,向地球发射“地震波”,“地震波”分两种一种是“横波”,另一种是“纵波”。“横波”只能穿透固体,而“纵波”既可穿透固体又可以穿透液体。通过在地面对纵波和横波的反回情况,我们就可以大体断定地球内部的构造了。 大家注意到这两个例子的共同特点,就是从一个对象的外部去了解对象内部的构造,而且都是利用了波的反射功能。在.NET中的反射也可以实现从对象的外部来了解对象(或程序集)内部结构的功能,哪怕你不知道这个对象(或程序集)是个什么东西,另外.NET中的反射还可以运态创建出对象并执行它其中的方法。 反射是.NET中的重要机制,通过反射,可以在运行时获得程序或程序集中每一个类型(包括类、结构、委托、接口和枚举等)的成员和成员的信息。有了反射,即可对每一个类型了如指掌。另外我还可以直接创建对象,即使这个对象的类型在编译时还不知道。 反射的用途: (1)使用Assembly定义和加载程序集,加载在程序集清单中列出模块,以及从此程序集中查找类型并创建该类型的实例。 (2)使用Module了解包含模块的程序集以及模块中的类等,还可以获取在模块上定义的所有全局方法或其他特定的非全局方法。 (3)使用ConstructorInfo了解构造函数的名称、参数、访问修饰符(如pulic 或private)和实现详细信息(如abstract或virtual)等。 (4)使用MethodInfo了解方法的名称、返回类型、参数、访问修饰符(如pulic 或private)和实现详细信息(如abstract或virtual)等。 (5)使用FiedInfo了解字段的名称、访问修饰符(如public或private)和实现详细信息(如static)等,并获取或设置字段值。 (6)使用EventInfo了解事件的名称、事件处理程序数据类型、自定义属性、声明类型和反射类型等,添加或移除事件处理程序。 (7)使用PropertyInfo了解属性的名称、数据类型、声明类型、反射类型和只读或可写状态等,获取或设置属性值。 (8)使用ParameterInfo了解参数的名称、数据类型、是输入参数还是输出参数,以及参数在方法签名中的位置等。 反射用到的命名空间: System.Reflection System.Type System.Reflection.Assembly 反射用到的主要类: System.Type 类--通过这个类可以访问任何给定数据类型的信息。 System.Reflection.Assembly类--它可以用于访问给定程序集的信息,或者把这个程序集加载到程序中。 System.Type类: System.Type 类对于反射起着核心的作用。但它是一个抽象的基类,Type有与每种数据类型对应的派生类,我们使用这个派生类的对象的方法、字段、属性来查找有关该类型的所有信息。 获取给定类型的Type引用有3种常用方式: ●使用 C# typeof 运算符。 Type t = typeof(string); ●使用对象GetType()方法。 string s = "grayworm"; Type t = s.GetType(); ●还可以调用Type类的静态方法GetType()。 Type t = Type.GetType("System.String"); 上面这三类代码都是获取string类型的Type,在取出string类型的Type引用t后,我们就可以通过t来探测string类型的结构了。 string n = "grayworm"; Type t = n.GetType(); foreach (MemberInfo mi in t.GetMembers()) { Console.WriteLine("{0}/t{1}",mi.MemberType,mi.Name); } Type类的属性: Name 数据类型名 FullName 数据类型的完全限定名(包括命名空间名) Namespace 定义数据类型的命名空间名 IsAbstract 指示该类型是否是抽象类型 IsArray 指示该类型是否是数组 IsClass 指示该类型是否是类 IsEnum 指示该类型是否是枚举 IsInterface 指示该类型是否是接口 IsPublic 指示该类型是否是公有的 IsSealed 指示该类型是否是密封类 IsValueType 指示该类型是否是值类型 Type类的方法: GetConstructor(), GetConstructors():返回ConstructorInfo类型,用于取得该类的构造函数的信息 GetEvent(), GetEvents():返回EventInfo类型,用于取得该类的事件的信息 GetField(), GetFields():返回FieldInfo类型,用于取得该类的字段(成员变量)的信息 GetInterface(), GetInterfaces():返回InterfaceInfo类型,用于取得该类实现的接口的信息 GetMember(), GetMembers():返回MemberInfo类型,用于取得该类的所有成员的信息 GetMethod(), GetMethods():返回MethodInfo类型,用于取得该类的方法的信息 GetProperty(), GetProperties():返回PropertyInfo类型,用于取得该类的属性的信息 可以调用这些成员,其方式是调用Type的InvokeMember()方法,或者调用MethodInfo, PropertyInfo和其他类的Invoke()方法。 查看类中的构造方法: NewClassw nc = new NewClassw(); Type t = nc.GetType(); ConstructorInfo[] ci = t.GetConstructors(); //获取类的所有构造函数 foreach (ConstructorInfo c in ci) //遍历每一个构造函数 { ParameterInfo[] ps = c.GetParameters(); //取出每个构造函数的所有参数 foreach (ParameterInfo pi in ps) //遍历并打印所该构造函数的所有参数 { Console.Write(pi.ParameterType.ToString()+" "+pi.Name+","); } Console.WriteLine(); }

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

熊思宇

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值