c#之反射详解

总目录


文章目录

  • 总目录
  • 一、反射是什么?
    • 1、C#编译运行过程
    • 2、反射与元数据
    • 3、反射的优缺点
  • 二、反射的使用
    • 1、反射相关的类和命名空间
      • 1、System.Type类的应用
      • 2、System.Activator类的应用
      • 3、System.Reflection.Assembly类的应用
      • 4、System.Reflection.Module类的应用
      • 5、System.AppDomain类的应用
      • 6、dynamic 在反射中的应用
    • 2、反射的应用
      • 1、 数据库辅助类反射
  • 结语


一、反射是什么?

1、C#编译运行过程

说到反射,就不得不说一下C#编译运行过程:

  • 首先我们在VS点击编译的时候,就会将C#源代码编译成程序集

程序集以可执行文件 (.exe) 或动态链接库文件 (.dll) 的形式实现

  • 程序集中包含有Microsoft 中间语言 (MSIL) 和必需的元数据。

元数据存储以下信息:

  • 程序集的说明:标识(名称、版本、区域性、公钥)、导出的类型、该程序集所依赖的其他程序集、运行所需的安全权限。
  • 类型的说明:名称、可见性、基类和实现的接口、成员(方法、字段、属性、事件、嵌套的类型)。
  • 特性:修饰类型和成员的其他说明性元素。
  • 在执行时,实时 (JIT) 编译器将 MSIL 转换为本机代码

运行 Microsoft 中间语言 (MSIL) 前,必须根据公共语言运行时将其编译为目标计算机基础结构的本机代码。

  • 运行代码

公共语言运行时提供启用要发生的托管执行的基础结构以及执行期间可使用的服务

2、反射与元数据

反射 来自 System.Reflection命名空间,它可以读取程序集中的元数据,利用元数据创建对象,从而实现各种功能。

区分 反射 与 反编译,反射读取的是元数据,反编译读取的IL代码

3、反射的优缺点

  • 优点:提高了程序的灵活性和扩展性,降低耦合度
  • 缺点:由于反射多了一道程序,性能上相较于直接代码要慢

对于一些大型的项目,该用反射的地方还是要用,即使牺牲一点性能

二、反射的使用

1、反射相关的类和命名空间

  • 反射命名空间
using System.Reflection;
  • 反射相关的类
System.Type
System.AppDomain
System.Activator
System.Reflection.Assembly
System.Reflection.Module


System.Reflection.ConstructorInfo
System.Reflection.ParameterInfo
System.Reflection.MethodInfo
System.Reflection.PropertyInfo
System.Reflection.FieldInfo
System.Reflection.MemberInfo

1、System.Type类的应用

  • Type类中的基本属性
    在这里插入图片描述

FullName :获取该类型的完全限定名称,包括其命名空间,但不包括程序集

    • Type 中的 Assembly属性
        static void Main(string[] args)
        {
            Type type1 = typeof(User);

            //属性Assembly
            //获取声明该类型的 Assembly。
            //对于泛型类型,则获取定义该泛型类型的 Assembly。
            Assembly assembly = type1.Assembly;
            Console.WriteLine($"{assembly.FullName}");
            Console.ReadLine();
        }

通过Type对象我们也可以获取得到在其中申明了该类型的程序集,至于Assembly的用途将在Assembly那一节进行详细介绍

    • Type 中的 AssemblyQualifiedName 属性 和 FullName属性

