当前位置: 首页 > news >正文

C# 反射详解

反射是C#中的一个强大特性,允许程序在运行时检查和操作类型和对象的信息。

通过反射,你可以获取类型的属性方法构造函数等信息,并可以动态创建对象、调用方法或访问属性,甚至可以实现某些框架或库的核心功能。

反射的基本概念

  1. 类型信息:反射使我们能够获取类型的名称、成员(如属性、方法、字段)、基类、接口等信息。
  2. 动态创建对象:可以在运行时创建一个类的实例,而不需要在编译时知道具体的类。
  3. 访问成员:通过反射,可以访问类的私有成员,甚至是静态成员。

反射的常用类

反射的主要类都在 System.Reflection 命名空间中,以下是一些重要的类:

  • Type:表示类型的信息,可以用来获取类的信息,包括方法、属性、字段等。
  • Assembly:表示一个程序集,提供有关程序集的元数据。
  • MethodInfo:表示方法的信息,可以用来调用方法。
  • PropertyInfo:表示属性的信息,可以用来获取或设置属性的值。

反射常用类及其常用方法表

类名常用方法描述
TypeGetProperties()获取公共属性的数组。
GetMethods()获取公共方法的数组。
GetFields()获取公共字段的数组。
GetConstructors()获取构造函数的信息。
GetInterfaces()获取该类型实现在的所有接口。
GetCustomAttributes()获取应用于当前类型的所有自定义特性(Attributes)。
BaseType获取此类型的基类型(父类)。
PropertyInfoGetValue(object obj)获取指定对象上属性的值。
SetValue(object obj, object value)设置指定对象上属性的值。
CanRead指示属性是否有获取访问器。
CanWrite指示属性是否有设置访问器。
MethodInfoInvoke(object obj, object[] parameters)调用方法。
GetParameters()获取方法的参数信息。
ReturnType获取方法的返回类型。
IsStatic指示方法是否为静态方法。
FieldInfoGetValue(object obj)获取指定对象上字段的值。
SetValue(object obj, object value)设置指定对象上字段的值。
FieldType获取字段的类型。
ConstructorInfoInvoke(object[] parameters)调用构造函数以创建实例。
GetParameters()获取构造函数的参数信息。
IsStatic指示构造函数是否为静态构造函数。
AssemblyGetTypes()获取程序集中的所有类型。
GetName()获取程序集的名称信息。
GetCustomAttributes()获取应用于该程序集的所有自定义特性。

typeof 与 GetType

1. typeof 操作符

定义

  • typeof 是一个操作符,用于在编译时获取某个类型的 Type 对象。它适用于任何类型,包括类、结构、接口和基本类型。

用法

  • typeof 的语法是:Type type = typeof(SomeType);
  • 它返回的结果是 System.Type 的一个实例,表示指定类型的信息,能够用于反射。

2. GetType 方法

定义

  • GetType 是一个实例方法,它属于所有 .NET 对象的基类 Object。当你在一个对象实例上调用 GetType 方法时,它会返回该实例的 Type 对象。

用法

  • GetType 的语法是:Type type = someObject.GetType();
  • 它用于获取对象的实际类型信息。

特点typeofGetType
类型获取方式在编译时获取类型信息在运行时获取对象实例的类型信息
使用场景获取静态类型的 Type 对象获取对象实例的 Type 对象
返回类型总是返回特定类型的 Type返回调用该方法的对象的实际类型的 Type
语法Type type = typeof(ClassName);Type type = instance.GetType();

反射的使用示例

1. 获取类型信息

使用 Type 类可以获取对象的类型信息:

我们定义一个简单的类 Person,然后使用反射获取它的类型信息。

在此,我们将typeof和GetType进行对比。

