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.ModuleSystem.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 调用Showdynamic 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进阶

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

如若内容造成侵权/违法违规/事实不符,请联系瞎胡闹网进行投诉反馈,一经查实,立即删除!

相关文章

9. 解谜游戏

目录 题目 Description Input Notes 思路 暴力方法 递归法 注意事项 C代码(递归法) 关于DFS 题目 Description 小张是一个密室逃脱爱好者,在密室逃脱的游戏中,你需要解开一系列谜题最终拿到出门的密码。现在小张需要打…

【Windows10】利用分区助手扩展C盘分区

文章目的:   在开始对C盘进行空间分配时,配额空间过小,后续使用希望扩展C盘的空间大小 分区助手百度网盘下载链接:link   提取码:go16 1、下载安装好分区助手后打开,点击右上角Tools,再点击…

将C盘分区部分容量分配给其他分区

最近在学习OpenStack相关的实验,硬盘容量需要分配一百多两百个G。电脑用久了,不知不觉磁盘也将要挤满的感觉,除C盘外。就想着把C盘的一些容量转到E盘里。 以下为具体步骤: 下载分区工具 选择合适的分区工具下载。我使用的是 win1…

Windows: 如何调整C盘分区大小

最开始安装系统,做分区时给C分区的大小一般都太小,后面发现不够时却无从下手。 Windows磁盘管理做不了这个事。因为它只能扩展后面为空白的。但D分区已有数据,无法扩展。 所以,我们得借助另一个工具:DiskGenius. 操…

【八股】2023秋招八股复习笔记5(计算机网络-CN)

文章目录 八股目录目录1、应用层 & HTTP一些http题HTTPS 加密原理(问过)HTTP/1.1 新特性HTTP/2.0 与 RPC(问过)GET 和 POST 比较 2、传输层 & TCPTCP三次握手 & 四次挥手(问过)为什么每次TCP 连…

基于vue和element的脚手架【vue-element-admin 和vue-element-plus-admin 】

vue-element-admin vue-element-admin 是一个后台前端解决方案,它基于 vue 和 element-ui实现 介绍 | vue-element-adminA magical vue adminhttps://panjiachen.github.io/vue-element-admin-site/zh/guide/ vue-element-plus-admin vue-element-plus-admin 是一…

keras深度学习框架通过简单神经网络实现手写数字识别

背景 keras深度学习框架,并不是一个独立的深度学习框架,它后台依赖tensorflow或者theano。大部分开发者应该使用的是tensorflow。keras可以很方便的像搭积木一样根据模型搭出我们需要的神经网络,然后进行编译,训练,测试…

脑优化全集

1电脑优化全集 一、系统优化设置 1、删除Windows强加的附件: a.用记事本NOTEPAD修改sysoc.inf(在windiws/inf文件夹),用查找/替换功能,在查找框中输入,hide,7(一个英文逗号紧跟hide,一个英文逗…

个人永久性免费-Excel催化剂功能第49波-标准数据结构表转报表样式结果

中国的企业信息化,已经过去了20年,企业里也产生了大量的数据,IT技术的信息化管理辅助企业经营管理也已经得到广泛地认同,现在就连一个小卖部都可以有收银系统这样的信息化管理介入。但同时也有一个很现实的问题,不是所…

个人永久性免费-Excel催化剂功能第37波-把Sqlserver的强大分析函数拿到Excel中用...

本人一直钟情于使用Sqlserver数据库的一大原因是其提供了非常好用、高效的数据分析函数(窗口函数),可以在做数据清洗和数据分析场合等多个场景使用。只需简单的一个函数即可做出常规SQL语句很难以实现的效果。这么好用的函数,如今…

个人永久性免费-Excel催化剂功能第43波-文本处理类函数增强

Excel的函数有400多个,真正常用的50多个,而常有的文本处理类函数也不多,不是因为文本类处理简单,而是Excel真的有点挤牙膏式的每个版本更新那么几个小函数,普通用户等得急切,但实际上这些小函数&#xff0c…

个人永久性免费-Excel催化剂功能第30波-工作表快捷操作(批量创建、命名、排序、工作表目录)...

日常使用Excel过程中,最多的操作无外乎单元格和工作表的操作,单元格的操作在前面已经有详细的辅助功能提供,此篇提供工作表相关的操作。这两项的操作若能有提速,日常大量的工作叠加起来真是省下不少时间。 文章出处说明 原文在简书…

自定义镜像上传阿里云

目录 一、Docker制作jdk镜像 jdkv1.0版本 ① 编写Dockerfile文件 ② 执行Dockerfile文件,初次依赖镜像的时候会下载相应的镜像 ③ 查看镜像 ④ 创建并启动容器 二、alpine 制作jdk镜像 jdkv2.0的版本 1.alpine Linux 简介 2.基于 alpine 制作 j…

个人永久性免费-Excel催化剂功能第28波-工作薄瘦身,安全地减少非必要冗余

Excel催化剂在完善了数据分析场景的插件需求后,决定再补充一些日常绝大多数Excel用户同样可以使用到的小功能,欢迎小白入场,在不违背太多Excel最佳实践的前提下,Excel催化剂乐意为广大Excel用户们增添有价值和高频使用的快捷操作类…

计算机英语总结800,高三英语教师工作总结800字(通用5篇)

高三英语教师工作总结800字(通用5篇) 难忘的工作生活已经告一段落了,回顾这段时间的工作,相信你有很多感想吧,来为这一年的工作写一份工作总结吧。大家知道工作总结的格式吗?下面是小编精心整理的高三英语教师工作总结800字(通用5…

如何提高英语听力(内容摘自NECCS)+ 乘法表

乘法表 print(\n.join([ .join([%s*%s%-2s%(y,x,x*y) for y in range(1,x1)]) for x in range(1,10)])) 如何提高英语听力 很喜欢这篇关于提高英语听力的文章,所以收藏下来和大家一同分享一下 人走路时要用两条腿,没有任何人会觉得走路费劲。可如果让人…

张晓楠讲如何提高英语听力

张晓楠:现在是中央电视台财经频道(CCTV-2)的记者。曾任北京电视台青少频道主持人。原新浪网财经频道的主持人兼记者。毕业于美国哥伦比亚大学,主修金融方向,获公共管理硕士学位;曾在纽约摩根大通银行、瑞士信贷投资银行…

2023年6月GESP C++ 四级试卷解析

一、单选题(每题2分,共30分) 1.高级语言编写的程序需要经过以下( )操作,可以生成在计算机上运行的可执行代码。 A.编辑 B.保存 C.调试 D.编译 【答案】D 【考纲知识点】编程环境(一级) 【解析】本题…

红警 1 游戏开源,代码非常规范,网友:秀色可餐

作者 | 程序员的那些事 来源 | 程序员的那些事(id:iProgrammer) 最后有一个小测试!测测你是不是红警老玩家! EA 部分开源红警啦! 5 月 27 日,知名游戏公司 EA 在 GitHub 上搞了个大新闻&#xf…

《网络安全等级保护基本要求》(GB/T 22239-2019)标准解读

关键词: 等级保护对象; 安全通用要求; 安全扩展要求 中图分类号:TP309 文献标志码:A 文章编号:1671-1122(2019)02-0077-08 Baseline for Classified Protection of Cybersecurity (GB/T 22239-2019) Standard Interpretation MA Li1, ZHU Guobang2, L…