在这里插入图片描述
上图中:类型的程序集限定名 的格式中的 全名称部分 即是 Type中的FullName属性值
FullName 获取该类型的完全限定名称,包括其命名空间,但不包括程序集

  • Type类的常用的方法
    // UserInfo类是为 介绍 Type类中常用方法 而准备的对象
    public class UserInfo
    {
        private int _num = 0;

        public string Phone = "1311111111";
        
        public string Name { get; set; }
        
        public string Address { get; set; }

        public UserInfo()
        {
            Console.WriteLine("UserInfo默认构造函数");
        }

        public UserInfo(string name)
        {
            Console.WriteLine($"UserInfo参数化构造函数:{name}");
        }

        public int PublicMethod()
        {
            return int.MinValue;
        }

        internal void InternalMethod ()
        { 
        
        }
        private void PrivateMethod()
        {
 
        }
    }
	class Program
    {
        static void Main(string[] args)
        {
            UserInfo userInfo = new UserInfo();
            //【*】通过System.Object中的GetType()获取Type实例
            Type type = userInfo.GetType();

            //GetConstructors()获取所有的公共的构造函数
            ConstructorInfo[] constructorInfos= type.GetConstructors();

            foreach (var item in constructorInfos)
            {
                //GetParameters()获取指定方法或构造函数的参数
                ParameterInfo[] parameterInfos = item.GetParameters();
                foreach (var pi in parameterInfos)
                {
                    Console.WriteLine($"{item.Name}:{pi.Name}:{pi.ParameterType}");
                }              
            }

            //获取当前Type 实例的所有Public方法
            MethodInfo[] methodInfos = type.GetMethods();
            foreach (var item in methodInfos)
            {               
                Console.WriteLine($"{type.Name}类型中有:{item.Name}方法,返回类型为{item.ReturnType}");
            }

            //获取当前Type 实例的所有Public属性
            PropertyInfo[] propertyInfos = type.GetProperties();
            foreach (var item in propertyInfos)
            {
                Console.WriteLine($"{type.Name}类中有 属性-{item.Name} 类型为-{item.PropertyType}");
            }

            //获取当前Type 实例的所有Public字段
            FieldInfo[] fieldInfos = type.GetFields();
            foreach (var item in fieldInfos)
            {
                Console.WriteLine($"{type.Name}类中有 字段-{item.Name} 类型为-{item.FieldType}");
            }

            MemberInfo[] memberInfos = type.GetMembers();
            foreach (var item in memberInfos)
            {
                Console.WriteLine($"{type.Name}类中有 成员名称-{item.Name} 类型为-{item.MemberType}");
            }

            Console.ReadLine();
        }
    }

来张 代码贴图可能更为直观
在这里插入图片描述
由上可知,Type类给我们提供了很全面的 类型的 元数据的 获取方式。

    • BindingFlags
            Type type1 = Type.GetType("ConsoleApp1.UserInfo");
            //GetMembers 中传入 BindingFlags 相当于是对成员信息进行一个过滤
            //BindingFlags 不仅仅是GetMembers 专有,很多方法中都可以传入BindingFlags进行过滤

            //BindingFlags 是位标志枚举,可使用 | & ^ 等运算符 | 表示取并集,& 表示取交集,^ 表示取差集
            //BindingFlags.Public 表示公共成员
            //BindingFlags.NonPublic 表示非公共成员
            //BindingFlags.Instance 表示实例成员
            //BindingFlags.Static 表示静态成员
            MemberInfo[] memberInfos = type1.GetMembers(BindingFlags.NonPublic|BindingFlags.Instance);
            foreach (var item in memberInfos)
            {
                Console.WriteLine($"BindingFlags.NonPublic|BindingFlags.Instance(实例非公共成员)名称:{item.Name}");
            }

BindingFlags.Instance 和BindingFlags.Static :实例成员是相对于静态成员而言的,多数情况下我们都省略了BindingFlags 这个参数,少数需要筛选成员的时候,传入该参数

  • 获取Type 实例对象的三个方法

在这里插入图片描述

2、System.Activator类的应用

//Activator类主要用于创建对象的实例
Type type = typeof(UserInfo);
UserInfo userInfo=(UserInfo)Activator.CreateInstance(type);

3、System.Reflection.Assembly类的应用

  • 对于程序集的限定名称使用小结

