C#进阶-反射的详解与应用

一、反射的概念

反射是.NET框架提供的一个功能强大的机制,它允许程序在运行时检查和操作对象的类型信息。通过使用反射,程序可以动态地创建对象、调用方法、访问字段和属性,无需在编译时显式知道类型信息。在.NET中,所有类型的信息最终都是存储在元数据中的。反射就是.NET提供的一组API,允许我们在运行时访问这些元数据,从而获得关于程序集、模块、类型、成员等的详细信息。

反射概念图:

在这里插入图片描述


二、反射的应用

反射的应用非常广泛,包括动态类型创建、动态方法调用、属性访问、自定义属性处理等。
我们可以根据反射的对象不同,分为两类:字段反射和方法反射。

1、字段反射

字段反射是指在运行时使用反射API来访问和修改对象的字段。这在需要动态访问对象的内部字段时非常有用,尤其是在不具有对象类型显式知识的情况下。

① 获取字段值

假设我们想要获取一个对象的字段值,可以使用FieldInfo.GetValue方法。仍然以User类为例,假设我们想获取Name字段的值。

举个例子:

using System;public class User
{public string Name = "Initial Name";
}class Program
{static void Main(){User user = new User();Type userType = typeof(User);var fieldName = "Name";// 获取User类的Name字段var fieldInfo = userType.GetField(fieldName);// 获取User实例的Name字段值var value = fieldInfo.GetValue(user);Console.WriteLine(value); // 输出: Initial Name}
}

可以看到我们通过反射的方式,将Name属性的值成功输出。


② 修改字段值

假设有一个User类,包含一个Name字段。我们想要在运行时修改某个User实例的Name字段值。

举个例子:

using System;public class User
{public string Name;
}class Program
{static void Main(){User user = new User();Type userType = typeof(User);var fieldName = "Name";// 获取User类的Name字段var fieldInfo = userType.GetField(fieldName);// 设置User实例的Name字段值fieldInfo.SetValue(user, "Damon");Console.WriteLine(user.Name); // 输出: Damon}
}

上述代码演示了如何使用字段反射来动态修改User实例的Name字段。首先,通过typeof(User)获取User类型的Type对象。然后,使用Type对象的GetField方法获取Name字段的FieldInfo对象。最后,使用FieldInfo对象的SetValue方法来修改字段的值。


③ 检查字段属性

反射还允许我们检查字段的属性,例如判断字段是否为公有(Public)、私有(Private)、静态(Static)等。这可以通过FieldInfo对象的属性来实现,例如IsPublicIsPrivateIsStatic等。

举个例子:

using System;public class User
{public string Name;private int age;public static string Category = "General";
}class Program
{static void Main(){Type userType = typeof(User);// 获取并检查字段属性var publicField = userType.GetField("Name");Console.WriteLine($"Name is Public: {publicField.IsPublic}"); // 输出: Truevar privateField = userType.GetField("age", System.Reflection.BindingFlags.NonPublic | System.Reflection.BindingFlags.Instance);Console.WriteLine($"Age is Private: {privateField.IsPrivate}"); // 输出: Truevar staticField = userType.GetField("Category");Console.WriteLine($"Category is Static: {staticField.IsStatic}"); // 输出: True}
}

在上述代码示例中,我们展示了如何使用通过FieldInfo对象的属性来实现分类。User类定义了一个公有字段Name和一个私有字段age。通过反射,我们能够获取并打印出这些字段的公有或私有信息。


④ 使用BindingFlags枚举

BindingFlags枚举用于指定控制反射的绑定和搜索方式。在使用Type.GetFieldType.GetFields方法时,可以通过BindingFlags来精确控制要检索的字段类型(如公有/私有、静态/实例等)。

举个例子:

using System;
using System.Reflection;public class User
{public string Name = "Damon";private int age = 30;
}class Program
{static void Main(){Type userType = typeof(User);// 使用BindingFlags枚举获取所有公有字段var publicFields = userType.GetFields(BindingFlags.Public | BindingFlags.Instance);foreach (var field in publicFields){Console.WriteLine($"Public Field: {field.Name}");}// 使用BindingFlags枚举获取所有私有字段var privateFields = userType.GetFields(BindingFlags.NonPublic | BindingFlags.Instance);foreach (var field in privateFields){Console.WriteLine($"Private Field: {field.Name}");}}
}

通过这个例子,我们可以看到BindingFlags枚举在使用反射进行成员访问时的强大能力。它允许开发者以非常精确的方式指定想要访问的成员类型和访问模式,无论这些成员是公有的、私有的、静态的还是实例的。这种灵活性使得BindingFlags在处理复杂反射场景时成为不可或缺的工具。


2、方法反射

方法反射允许在运行时动态地调用类型的方法。这对于实现插件架构、调用不确定或未知方法特别有用。

举个例子:

using System;public class Calculator
{public int Add(int a, int b){return a + b;}
}class Program
{static void Main(){Calculator calc = new Calculator();Type calcType = typeof(Calculator);// 获取Add方法的信息var methodInfo = calcType.GetMethod("Add");// 动态调用Add方法object[] parameters = new object[] { 10, 20 };var result = methodInfo.Invoke(calc, parameters);Console.WriteLine(result); // 输出: 30}
}

在这个例子中,我们首先创建了Calculator类的一个实例。接着,通过typeof(Calculator)获取Calculator类型的Type对象。然后,使用Type对象的GetMethod方法获取Add方法的MethodInfo对象。最后,我们使用MethodInfo对象的Invoke方法动态地调用Add方法,并传入参数。

这种方法的强大之处在于,我们不需要在编译时明确知道Calculator类的实现细节,就能够在运行时调用其方法。这在处理插件或者需要大量反射的框架时尤其有用。

在方法反射的应用中,除了简单地调用方法之外,还可以用于更复杂的场景,如调用带有不同参数的方法、访问私有方法或者调用泛型方法等。下面我们通过一些例子来展示方法反射的这些高级用法。


① 调用有参方法

假设我们有一个Calculator类,它有一个方法Add,这个方法接受两个int类型的参数,并返回它们的和。我们可以使用反射来调用这个方法,即使我们在编译时不知道这个方法的存在。

举个例子:

using System;public class Calculator
{public int Add(int a, int b){return a + b;}
}class Program
{static void Main(){Calculator calc = new Calculator();Type calcType = typeof(Calculator);// 获取Add方法var methodInfo = calcType.GetMethod("Add");// 调用Add方法,传入参数var result = methodInfo.Invoke(calc, new object[] { 10, 20 });Console.WriteLine($"10 + 20 = {result}");}
}

在这个例子中,我们首先实例化了Calculator类。然后,通过使用typeof(Calculator)获得Calculator类型的Type对象,我们利用GetMethod获取名为Add的方法的MethodInfo对象。通过MethodInfo对象的Invoke方法,我们可以动态地调用Add方法,并传递两个整数作为参数,最后打印出这两个整数的和。


② 访问私有方法

在某些情况下,你可能需要调用一个类的私有方法。通过反射,可以实现这一点,即使这通常被认为是破坏封装原则的行为。

举个例子:

using System;
using System.Reflection;public class Messenger
{private void DisplayMessage(string message){Console.WriteLine($"Message: {message}");}
}class Program
{static void Main(){Messenger messenger = new Messenger();Type messengerType = messenger.GetType();// 获取私有方法var methodInfo = messengerType.GetMethod("DisplayMessage", BindingFlags.NonPublic | BindingFlags.Instance);// 调用私有方法methodInfo.Invoke(messenger, new object[] { "Hello, Reflection!" });}
}

这里,我们定义了一个Messenger类,其中包含一个私有方法DisplayMessage。在Main方法中,我们创建了Messenger的一个实例,并通过调用GetType方法获得其类型对象。然后,我们使用GetMethod方法并配合BindingFlags.NonPublic | BindingFlags.Instance参数来获取私有方法的MethodInfo对象。有了这个对象,我们就可以使用Invoke方法来调用DisplayMessage,即使它是私有的。


③ 调用泛型方法

反射还允许调用泛型方法。这在处理需要在运行时确定泛型类型参数的场景下非常有用。

举个例子:

using System;
using System.Reflection;public class Utility
{public void Print<T>(T message){Console.WriteLine($"Message: {message}");}
}class Program
{static void Main(){Utility utility = new Utility();Type utilityType = typeof(Utility);// 获取泛型方法的原始定义var methodInfo = utilityType.GetMethod("Print").MakeGenericMethod(new Type[] { typeof(string) });// 调用泛型方法methodInfo.Invoke(utility, new object[] { "Hello, Generic Reflection!" });}
}

在此例中,Utility类包含一个泛型方法Print<T>,它接受一个类型为T的参数,并将其打印到控制台。在Main方法中,我们首先实例化了Utility类。使用GetMethod获取到Print方法的MethodInfo对象后,我们通过MakeGenericMethod方法指定泛型方法的具体类型。在这个例子中,我们将T指定为string类型。最后,我们使用Invoke方法来调用Print方法,传递了一个字符串作为参数。

这种方法特别有用,因为它允许在运行时决定泛型方法的类型参数,从而提高了代码的灵活性和通用性。


④ 调用带有输出参数的方法

有时候,你可能需要调用的方法包含输出(out)参数。使用反射调用这样的方法时,你也可以获取输出参数的值。

举个例子:

using System;
using System.Reflection;public class Converter
{public bool TryParse(string input, out int result){return int.TryParse(input, out result);}
}class Program
{static void Main(){Converter converter = new Converter();Type converterType = typeof(Converter);// 获取方法信息var methodInfo = converterType.GetMethod("TryParse");// 创建参数数组,包括输入和输出参数object[] parameters = new object[] { "123", null };// 调用方法var success = (bool)methodInfo.Invoke(converter, parameters);// 获取输出参数的值int parsedValue = (int)parameters[1];if (success){Console.WriteLine($"Parsing successful: {parsedValue}");}else{Console.WriteLine("Parsing failed.");}}
}

这个例子中,我们定义了一个Converter类,其中包含一个方法TryParse,这个方法尝试将一个字符串转换为整数,并通过输出参数返回转换结果。在调用这个方法时,我们首先准备了一个参数数组parameters,其中第一个元素是输入字符串,第二个元素是用于接收输出值的占位符(初始化为null)。调用Invoke方法后,输出参数的值被填充到了parameters数组的相应位置,我们可以通过索引访问并使用这个值。

这种调用方法对于处理需要输出参数的方法非常有用,尤其是在动态场景下,它允许开发者在运行时与方法的输入和输出交互,增加了代码的灵活性。


⑤ 调用重载方法

在有些情况下,一个类可能有多个同名方法(即方法重载)。使用反射调用特定的重载版本时,可以通过指定参数类型来获取正确的MethodInfo对象。

举个例子:

using System;
using System.Reflection;public class Calculator
{public int Add(int a, int b){return a + b;}public double Add(double a, double b){return a + b;}
}class Program
{static void Main(){Calculator calc = new Calculator();Type calcType = typeof(Calculator);// 指定要调用的重载方法的参数类型Type[] paramTypes = { typeof(int), typeof(int) };MethodInfo methodInfo = calcType.GetMethod("Add", paramTypes);// 调用重载方法var result = methodInfo.Invoke(calc, new object[] { 10, 20 });Console.WriteLine($"10 + 20 = {result}");}
}

在这个例子中,Calculator类有两个Add方法的重载版本:一个接受两个int类型的参数,另一个接受两个double类型的参数。为了调用特定的重载版本(在这里是接受int参数的版本),我们在GetMethod调用中提供了一个表示参数类型的Type数组。这样,就可以准确地获取到所需的MethodInfo对象,并通过Invoke方法调用它。


三、反射的使用场景

① 类型检查和元数据访问

这一类应用涉及到在运行时获取类型的信息,如类的名称、方法、属性、字段等。通过元数据访问,程序可以动态地获取和操作类型信息,实现高度的灵活性。

  • 获取类型信息:包括类名、命名空间、继承层次结构等。
  • 成员访问:访问和操作字段、属性、方法、事件等。

② 动态对象创建和方法调用

反射最直观的用途之一是动态地创建对象和调用方法。这使得开发者可以在不知道对象确切类型的情况下,进行对象的实例化和方法调用。

  • 动态对象创建:通过类型名称动态创建对象实例。
  • 动态方法执行:在运行时调用方法,包括公有、私有方法和重载方法的调用。

③ 动态代理和拦截

反射可以用来实现动态代理和方法拦截,这在很多高级编程场景中非常有用,比如实现AOP(面向切面编程)。

  • 动态代理:创建一个对象的代理,代理对象可以在目标对象的方法调用前后执行额外的逻辑。
  • 方法拦截:拦截对特定方法的调用,可以用于日志记录、性能监测、事务处理等。

④ 自定义属性(Attribute)处理

反射允许程序检查代码中的自定义属性,这是实现各种框架(如测试框架、ORM框架等)的基础。

  • 属性读取:读取类、方法、字段等上的自定义属性,用于配置或特殊处理。
  • 属性驱动的逻辑:基于自定义属性执行特定逻辑,如序列化/反序列化、数据库操作等。

⑤ 动态代码生成和编译

利用反射,结合表达式树(Expression Trees)或其他动态代码生成技术,可以在运行时生成和编译代码。这对于需要大量动态性的应用非常有用。

  • 动态代码生成:生成新的方法或类定义。
  • 运行时编译:将动态生成的代码编译成可执行代码。

反射的应用覆盖了从基础的类型探查到复杂的动态代理和代码生成等高级场景,为开发高度灵活和动态的应用程序提供了强大的支持。每种应用场景都展示了反射机制如何使得代码能够在运行时适应和响应不同的需求,从而实现高度的灵活性和动态性。


四、反射总结

反射是C#中一个非常强大的特性是C#高级编程中不可或缺的一部分,了解和掌握反射的使用可以帮助开发者编写更加灵活和强大的.NET应用程序。它提供了一种在运行时查询和操作类型信息的能力,通过反射,我们可以动态地创建对象、调用方法、访问字段和属性,这为编写灵活和动态的代码提供了极大的便利。

尽管反射提供了强大的功能,但它也有一些缺点。反射操作通常比直接代码调用要慢,因为它需要在运行时解析类型信息。此外,过度使用反射可能会使代码变得难以理解和维护。因此,我们应该谨慎使用,在使用反射时应该权衡其给项目带来的好处和成本,避免不必要的性能开销和复杂性增加。

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

【STL学习】(2)string的模拟实现

前言 本文将模拟实现string的一些常见功能&#xff0c;目的在于加深理解string与回顾类与对象的相关知识。 一、前置知识 string是表示可变长的字符序列的类string的底层是使用动态顺序表存储的string对象不以’\0’字符为终止算长度&#xff0c;而是以size有效字符的个数算长…

分布式系统面试全集通第一篇(dubbo+redis+zookeeper----分布式+CAP+BASE+分布式事务+分布式锁)

目录 分布式系统面试全集通第一篇什么是分布式?和微服务的区别什么是分布式分布式与微服务的区别 什么是CAP?为什么不能三者同时拥有分区容错性一致性可用性 Base理论了解吗基本可用软状态最终一致性 什么是分布式事务分布式事务有哪些常见的实现方案?2PC&#xff08;Two Ph…

2024-03-26 Android8.1 px30 WI-FI 模块rtl8821cu调试记录

一、kernel 驱动&#xff0c;我这里使用v5.8.1.2_35530.20191025_COEX20191014-4141这个版本&#xff0c;下载这个版本的驱动可以参考下面的文章。 2021-04-12 RK3288 Android7.1 USB wifi bluetooth 模块RTL8821CU 调试记录_rk平台rtl8821cu蓝牙调试-CSDN博客 二、Makefile文…

Postwoman 安装

Postwoman作为Postman的女朋友&#xff0c;具有免费开源、轻量级、快速且美观等特性&#xff0c;是一款非常好用的API调试工具。能帮助程序员节省时间&#xff0c;提升工作效率。 Github地址&#xff1a;GitHub - hoppscotch/hoppscotch: &#x1f47d; Open source API devel…

如何打包springboot项目并部署服务器

创建一个springboot项目&#xff0c;先写一个接口&#xff0c;我这里是dabaimao/jiekou,启动访问 在pom中加上maven插件 <build><plugins><plugin><groupId>org.springframework.boot</groupId><artifactId>spring-boot-maven-plugin<…

2010年之前电脑ubuntu安装nvidia驱动黑屏处理

装好驱动 仿真fps直接到60Hz 陈旧设备 都是非常老旧的电脑&#xff0c;没钱换新电脑&#xff0c;就这么穷…… 电脑详细配置&#xff1a; 冲动 想装显卡驱动提升一下性能&#xff0c;结果……黑了 黑习惯了也无所谓&#xff0c;几分钟就能解决&#xff0c;关键还是太穷&…

ES6 字符串/数组/对象/函数扩展

文章目录 1. 模板字符串1.1 ${} 使用1.2 字符串扩展(1) ! includes() / startsWith() / endsWith()(2) repeat() 2. 数值扩展2.1 二进制 八进制写法2.2 ! Number.isFinite() / Number.isNaN()2.3 inInteger()2.4 ! 极小常量值Number.EPSILON2.5 Math.trunc()2.6 Math.sign() 3.…

YOLOv9改进策略:注意力机制 | 动态稀疏注意力的双层路由方法BiLevelRoutingAttention | CVPR2023

&#x1f4a1;&#x1f4a1;&#x1f4a1;本文改进内容&#xff1a; CVPR2023 动态稀疏注意力的双层路由方法BiLevelRoutingAttention&#xff0c;强烈推荐&#xff0c;涨点很不错&#xff0c;同时被各个领域的魔改次数甚多&#xff0c;侧面验证了性能。 &#x1f4a1;&#x1…

我们该如何优化迭代自己?

哈喽&#xff0c;你好啊&#xff0c;我是雷工&#xff01; 一款软件如果想变得完美&#xff0c;那么肯定需要不断的试运行和更新迭代。 我们和软件一样&#xff0c;生活中难免会有错误的决策&#xff0c;失误的事件&#xff0c;为了能够解决我们自身存在的BUG&#xff0c;我们该…

设计用于驱动12 V汽车接地负载,VN5E160ASTR、VND5E160MJTR、VND5E025AKTR、VND5E050ACKTR 单/双通道高侧驱动器

摘要 意法半导体VIPower系列高侧开关符合汽车应用要求&#xff0c;内嵌先进的控制功能&#xff0c;其新型保护机制适用于各种负载类型及额定功率。 此类开关是汽车系统的理想选择&#xff0c;如&#xff1a;接线盒、内部/外部照明、直流电机驱动等&#xff0c;并适用于任何需…

基于nodejs+vue基于协同过滤算法的私人诊python-flask-django-php

实现后的私人诊所管理系统基于用户需求分析搭建的&#xff0c;并且会有个人中心&#xff0c;患者管理&#xff0c;医生管理&#xff0c;科室管理&#xff0c;出诊医生管理&#xff0c;预约挂号管理&#xff0c;预约取消管理&#xff0c;病历信息管理&#xff0c;药品信息管理&a…

qt事件机制学习笔记

实现闹钟功能 #include "widget.h" #include "ui_widget.h"Widget::Widget(QWidget *parent): QWidget(parent), ui(new Ui::Widget), speecher(new QTextToSpeech(this)) //给语音播报者实例化空间 {ui->setupUi(this); }Widget::~Widget() {delete …

【GameFramework框架内置模块】18、界面(UI)

推荐阅读 CSDN主页GitHub开源地址Unity3D插件分享简书地址QQ群&#xff1a;398291828 大家好&#xff0c;我是佛系工程师☆恬静的小魔龙☆&#xff0c;不定时更新Unity开发技巧&#xff0c;觉得有用记得一键三连哦。 一、前言 【GameFramework框架】系列教程目录&#xff1a;…

WebGIS开发应该从哪些方面做准备

工程化思想 环境配置项目构建npm&#xff1a;Node包管理器&#xff0c;是 JavaScript 运行时 Node.js 的默认程序包管理器。 https://www.freecodecamp.org/chinese/news/what-is-npm-a-node-package-manager-tutorial-for-beginners/新建一个前端工程项目&#xff1a;前端框…

在项目中数据库如何优化?【MySQL主从复制(创建一个从节点复制备份数据)】【数据库读写分离ShardingJDBC(主库写,从库读)】

MySQL主从复制 MySQL主从复制介绍MySQL复制过程分成三步&#xff1a;1). MySQL master 将数据变更写入二进制日志( binary log)2). slave将master的binary log拷贝到它的中继日志&#xff08;relay log&#xff09;3). slave重做中继日志中的事件&#xff0c;将数据变更反映它自…

Vue 02 组件、Vue CLI

Vue学习 Vue 0201 组件引入概念组件的两种编写形式① 非单文件组件基本使用使用细节组件嵌套组件本质 VueComponent重要的内置关系 ② 单文件组件 02 Vue CLI介绍 & 文档安装使用步骤脚手架结构render默认配置ref 属性props配置mixin配置项插件scoped 样式案例&#xff1a;…

MySQL将id相同的两行数据合并group_concat

MySQL将id相同的两行数据合并 group_concat这个函数能将相同的行组合起来&#xff0c;省老事了。 MySQL中group_concat函数 完整的语法如下&#xff1a; group_concat([DISTINCT] 要连接的字段 [Order BY ASC/DESC 排序字段] [Separator ‘分隔符’]) 1.基本查询 Sql代码 2.…

java Web会议信息管理系统 用eclipse定制开发mysql数据库BS模式java编程jdbc

一、源码特点 jsp 会议信息管理系统是一套完善的web设计系统&#xff0c;对理解JSP java SERLVET mvc编程开发语言有帮助&#xff0c;系统具有完整的源代码和数据库&#xff0c;系统主要采用B/S模式开发。开发环境为TOMCAT7.0,eclipse开发&#xff0c;数据库为Mysql5.0&am…

ActiveMQ-04如何搭建一个完美的ActiveMQ集群

集群架构是一个很大的话题&#xff0c;官网就给我们介绍了几种 客户端&#xff1a;队列消费者集群-Queue Consumer Clusters服务端&#xff1a;Broker集群 - Broker Clusters 静态发现动态发现 服务端&#xff1a;Master-Slave 主从集群 Shared File System Master SlaveJDBC …

力扣hot100:207. 课程表

这是一道拓扑排序问题&#xff0c;也可以使用DFS判断图中是否存在环。详情请见&#xff1a;官方的BFS算法请忽略&#xff0c;BFS将问题的实际意义给模糊了&#xff0c;不如用普通拓扑排序思想。 数据结构&#xff1a;图的拓扑排序与关键路径 拓扑排序&#xff1a; class Sol…