using System;
namespace Tdm;
public class Person
{public string Name { get; set; }public int Age { get; set; }public void DisplayInfo(){Console.WriteLine($"姓名: {Name}, 年龄: {Age}");}
}class Program
{static void Main(){Person person = new Person(); // 创建 Person 对象// 使用 GetType 方法Type typeFromInstance = person.GetType();Console.WriteLine("使用 GetType() 输出:");Console.WriteLine("类型名称: " + typeFromInstance.Name);       // 输出:类型名称: PersonConsole.WriteLine("命名空间: " + typeFromInstance.Namespace); // 输出:命名空间: (Tdm)// 使用 typeof 操作符Type typeFromTypeOf = typeof(Person); Console.WriteLine("\n使用 typeof 输出:");Console.WriteLine("类型名称: " + typeFromTypeOf.Name);       // 输出:类型名称: PersonConsole.WriteLine("命名空间: " + typeFromTypeOf.Namespace); // 输出:命名空间: (Tdm)}
}
使用 GetType() 输出:
类型名称: Person
命名空间: (Tdm)使用 typeof 输出:
类型名称: Person
命名空间: (Tdm)

2. 动态创建对象

我们使用反射动态创建 Person 对象,并设置其属性。

using System;public class Person
{public string Name { get; set; }public int Age { get; set; }public void DisplayInfo(){Console.WriteLine($"姓名: {Name}, 年龄: {Age}");}
}class Program
{static void Main(){Type personType = typeof(Person); // 获取 Person 类型Person person = (Person)Activator.CreateInstance(personType); // 动态创建 Person 实例// 设置属性值person.Name = "小明"; // 设置姓名person.Age = 25;      // 设置年龄person.DisplayInfo(); // 调用方法显示信息,输出:姓名: 小明, 年龄: 25}
}

3. 访问和修改属性

我们将使用反射访问和修改 Person 类的属性。

using System;
using System.Reflection;public class Person
{public string Name { get; set; } // 姓名属性public int Age { get; set; }      // 年龄属性
}class Program
{static void Main(){Person person = new Person(); // 创建 Person 对象Type type = person.GetType(); // 获取类型信息// 获取属性信息PropertyInfo nameProperty = type.GetProperty("Name"); // 获取 Name 属性PropertyInfo ageProperty = type.GetProperty("Age");   // 获取 Age 属性// 设置属性值nameProperty.SetValue(person, "张三"); // 通过反射设置 Name 属性ageProperty.SetValue(person, 30);      // 通过反射设置 Age 属性// 获取属性值并输出Console.WriteLine($"{nameProperty.Name}: {nameProperty.GetValue(person)}"); // 输出:Name: 张三Console.WriteLine($"{ageProperty.Name}: {ageProperty.GetValue(person)}");     // 输出:Age: 30}
}

4.调用方法

我们将使用反射调用 Person 类的方法。

using System;
using System.Reflection;public class Person
{public void DisplayInfo(string greeting) // 显示信息的方法{Console.WriteLine($"{greeting}, 我是C#开发工程师!。"); // 输出问候信息}
}class Program
{static void Main(){Person person = new Person(); // 创建 Person 对象Type type = person.GetType(); // 获取类型信息// 获取方法信息MethodInfo methodInfo = type.GetMethod("DisplayInfo"); // 获取 DisplayInfo 方法// 调用方法并传递参数methodInfo.Invoke(person, new object[] { "你好" }); // 输出:你好,我是C#开发工程师!}
}

反射的使用场景

1. 序列化和反序列化

反射非常适合于在对象和数据格式(如 JSON 或 XML)之间进行转换。使用反射,可以动态遍历对象的属性,将其值提取出来,构建相应的序列化格式。

示例:将一个对象转换为 JSON 字符串,或将一个 JSON 字符串转换为对象。

2. 依赖注入

许多现代框架(如 ASP.NET Core)使用反射来实现依赖注入。在运行时,框架可以通过反射创建对象实例和解析依赖关系。

示例:在 ASP.NET Core 中,使用反射自动识别控制器并注入其依赖服务。

3. 动态类型处理

在处理动态类型(例如,使用动态对象或 COM 组件)时,反射可以帮助获取类型信息并处理其方法和属性。

示例:通过反射操作一个未知类型的对象,尤其是在不确定其类型时。

注意事项

1. 性能问题

反射涉及到查找和动态调用,通常比直接调用方法慢。在性能敏感的代码中,应尽量减少反射的使用,或者对反射结果进行缓存。

2. 类型安全

反射不进行编译时类型检查,因此可能导致运行时错误。如果获取的成员不存在,程序将抛出异常。在使用反射时,务必保证类型的匹配。

        

相关文章:

C# 反射详解

反射是C#中的一个强大特性,允许程序在运行时检查和操作类型和对象的信息。 通过反射,你可以获取类型的属性、方法、构造函数等信息,并可以动态创建对象、调用方法或访问属性,甚至可以实现某些框架或库的核心功能。 反射的基本概念…...

pgadmin安装后运行不能启动界面的问题

在本人机器上安装了pgsql10后,自带的pgadmin安装后运行时能打开edge并显示数据库server和数据库的,后来又安装了pgsql17,结果安装后想打开pgadmin,结果一直在等待最后,爆出类似于下面的错误。 pgAdmin Runtime Enviro…...

跳表(Skip List)

跳表(Skip List) 跳表是一种用于快速查找、插入和删除的概率型数据结构,通常用于替代平衡二叉搜索树(如 AVL 树或红黑树)。跳表通过在有序链表的基础上增加多层索引,使得查找操作的平均时间复杂度降低&…...

前端实现把整个页面转成PDF保存到本地(DOM转PDF)

一、问题 遇到一个需求,就是要把整个看板页面导出成PDF用在汇报,也就是要把整个DOM生成一个PDF保存到本地。 二、解决方法 1、解决思路:使用插件 jspdf 和 html2canvas,我用的版本如下图 2、代码实现 import { jsPDF } from …...

Vue 3 学习文档(一)

最近打算做一个项目,涉及到一些前端的知识,因上一次接触前端已经是三四年前了,所以捡一些简单的功能做一下复习。 响应式函数:reactive 和 ref属性绑定:v-bind 和简写语法事件监听:v-on 和简写语法 双向绑…...

【适配】屏幕拖拽-滑动手感在不同分辨率下的机型适配

接到一个需求是类似下图的3D多房间视角,需要拖拽屏幕 问题 在做这种屏幕拖拽的时候发现,需要拖拽起来有跟手的感觉,会存在不同分辨率机型的适配问题。 即:美术调整好了机型1的手感,能做到手指按下顶层地板上下挪动&…...

牛客周赛 Round 69(A~E)

文章目录 A 构造C的歪思路code B 不要三句号的歪思路code C 仰望水面的歪思路code D 小心火烛的歪思路code E 喜欢切数组的红思路code 牛客周赛 Round 69 A 构造C的歪 思路 签到题,求出公差d,让最大的数加上公差d即可 code int a,b;cin >> a &…...

Spring Boot 实战:分别基于 MyBatis 与 JdbcTemplate 的数据库操作方法实现与差异分析

1. 数据库新建表 CREATE TABLE table_emp(id INT AUTO_INCREMENT,emp_name CHAR(100),age INT,emp_salary DOUBLE(10,5),PRIMARY KEY(id) );INSERT INTO table_emp(emp_name,age,emp_salary) VALUES("tom",18,200.33); INSERT INTO table_emp(emp_name,age,emp_sala…...

【jmeter】服务器使用jmeter压力测试(从安装到简单压测示例)

一、服务器上安装jmeter 1、官方下载地址,https://jmeter.apache.org/download_jmeter.cgi 2、服务器上用wget下载 # 更新系统 sudo yum update -y# 安装 wget 以便下载 JMeter sudo yum install wget -y# 下载 JMeter 压缩包(使用 JMeter 官方网站的最…...

使用Python实现自动化邮件通知:当长时程序运行结束时

使用Python实现自动化邮件通知:当长时程序运行结束时 前提声明 本代码仅供学习和研究使用,不得用于商业用途。请确保在合法合规的前提下使用本代码。 目录 引言项目背景项目设置代码分析 导入所需模块定义邮件发送函数发送邮件 实现步骤结语全部代码…...

框架学习07 - SpringMVC 其他功能实现

一. 拦截器实现HandlerInterceptor 接⼝ SpringMVC 中的 Interceptor 拦截器也是相当重要和相当有⽤的,它的主要作⽤是拦截⽤户的请求并进⾏相应的处理。⽐如通过它来进⾏权限验证,或者是来判断⽤户是否登陆等操作。对于 SpringMVC 拦截器的定义⽅式有两…...

NAT:连接私有与公共网络的关键技术(4/10)

一、NAT 的工作原理 NAT 技术的核心功能是将私有 IP 地址转换为公有 IP 地址,使得内部网络中的设备能够与外部互联网通信。其工作原理主要包括私有 IP 地址到公有 IP 地址的转换、端口号映射以及会话表维护这几个步骤。 私有 IP 地址到公有 IP 地址的转换&#xff1…...

RabbitMQ2:介绍、安装、快速入门、数据隔离

欢迎来到“雪碧聊技术”CSDN博客! 在这里,您将踏入一个专注于Java开发技术的知识殿堂。无论您是Java编程的初学者,还是具有一定经验的开发者,相信我的博客都能为您提供宝贵的学习资源和实用技巧。作为您的技术向导,我将…...

衡山派D133EBS 开发环境安装及SDK编译烧写镜像烧录

1.创建新文件夹,用来存放SDK包(其实本质就是路径要对就ok了),右键鼠标通过Open Git Bash here来打开git 输入命令 git clone --depth1 https://gitee.com/lcsc/luban-lite.git 来拉取,如下所示:&#xff0…...

【Spring MVC】如何获取cookie/session以及响应@RestController的理解,Header的设置

前言 🌟🌟本期讲解关于SpringMVC的编程之参数传递~~~ 🌈感兴趣的小伙伴看一看小编主页:GGBondlctrl-CSDN博客 🔥 你的点赞就是小编不断更新的最大动力 🎆那么废…...

C++设计模式行为模式———策略模式

文章目录 一、引言二、策略模式三、总结 一、引言 策略模式是一种行为设计模式, 它能让你定义一系列算法, 并将每种算法分别放入独立的类中, 以使算法的对象能够相互替换。与模板方法模式类似,都是以扩展的方式来支持未来的变化。…...

Spring Cloud 中 bootstrap.yml 配置文件详解

Spring Cloud 中 bootstrap.yml 配置文件详解 1. 什么是 bootstrap.yml? bootstrap.yml 是 Spring Cloud 提供的一个特殊配置文件,主要用于初始化 Spring Cloud 应用程序的环境。与常见的 application.yml 不同,bootstrap.yml 在 Spring 应用…...

Java项目实战II基于SpringBoot前后端分离的网吧管理系统(开发文档+数据库+源码)

目录 一、前言 二、技术介绍 三、系统实现 四、核心代码 五、源码获取 全栈码农以及毕业设计实战开发,CSDN平台Java领域新星创作者,专注于大学生项目实战开发、讲解和毕业答疑辅导。获取源码联系方式请查看文末 一、前言 随着互联网技术的不断发展…...

ASP网络安全讲述

一 前言   Microsoft Active Server Pages(ASP)是服务器端脚本编写环境,使用它可以创建和运行动态、交互的 Web 服务器应用程序。使用 ASP 可以组合 HTML 页 、脚本命令和 ActiveX 组件以创建交互的 Web 页和基于 Web 的功能强大的应用程序…...

DFS 创建分级菜单

菜单级别不确定&#xff0c;想要自适应&#xff0c;且可以折叠的菜单。 数据是一个数组。 <template><div class"Level" ref"Level"></div> </template>import {ref} from vue export default{data(){Level:ref(null),menuData…...

Chapter03-Authentication vulnerabilities

文章目录 1. 身份验证简介1.1 What is authentication1.2 difference between authentication and authorization1.3 身份验证机制失效的原因1.4 身份验证机制失效的影响 2. 基于登录功能的漏洞2.1 密码爆破2.2 用户名枚举2.3 有缺陷的暴力破解防护2.3.1 如果用户登录尝试失败次…...

服务器硬防的应用场景都有哪些?

服务器硬防是指一种通过硬件设备层面的安全措施来防御服务器系统受到网络攻击的方式&#xff0c;避免服务器受到各种恶意攻击和网络威胁&#xff0c;那么&#xff0c;服务器硬防通常都会应用在哪些场景当中呢&#xff1f; 硬防服务器中一般会配备入侵检测系统和预防系统&#x…...

1.3 VSCode安装与环境配置

进入网址Visual Studio Code - Code Editing. Redefined下载.deb文件&#xff0c;然后打开终端&#xff0c;进入下载文件夹&#xff0c;键入命令 sudo dpkg -i code_1.100.3-1748872405_amd64.deb 在终端键入命令code即启动vscode 需要安装插件列表 1.Chinese简化 2.ros …...

从零开始打造 OpenSTLinux 6.6 Yocto 系统(基于STM32CubeMX)(九)

设备树移植 和uboot设备树修改的内容同步到kernel将设备树stm32mp157d-stm32mp157daa1-mx.dts复制到内核源码目录下 源码修改及编译 修改arch/arm/boot/dts/st/Makefile&#xff0c;新增设备树编译 stm32mp157f-ev1-m4-examples.dtb \stm32mp157d-stm32mp157daa1-mx.dtb修改…...

第一篇:Agent2Agent (A2A) 协议——协作式人工智能的黎明

AI 领域的快速发展正在催生一个新时代&#xff0c;智能代理&#xff08;agents&#xff09;不再是孤立的个体&#xff0c;而是能够像一个数字团队一样协作。然而&#xff0c;当前 AI 生态系统的碎片化阻碍了这一愿景的实现&#xff0c;导致了“AI 巴别塔问题”——不同代理之间…...

相机Camera日志分析之三十一:高通Camx HAL十种流程基础分析关键字汇总(后续持续更新中)

【关注我,后续持续新增专题博文,谢谢!!!】 上一篇我们讲了:有对最普通的场景进行各个日志注释讲解,但相机场景太多,日志差异也巨大。后面将展示各种场景下的日志。 通过notepad++打开场景下的日志,通过下列分类关键字搜索,即可清晰的分析不同场景的相机运行流程差异…...

Aspose.PDF 限制绕过方案:Java 字节码技术实战分享(仅供学习)

Aspose.PDF 限制绕过方案&#xff1a;Java 字节码技术实战分享&#xff08;仅供学习&#xff09; 一、Aspose.PDF 简介二、说明&#xff08;⚠️仅供学习与研究使用&#xff09;三、技术流程总览四、准备工作1. 下载 Jar 包2. Maven 项目依赖配置 五、字节码修改实现代码&#…...

处理vxe-table 表尾数据是单独一个接口,表格tableData数据更新后,需要点击两下,表尾才是正确的

修改bug思路&#xff1a; 分别把 tabledata 和 表尾相关数据 console.log() 发现 更新数据先后顺序不对 settimeout延迟查询表格接口 ——测试可行 升级↑&#xff1a;async await 等接口返回后再开始下一个接口查询 ________________________________________________________…...

boost::filesystem::path文件路径使用详解和示例

boost::filesystem::path 是 Boost 库中用于跨平台操作文件路径的类&#xff0c;封装了路径的拼接、分割、提取、判断等常用功能。下面是对它的使用详解&#xff0c;包括常用接口与完整示例。 1. 引入头文件与命名空间 #include <boost/filesystem.hpp> namespace fs b…...

Python第七周作业

Python第七周作业 文章目录 Python第七周作业 1.使用open以只读模式打开文件data.txt&#xff0c;并逐行打印内容 2.使用pathlib模块获取当前脚本的绝对路径&#xff0c;并创建logs目录&#xff08;若不存在&#xff09; 3.递归遍历目录data&#xff0c;输出所有.csv文件的路径…...