在这里插入图片描述
由上图代码可知:

  • 程序集的显示名称,可通过Assembly.FullNameAssembly.GetName().FullName(即AssemblyName.FullName) 两种方式获取,这种获取的名称,一般是作为 Assembly.Load()的标准参数值
  • 类型的程序集限定名,可通过Type类中的AssemblyQualifiedName属性获取(通常作为Type.GetType()方法中的参数值), 相较于Assembly.FullName,名称格式上多了 Type.FullName 这一部分
  • Assembly类中的常用方法
    • Assembly.Load()方法接收一个String或AssemblyName类型作为参数,这个参数需要程序集的强名称
  • 程序集的强名称:是程序集的FullName(具有名称,版本,语言,公钥标记);
  • 程序集的弱命名:只有程序集名称而没有版本,语言和公钥标记;平常我们创建的一个类库,如果没有特殊操作都属于是是弱名称程序集
  • Load(“强名称程序集”)查找程序集的顺序:首先它会去全局程序集缓存查找,然后到应用程序的根目录查找,最后会到应用程序的私有路径查找。
  • Load(“弱名称程序集”)查找程序集的顺序:首先到应用程序的根目录查找,最后会到应用程序的私有路径查找。
    • Assembly.LoadFrom() 根据程序集的文件名或路径,加载程序集;这个方法会加载此程序集引用的其他程序集
    • Assembly.LoadFile() 加载指定路径上的程序集文件内容,和上面方法的不同之处是这个方法不会加载此程序集引用的其他程序集

在这里插入图片描述
程序集加载的三种方式,可以在项目中添加该程序集的引用后使用,也可在未添加该程序集的情况下使用(某些情况下),这样就极大的丰富的项目的灵活性和扩展性

    • 反射:创建对象的三种方式

在这里插入图片描述

    • 反射: 调用构造函数创建对象详解

在这里插入图片描述

    • 常用方法(包含Type类中的方法)

在这里插入图片描述
Invoke 调用静态方法,对象可以为null ,形如
methodInfo.Invoke(null, new object[] { "Activator.CreateInstance + type.GetMethod" });

		Assembly assembly1 = Assembly.Load("ClassLibrary1");

            // GetTypes 获取程序集中的所有类型
            Type[] types = assembly1.GetTypes();
            foreach (var item in types)
            {
                if (item.FullName == "ClassLibrary1.Class1")
                {
                    Activator.CreateInstance(item);
                }
            }
            // 通过反射获取方法,然后执行
            // GetType("类型全名",是否引发异常,是否忽略大小写)
            Type type = assembly1.GetType("ClassLibrary1.Class1", false,false);
            object objt = Activator.CreateInstance(type);
            MethodInfo methodInfo = type.GetMethod("Show");

            //通过GetParameters获取方法的参数信息
            ParameterInfo[] parameterInfos = methodInfo.GetParameters();

            // 通过Invoke 调用方法
            // 对于方法而言,Invoke 至少需要传入两个参数,一个参数为 对象实例object,一个参数为方法参数列表 new object[]
            methodInfo.Invoke(objt,new object[] { "Activator.CreateInstance + type.GetMethod" });

            // 通过GetProperty 获取指定名称的属性
            PropertyInfo propertyInfo = type.GetProperty("Name");
            // SetValue 给属性赋值
            propertyInfo.SetValue(objt,"测试类");

            // 通过GetField 获取指定名称的字段
            FieldInfo fieldInfo = type.GetField("_num",BindingFlags.NonPublic);
            // SetValue 给属性赋值
            fieldInfo.SetValue(objt,12);

            Console.ReadLine();

4、System.Reflection.Module类的应用

暂无

5、System.AppDomain类的应用

一个AppDomain可以包含N个Assembly,一个Assembly可以包含N个Module,而一个Module可以包含N个Type.

暂无

6、dynamic 在反射中的应用

变量可以具有不同的编译时和运行时类型。 编译时类型是源代码中变量的声明或推断类型。 运行时类型是该变量所引用的实例的类型。

        static void Main(string[] args)
        {
            Type type = typeof(User);
            object o_user = Activator.CreateInstance(type);
            //o_user.Show() 
            //不可能通过o_class1 调用Show
            dynamic d_user = Activator.CreateInstance(type);
            d_user.Show("sss");
            //可以通过d_user 调用方法Show

            //其实o_user 和 d_user得到结果都是一样的,
            // 但是因为 object 时编译时类型,object本身没有Show方法,因此调用会报错
            // 而dynamic 是运行时类型,编译状态下会绕过编译器的检查,直到真正运行后才确定其数据类型
            Console.ReadLine();
        }

2、反射的应用

1、 数据库辅助类反射

  • 原始情况下,我们写一个简单的SqlServer帮助类
    public class SqlServerHelper
    {
        private static readonly string _connectionString = "server=.;database=test;uid=sa;pwd=123";
        private SqlConnection _sqlConnection;

        //执行增删改
        public int ExecDML(string sql)
        {
            using (_sqlConnection = new SqlConnection(_connectionString))
            {
                _sqlConnection.Open();
                SqlCommand sqlCommand = new SqlCommand(sql,_sqlConnection);
                return sqlCommand.ExecuteNonQuery();
            }
        }
        //执行查询
        public DataSet ExecDQL(string sql)
        {
            using (_sqlConnection=new SqlConnection(_connectionString))
            {
                SqlDataAdapter sqlDataAdapter = new SqlDataAdapter(sql,_sqlConnection);
                DataSet dataSet = new DataSet();
                sqlDataAdapter.Fill(dataSet);
                return dataSet;
            }
        }
    }

	//调用SqlServerHelper 中方法
    class Program
    {
        static void Main(string[] args)
        {
            SqlServerHelper sqlServerHelper = new SqlServerHelper();
            var data= sqlServerHelper.ExecDQL("select * from userinfo");
            var userName = data.Tables[0].Rows[0][1];
            Console.WriteLine(userName.ToString());
            Console.ReadLine();
        }
    }
  • 选择 反射+配置文件的方式 实现,具体实现步骤如下:
    • 1 创建一个接口
    interface IDbHelper
    {
        int ExecDML(string sql);
        DataSet ExecDQL(string sql);
    }
    • 2 增加配置文件

在这里插入图片描述

    • 3 实现接口
public class SqlServerHelper : IDbHelper
    {
        private static readonly string _connectionString = ConfigurationManager.ConnectionStrings["DbConnection"].ToString();
        
        public int ExecDML(string sql)
        {
            using (SqlConnection sqlConnection = new SqlConnection(_connectionString))
            {
                sqlConnection.Open();
                SqlCommand sqlCommand = new SqlCommand(sql, sqlConnection);
                return sqlCommand.ExecuteNonQuery();
            }
        }

        public DataSet ExecDQL(string sql)
        {
            using (SqlConnection sqlConnection = new SqlConnection(_connectionString))
            {
                SqlDataAdapter sqlDataAdapter = new SqlDataAdapter(sql, sqlConnection);
                DataSet dataSet = new DataSet();
                sqlDataAdapter.Fill(dataSet);
                return dataSet;
            }
        }
    }

这里主要是通过读取配置文件,确定数据库连接字符串:
ConfigurationManager.ConnectionStrings["DbConnection"].ToString();

    • 4 通过反射+配置文件 调用 数据库执行语句的方法
        static void Main(string[] args)
        {
            string fullName = $"DbHelper.{ConfigurationManager.AppSettings["DbType"].ToString()}";
            IDbHelper dbHelper = (IDbHelper)Assembly.Load("DbHelper").CreateInstance(fullName);
            var data = dbHelper.ExecDQL("select * from userinfo");
            var userName = data.Tables[0].Rows[0][1];
            Console.WriteLine(userName.ToString());
            Console.ReadLine();
        }
  • 从变更使用的数据库为MySql,分析两种方式应对需求的变动
    • 对于原始方法我们需要再重写一个数据库帮助类(如MySqlHelper),然后重新生成帮助类类库文件,最后该调用的代码
    • 如果按照反射+配置文件的方式实现,我们需要实现MySqlHelper类,然后重新生成类库,替换dll文件即可

这个案例只是一个初级的应用,便于理解反射;
反射的应用场景有:IOC容器,MVC框架,ORM,AOP等,因此理解好反射,对于上述知识点的掌握也是有帮助


结语

以上就是本文的内容,希望以上内容可以帮助到您,如文中有不对之处,还请批评指正。


参考资料:
Type类
AssemblyName 类
C#高级–反射详解
C#通过反射调用类及方法
深入浅出C#反射(Reflection)原理和应用场景
C#语法——反射,架构师的入门基础。
C#基础知识学习之 ☀️ | 反射(Reflection) 的含义和用法
最全的 .NET(C#) 反射使用总结
【C#入门详解16】-反射、依赖注入
C#反射-Assembly.Load、LoadFrom与LoadFile进阶

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/5910.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

SpringBoot 整合RabbitMq 自定义消息监听容器来实现消息批量处理

SpringBoot 整合RabbitMq 自定义消息监听容器来实现消息批量处理前言添加依赖配置文件编写监听器创建SimpleRabbitListenerContainerFactory发送消息前言 RabbitMQ是一种常用的消息队列,Spring Boot对其进行了深度的整合,可以快速地实现消息的发送和接收…

PCB模块化设计16——RS232,RS485接口模块PCB布局布线设计规范

目录PCB模块化设计16——RS232,RS485接口模块PCB布局布线设计规范RS232接口模块1、接口概述2、接口电路 原理图的EMC设计3、连接器设计4、线缆设计5、RS-232常规管脚定义:6、RS-232知识要点RS485接口模块1、原理图设计方案1、RS485接口6KV防雷电路设计方…

c语言程序笔记(1)

C语言笔记&#xff08;1&#xff09;——B站翁恺视频 程序框架 #include <stdio.h> int main() {//printf("hello world!\n");return 0; }1、变量与常量。 例子1&#xff1a; #include <stdio.h> int main() {printf("1234%d",1234);return …

图解LeetCode——合并两个有序链表

如果你喜欢这篇文章的话&#xff0c;请给作者点赞关注哟&#xff0c;你的支持是我不断前进的动力&#xff01; 目录 题目描述&#xff1a; 解法&#xff1a; 完整代码&#xff1a; 结果 题目链接&#xff1a;力扣 题目描述&#xff1a; 将两个升序链表合并为一个新的 升序…

2017世界互联网领先成果来了 光量子计算机

演讲者&#xff1a;陆朝阳中国科学技术大学教授 发布了世界上首台超越早期经典计算机的光量子计算机 陆朝阳&#xff1a;很高兴向大家报告中国科学院在量子计算这个领域取得的基础性的研究成果。 我们知道50多年以来摩尔定律一直见证着计算机的更新换代&#xff0c;之前每过18个…

【新2023Q2模拟题JAVA】华为OD机试 - 绘图机器

最近更新的博客 华为od 2023 | 什么是华为od,od 薪资待遇,od机试题清单华为OD机试真题大全,用 Python 解华为机试题 | 机试宝典【华为OD机试】全流程解析+经验分享,题型分享,防作弊指南华为od机试,独家整理 已参加机试人员的实战技巧本篇题解:绘图机器 题目 绘图机器的绘…

读书笔记-纳瓦尔宝典-2023.04.01

重点 财富 如何构造高价值信息 判断力 何为幸福 启发 最近看了这本书的大部分内容&#xff0c;感悟颇多&#xff0c;及时记录下来。 因为是快速阅读&#xff0c;还未做深入思考和实践&#xff0c;但对总体的内容有一个大致把握&#xff0c;未来会结合行动反复阅读和思考&…

python画爱心代码

前几天在网上看到了一个画爱心的教程&#xff0c;就是在 Python里面画一个爱心&#xff0c;但是我在网上找到的代码不是很好用&#xff0c;所以我就自己写了一遍。 首先我们先创建一个新的 python文件。新建一个 python文件夹&#xff0c;将我们之前的那个 python文件夹复制到这…

蓝桥杯·3月份刷题集训Day03

本篇博客旨在记录自已打卡蓝桥杯3月份刷题集训&#xff0c;同时会有自己的思路及代码解答希望可以给小伙伴一些帮助。本人也是算法小白&#xff0c;水平有限&#xff0c;如果文章中有什么错误之处&#xff0c;希望小伙伴们可以在评论区指出来&#xff0c;共勉&#x1f4aa;。 文…

2021年第十二届蓝桥杯省赛Java B组真题及详细题解

A试题 : ASC【填空题】 本题总分&#xff1a; 5 分 【1、问题描述】 已知大写字母 A 的 ASCII 码为 65&#xff0c;请问大写字母 L 的 ASCII 码是多少&#xff1f; 【2、答案提交】 这是一道结果填空的题&#xff0c;你只需要算出结果后提交即可。本题的结果为一个整数&#…

二十、Javascript API(一)

1. Atomics和SharedArrayBuffer 多个上下文访问 SharedArrayBuffer时&#xff0c;如果同时对缓冲区执行操作&#xff0c;就可能出现资源争用问题。Atomics API 通过强制同一时刻只能对缓冲区执行一个操作&#xff0c;可以让多个上下文安全地读写一个SharedArrayBuffer。 1.1 …

Android HTTP请求方式

1.HttpClient使用流程 基本流程&#xff1a; 2.HttpClient使用示例 1&#xff09;使用HttpClient发送GET请求 直接贴下简单的发送Get请求的代码&#xff1a; public class MainActivity extends Activity implements OnClickListener { private Button btnGet; private WebV…

STM-32:GPIO 输出-点亮LED-流水灯-蜂鸣器

目录一、GPIO1.1GPIO简介1.2GPIO 硬件解析1.2.1保护二极管1.2.2 P-MOS、N-MOS 管1.2.3数据输入输出寄存器1.2.4复用功能输出1.2.5模拟输入输出1.3GPIO 的工作模式1.3.1 输入模式 (模拟/浮空/上拉/下拉)1.3.2 输出模式 (推挽/开漏)1.3.3 复用功能 (推挽/开漏)1.3.4 小结二、GPIO…

ChatGPT将引发大量而普遍的网络安全隐患

ChatGPT是一个基于人工智能的语言生成模型&#xff0c;它可以在任何给定的时间&#xff0c;使用自然语言生成技术&#xff0c;生成文本、对话和文章。它不仅可以被用来编写文本&#xff0c;还可以用来编写语言、生成图像和视频。目前&#xff0c; ChatGPT已广泛应用于语言翻译、…

【数据结构篇】-树(共计两万字,你真的搞懂了它吗)

友情链接&#xff1a;【数据结构与算法】首篇 - 思维导图 - 各部分内容目录 文章目录&#x1f680;树&#x1f6a2;一、树的原理精讲&#xff08;一&#xff09;树的定义&#xff08;二&#xff09;基本术语&#xff08;三&#xff09;树的性质&#x1f6a2;二、树的存储结构&a…

C++ STL:stack和queue的使用和底层实现

目录 一. 什么是stack和deque 二. stack和queue的使用方法 2.1 stack的常用接口 2.2 queue的常用接口 三. stack和queue的底层实现原理 3.1 容器适配器 3.2 deque&#xff08;双端队列&#xff09;的概念及抽象结构 3.3 deque的底层实现结构 3.4 deque的优缺点 —— 为…

try... excpet BaseException(异常处理捕获)

try ...except 是最常见的捕获处理异常的结构&#xff0c;其主要作用是将可能出现问题的代码块用try &#xff1a;包裹起来&#xff0c;不至于出现错误让程序崩溃&#xff0c;无法执行下去常见的try ...excpet 的结构有三种try&#xff1a;pass except BaseException as e &…

Azure SQL基础到实战(2)-部署

目录Azure 上的数据库服务的演变Azure SQL 部署选项Azure 虚拟机上的 SQL ServerIaaS 与PaaS无版本数据库服务SQL 托管实例SQL 数据库弹性数据库池Azure 上的数据库服务的演变 Azure SQL 是 Microsoft 作为 Azure 云计算平台的一部分提供的云数据库产品/服务。 与其他版本的 S…

含光热电站、有机有机朗肯循环、P2G的综合能源优化调度(Matlab代码实现)

&#x1f4a5;&#x1f4a5;&#x1f49e;&#x1f49e;欢迎来到本博客❤️❤️&#x1f4a5;&#x1f4a5; &#x1f3c6;博主优势&#xff1a;&#x1f31e;&#x1f31e;&#x1f31e;博客内容尽量做到思维缜密&#xff0c;逻辑清晰&#xff0c;为了方便读者。 ⛳️座右铭&a…

算法---扫雷游戏

题目 让我们一起来玩扫雷游戏&#xff01; 给你一个大小为 m x n 二维字符矩阵 board &#xff0c;表示扫雷游戏的盘面&#xff0c;其中&#xff1a; ‘M’ 代表一个 未挖出的 地雷&#xff0c; ‘E’ 代表一个 未挖出的 空方块&#xff0c; ‘B’ 代表没有相邻&#xff08;…