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

CH03_反射

第3章:反射


本章目标

  1. 掌握反射的原理

  2. 熟悉反射的基本运用

本章内容

反射是什么

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

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

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

    • 元数据存储以下信息:

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

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

  • 运行代码

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

反射的工作原理

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

提示:

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

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

反射的使用

反射相关的类和命名空间

反射的命名空间:

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		//成员
Type类的应用
Type类中的基本属性

在这里插入图片描述

/// <summary>
/// 学生类
/// </summary>
class Student
{private int _num = 0;public string Phone = "15818704257";public string Name { get; set; }public string Address { get; set; }public Student(){//Console.WriteLine("Student 默认构造函数");}public Student(string name){//Console.WriteLine($"Student 参数化构造函数:{name}");}public Student(string name,string phone,string address){}public int PublicMethod(){return int.MinValue;}internal void InternalMethod(){}private void PrivateMethod(){}public void Show(int id){Console.WriteLine("调用了Show()方法!"+id);}
}class MyArray<T>
{}/// <summary>
/// USB接口
/// </summary>
interface IUsb
{}struct Teacher
{}
/// <summary>
/// Type类的基本属性
/// </summary>
static void Fun1()
{Type t0 = typeof(MyArray<int>);Type t1= typeof(Student);Type t2 = typeof(IUsb);Type t3 = typeof(Teacher);Type t4 = typeof(string);Console.WriteLine("名称:"+t1.Name);Console.WriteLine("全名:"+t1.FullName);Console.WriteLine("命名空间:"+t1.Namespace);Console.WriteLine("是否是抽象的:"+t1.IsAbstract);Console.WriteLine("是否是公共的:" + t1.IsPublic);Console.WriteLine("是否是类:" + t1.IsClass);Console.WriteLine("是否是枚举:" + t1.IsEnum);Console.WriteLine("是否是接口:" + t1.IsInterface);Console.WriteLine("是否是嵌套定义的:" + t1.IsNested);Console.WriteLine("是否是值类型:" + t1.IsValueType);Console.WriteLine("是否是泛型类型:" + t1.IsGenericType);
}
using ClassLibrary1;
using System;
using System.Collections.Generic;
using System.Linq;
using System.Reflection;
using System.Text;
using System.Threading.Tasks;namespace CH03Demo
{internal class Program{static void Main(string[] args){Fun11();Console.ReadLine();}}
}

提示:

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

Type类的Assembly属性

在这里插入图片描述

/// <summary>
/// Type类的Assembly属性
/// </summary>
static void Fun2()
{Type type = typeof(Student);//程序集Assembly a1 = type.Assembly;Console.WriteLine("位置:" + a1.CodeBase);Console.WriteLine("全名:" + a1.FullName);Console.WriteLine("--------------------------------");Type t1 = typeof(Student);Type t2 = typeof(string);Console.WriteLine("全名:" + t1.FullName);Console.WriteLine("全名:" + t2.FullName);//查看程序集限定名Console.WriteLine(t1.AssemblyQualifiedName);Console.WriteLine(t2.AssemblyQualifiedName);}
Type类对象获取构造函数

在这里插入图片描述

/// <summary>
/// 查看构造函数
/// </summary>
static void Fun3()
{Type t= typeof(Student);ConstructorInfo[] ciArray= t.GetConstructors();//遍历构造函数foreach (ConstructorInfo ci in ciArray){Console.WriteLine("构造函数名:" + ci.Name);foreach (ParameterInfo item in ci.GetParameters()){Console.WriteLine("参数:{0},类型:{1}", item.Name, item.ParameterType);}Console.WriteLine("-------------------------------");}
}
Type类对象获取方法

在这里插入图片描述

/// <summary>
/// 查看当前实例的所有public方法
/// </summary>
static void Fun4()
{Type t = typeof(Student);MethodInfo[] miArray = t.GetMethods();//遍历构造函数foreach (MethodInfo mi in miArray){Console.WriteLine("方法名:{0},返回类型:{1}",mi.Name,mi.ReturnType);foreach (ParameterInfo item in mi.GetParameters()){Console.WriteLine("参数:{0},类型:{1}" ,item.Name,item.ParameterType);}Console.WriteLine("-------------------------------");}
}
Type类对象获取属性

在这里插入图片描述

/// <summary>
/// 查看当前实例的所有public属性
/// </summary>
static void Fun5()
{Type t = typeof(Student);PropertyInfo[] miArray = t.GetProperties();//遍历foreach (PropertyInfo pi in miArray){Console.WriteLine("属性名:{0},类型:{1}",pi.Name,pi.PropertyType);Console.WriteLine("-------------------------------");}
}
Type类对象获取字段

在这里插入图片描述

/// <summary>
/// 查看当前实例的所有public字段
/// </summary>
static void Fun6()
{Type t = typeof(Student);FieldInfo[] fiArray = t.GetFields();//遍历foreach (FieldInfo fi in fiArray){Console.WriteLine("字段名:{0},类型:{1}", fi.Name, fi.FieldType);Console.WriteLine("-------------------------------");}
}
Type类对象获取成员

在这里插入图片描述

/// <summary>
/// 查看当前实例的所有public成员
/// </summary>
static void Fun7()
{Type t = Type.GetType("CH03Demo.Student"); //typeof(Student);MemberInfo[] miArray = t.GetMembers();//遍历foreach (MemberInfo mi in miArray){Console.WriteLine("成员名:{0},类型:{1}", mi.Name, mi.MemberType);Console.WriteLine("-------------------------------");}
}
使用BindingFlags筛选成员

在这里插入图片描述

/// <summary>
/// 使用绑定标志枚举筛选成员
/// </summary>
static void Fun8()
{Type t = Type.GetType("CH03Demo.Student"); //typeof(Student);//GetMembers 中传入 BindingFlags 相当于是对成员信息进行一个过滤//BindingFlags 不仅仅是GetMembers 专有,很多方法中都可以传入BindingFlags进行过滤//BindingFlags 是位标志枚举,可使用 | & ^ 等运算符, | 表示取并集,& 表示取交集,^ 表示取差集//BindingFlags.Public 表示公共成员//BindingFlags.NonPublic 表示非公共成员//BindingFlags.Instance 表示实例成员//BindingFlags.Static 表示静态成员MemberInfo[] miArray = t.GetMembers(BindingFlags.NonPublic | BindingFlags.Instance);//遍历foreach (MemberInfo mi in miArray){Console.WriteLine("成员名:{0},类型:{1}", mi.Name, mi.MemberType);Console.WriteLine("-------------------------------");}
}

提示:

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

获取Type实例的方式
/// <summary>
/// 获取Type实例的方式
/// </summary>
static void Fun9()
{Student stu = new Student();//方式1:System.Type类的ComDefaultInterface特性Type t1 = typeof(Student);//方式2:System.Object中的GetType()方法Type t2 = stu.GetType();//方式3:System.Type中GetType()方法//想通程序集,传入FullName即可(命名空间+类名)Type t3 = Type.GetType("CH03Demo.Student");//不同程序集,则还需传入程序集名Type t4 = Type.GetType("ClassLibrary1.Class1,ClassLibrary1");}
Activator类的应用
//Activator类主要用于创建对象的实例
Type type = typeof(UserInfo);
UserInfo userInfo=(UserInfo)Activator.CreateInstance(type);
Assembly类的应用
  • 对于程序集的限定名称使用小结
    • 程序集的显示名称,可通过Assembly.FullNameAssembly.GetName().FullName(即AssemblyName.FullName) 两种方式获取,这种获取的名称,一般是作为 Assembly.Load()的标准参数值
    • 类型的程序集限定名,可通过Type类中的AssemblyQualifiedName属性获取(通常作为Type.GetType()方法中的参数值), 相较于Assembly.FullName,名称格式上多了 Type.FullName 这一部分
  • Assembly类中的常用方法
    • Assembly.Load()方法接收一个String或AssemblyName类型作为参数,这个参数需要程序集的强名称
    • Assembly.LoadFrom() 根据程序集的文件名或路径,加载程序集;这个方法会加载此程序集引用的其他程序集
    • Assembly.LoadFile() 加载指定路径上的程序集文件内容,和上面方法的不同之处是这个方法不会加载此程序集引用的其他程序集

程序集的强名称:是程序集的FullName(具有名称,版本,语言,公钥标记);
程序集的弱命名:只有程序集名称而没有版本,语言和公钥标记;平常我们创建的一个类库,如果没有特殊操作都属于是是弱名称程序集
Load(“强名称程序集”)查找程序集的顺序:首先它会去全局程序集缓存查找,然后到应用程序的根目录查找,最后会到应用程序的私有路径查找。
Load(“弱名称程序集”)查找程序集的顺序:首先到应用程序的根目录查找,最后会到应用程序的私有路径查找。

 /// <summary>/// 程序集加载的3种方式/// </summary>static void Fun12(){//方式1:将ClassLibrary1.dll放在根目录下,传入程序集的简单名称,即可加载Assembly assembly1 = Assembly.Load("ClassLibrary1");//方式2:传入ClassLibrary1.dll的文件路径,会加载ClassLibrary1.dll依赖的程序集Assembly assembly2 = Assembly.LoadFrom("ClassLibrary1.dll");//方式3:传入ClassLibrary1.dll的文件路径,不会加载ClassLibrary1.dll依赖的程序集Assembly assembly3 = Assembly.LoadFile(AppDomain.CurrentDomain.BaseDirectory+ "ClassLibrary1.dll");}
Module类的应用
什么是模块

System.Reflection.Module类‌是C#中的一个重要类,它提供了有关程序集中模块的信息和功能。这个类的作用和使用方法在深入探讨模块类时会被详细讨论,通过一些示例代码来展示它的功能。在C#中,模块可以是一个源代码文件、一个编译后的文件(DLL或EXE)或者一个动态生成的程序集。每个模块都有自己的元数据和IL代码,而System.Reflection.Module类提供了访问和操作这些模块的能力。通过Module类,我们可以获取模块的元数据、类型信息、成员信息以及执行模块中的代码。

模块的属性

在这里插入图片描述

/// <summary>
/// 模块
/// </summary>
static void Fun13()
{//类型对象Type t1 = typeof(Student);Type t2 = typeof(string);//模块Module module1 = t1.Module;Module module2 = t2.Module;//基本属性Console.WriteLine("模块名:"+module1.Name);Console.WriteLine("完全限定名:" + module1.FullyQualifiedName);Console.WriteLine("程序集名:" + module1.Assembly.FullName);Console.WriteLine("-----------------------------");Console.WriteLine("模块名:" + module2.Name);Console.WriteLine("完全限定名:" + module2.FullyQualifiedName);Console.WriteLine("程序集名:" + module2.Assembly.FullName);
}
AppDomain类的应用

前提:

从.NET Core开始,不再支持运行时创建其他AppDomain(即仅可在.NET Framework下支持创建其他AppDomain)。

因此以下部分内容仅在.NET Framework上有效。

官方说明:.NET Framework 技术在 .NET 6 及更高版本上不可用

AppDomain应用程序域:一组程序集的逻辑容器,CLR创建的第一个AppDomain称为默认AppDomain,仅在进程终止时销毁。

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

在这里插入图片描述

/// <summary>
/// 应用程序域
/// </summary>
static void Fun14()
{//获取当前应用程序域中的所有程序集Assembly[] assemblies = AppDomain.CurrentDomain.GetAssemblies();foreach (Assembly assembly in assemblies){Console.WriteLine(assembly.FullName);}
}
通过反射创建对象
/// <summary>
/// 反射:创建对象的3种方式
/// </summary>
static void Fun10()
{//类型对象Type t = typeof(Class1);#region 方式1:通过Invoke 执行构造函数//获取构造函数对象ConstructorInfo ci1 = t.GetConstructor(new Type[] { });ConstructorInfo ci2 = t.GetConstructor(new Type[] {typeof(int) });//调用构造函数创建实例object obj1= ci1.Invoke(new object[] { });object obj2 = ci2.Invoke(new object[] {15 });//类型装换Class1 c1=obj1 as Class1;Class1 c2 = obj2 as Class1;#endregion#region 方式2:通过Assembly 创建实例//程序集对象Assembly assembly = Assembly.Load("ClassLibrary1");//创建实例object obj3 = assembly.CreateInstance("ClassLibrary1.Class1",true);//类型转换Class1 c3= obj3 as Class1;#endregion#region 方式3:通过 Activator 创建实例//创建对象object obj4 = Activator.CreateInstance(t);object obj5 = Activator.CreateInstance("ClassLibrary1", "ClassLibrary1.Class1");//类型转换Class1 c4 = obj4 as Class1;Class1 c5 = obj5 as Class1;#endregion
}
通过反射获取对象成员

在这里插入图片描述

/// <summary>
/// 反射:获取 方法、属性、字段
/// </summary>
static void Fun11()
{//类型对象Type t = typeof(Student);//目标实体object obj= Activator.CreateInstance(t);#region 反射:获取方法//获取方法对象MethodInfo mi= t.GetMethod("Show");//执行方法mi.Invoke(obj, new object[] {25 });//Invoke 调用静态方法,对象可以为null ,形如 //methodInfo.Invoke(null, new object[] { "hello" })#endregion#region 反射:获取属性//获取属性对象PropertyInfo pi = t.GetProperty("Name");pi.SetValue(obj, "张三");//调用属性string name= pi.GetValue(obj) as string;Console.WriteLine("姓名:"+name);#endregion#region 反射:获取字段//获取属性对象FieldInfo fi = t.GetField("Phone");fi.SetValue(obj, "13523983345");//调用字段string phone = fi.GetValue(obj) as string;Console.WriteLine("电话:" + phone);#endregion
}

反射的应用

数据库辅助类反射
常规情况下:编写固定DBHelper类
using System;
using System.Collections.Generic;
using System.Data.SqlClient;
using System.Data;
using System.Linq;
using System.Text;
using System.Threading.Tasks;namespace CH03Demo
{internal class DBHelper{private static readonly string _connectionString = "server=.;database=test;uid=sa;pwd=sa";/// <summary>/// 执行增删改/// </summary>/// <param name="sql"></param>/// <returns></returns>public int ExecuteNonQuery(string sql,params SqlParameter[] parameters){//略return 0;}/// <summary>/// 执行查询/// </summary>/// <param name="sql"></param>/// <returns></returns>public DataTable ExecuteTable(string sql,params SqlParameter[] parameters){//略return null;}}
}
反射+配置文件动态实现

1.创建一个接口

using System;
using System.Collections.Generic;
using System.Data.SqlClient;
using System.Data;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
using System.Data.Common;namespace CH03Demo
{/// <summary>/// 数据访问接口/// </summary>internal interface IDBHelper{/// <summary>/// 执行增删改/// </summary>/// <param name="sql"></param>/// <returns></returns>int ExecuteNonQuery(string sql, params DbParameter[] parameters);/// <summary>/// 执行查询/// </summary>/// <param name="sql"></param>/// <returns></returns>DataTable ExecuteTable(string sql, params DbParameter[] parameters);}
}

2.实现接口

using System;
using System.Collections.Generic;
using System.Data;
using System.Data.Common;
using System.Linq;
using System.Text;
using System.Threading.Tasks;namespace CH03Demo
{/// <summary>/// sqlserver 数据库的数据访问类/// </summary>internal class SqlServerDBHelper : IDBHelper{public int ExecuteNonQuery(string sql, params DbParameter[] parameters){//代码略return 0 ;}public DataTable ExecuteTable(string sql, params DbParameter[] parameters){//代码略return null;}}
}

3.增加配置文件

<?xml version="1.0" encoding="utf-8" ?>
<configuration><startup> <supportedRuntime version="v4.0" sku=".NETFramework,Version=v4.7.2" /></startup><appSettings><add key="DBType" value="SqlServer"/></appSettings>
</configuration>

4.过反射+配置文件 调用 数据库执行语句的方法

/// <summary>
/// 通过反射创建数据辅助类对象
/// </summary>
static void Fun15()
{string fullName = ConfigurationManager.AppSettings["DBType"].ToString();IDBHelper dbHelper = (IDBHelper)Assembly.Load("DBHelper").CreateInstance(fullName);var data1 = dbHelper.ExecuteNonQuery("delete from student where studentNo=@studentNo", new SqlParameter("@studentNo", "GCKJ101"));var data2 = dbHelper.ExecuteTable("select * from student");Console.WriteLine(data1);Console.WriteLine(data2.Rows.Count);
}

目录结构:

在这里插入图片描述

本章总结

课后作业

1.通过反射查看int类 Type基本信息
2.通过反射调用int类的实例方法: CompareTo
3.通过反射调用int类的静态方法: Parse

相关文章:

CH03_反射

第3章&#xff1a;反射 本章目标 掌握反射的原理 熟悉反射的基本运用 本章内容 反射是什么 C# 编译运行过程 首先我们在VS点击编译的时候&#xff0c;就会将C#源代码编译成程序集 程序集以可执行文件 (.exe) 或动态链接库文件 (.dll) 的形式实现 程序集中包含有Microsoft …...

vue2侧边导航栏路由

<template><div><!-- :default-active"$route.path" 和index对应其路径 --><el-menu:default-active"active"class"el-menu-vertical-demo"background-color"#545c64"text-color"#fff"active-text-col…...

core 不可变类型 线程安全 record

当一个类型的对象在创建时被指定状态后&#xff0c;就不会再变化的对象&#xff0c;我们称之为不可变类型。这种类型是线程安全的&#xff0c;不需要进行线程同步&#xff0c;非常适合并行计算的数据共享。它减少了更新对象会引起各种bug的风险&#xff0c;更为安全。 System.D…...

linux之调度管理(8)-SMP cpu 的 psci启动

一、psci介绍 psci是arm提供的一套电源管理接口&#xff0c;当前一共包含0.1、0.2和1.0三个版本。它可被用于以下场景&#xff1a; &#xff08;1&#xff09;cpu的idle管理 &#xff08;2&#xff09;cpu hotplug以及secondary cpu启动 &#xff08;3&#xff09;系统shutdo…...

review-消息中间件MQ

RabbitMQ RabbitMQ&#xff0c;作为当今流行的开源消息代理软件&#xff0c;以其卓越的可靠性、灵活性和易用性在微服务架构和分布式系统中扮演着至关重要的角色。它不仅能够确保消息在不同系统组件间的高效传递&#xff0c;还能通过其高级消息队列协议&#xff08;AMQP&#x…...

leetcode400第N位数字

代码 class Solution {public int findNthDigit(int n) {int base 1;//位数int weight 9;//权重while(n>(long)base*weight){//300n-base*weight;base;weight*10;}//n111 base3 weight900;n--;int res (int)Math.pow(10,base-1)n/base;int index n%base;return String…...

前端网页开发学习(HTML+CSS+JS)有这一篇就够!

目录 HTML教程 ▐ 概述 ▐ 基础语法 ▐ 文本标签 ▐ 列表标签 ▐ 表格标签 ▐ 表单标签 CSS教程 ▐ 概述 ▐ 基础语法 ▐ 选择器 ▐ 修饰文本 ▐ 修饰背景 ▐ 透明度 ▐ 伪类 ▐ 盒子模型 ▐ 浮动 ▐ 定位 JavaScript教程 ▐ 概述 ▐ 基础语法 ▐ 函数 …...

CSS遮罩:mask

CSS属性 mask 允许使用者通过遮罩或者裁切特定区域的图片的方式来隐藏一个元素的部分或者全部可见区域。 // 一般用位图图片做遮罩 mask: url(~/assets/images/mask.png); mask-size: 100% 100%;// 使用 SVG 图形中的形状来做遮罩 mask: url(~/assets/images/mask.svg#star);…...

Swift闭包的本质

1 闭包的本质其实是一个引用类型&#xff1a;存储在堆空间上&#xff0c;由堆分配空间&#xff0c;且生命周期由ARC&#xff08;自动引用计数机制&#xff09;管理 2 捕获值&#xff1a;闭包会捕获上下文使用到的变量&#xff08;引用类型会保持引用关系&#xff09;&#xff…...

时代变迁对传统机器人等方向课程的巨大撕裂

2020年之后&#xff0c;全面转型新质课程规划&#xff0c;传统课程规划全部转为经验。 农耕-代表性生产关系-封建分配制度主要生产力-人力工业-代表性生产关系-资本分配制度工业分为机械时代&#xff0c;电气时代&#xff0c;信息时代&#xff1b;主要生产力-人力转为人脑&…...

【算法设计与分析实训】第1关:求序列的最大字段和

务描述 本关任务&#xff1a;编写用动态规划解决最大字段和问题。 相关知识 为了完成本关任务&#xff0c;你需要掌握&#xff1a;动态规划。 编程要求 给定由n个整数&#xff08;可能为负数&#xff09;组成的序列&#xff1a;a1,a2,……,an, 求该序列的最大子段和。当所有整…...

【澜舟科技-注册/登录安全分析报告】

前言 由于网站注册入口容易被机器执行自动化程序攻击&#xff0c;存在如下风险&#xff1a; 暴力破解密码&#xff0c;造成用户信息泄露&#xff0c;不符合国家等级保护的要求。短信盗刷带来的拒绝服务风险 &#xff0c;造成用户无法登陆、注册&#xff0c;大量收到垃圾短信的…...

【读书笔记-《网络是怎样连接的》- 7】Chapter3_2 路由器

本篇继续介绍路由器及其转发过程。 1 路由器内部结构 路由器内部结构图如图所示。 即主要包含左侧的包转发模块和右侧的端口模块。转发模块负责查找包的发送目的地&#xff0c;端口模块完成包的发送。通过安装不同的硬件&#xff0c;转发模块不仅可以支持以太网&#xff0c;也…...

Android Activity 基础接口知识和常见问题

Activity 知识点及问题点 接口onMultiWindowModeChangedonConfigurationChanged 常见问题Android解决点击桌面图标&#xff0c;就重新启动应用程序问题 接口 onMultiWindowModeChanged 定义 onMultiWindowModeChanged是Android中Activity类的一个回调方法。它会在活动&#xf…...

利用python 检测当前目录下的所有PDF 并转化为png 格式

以下是一个完整的 Python 脚本&#xff0c;用于检测当前目录下的所有 PDF 文件并将每一页转换为 PNG 格式&#xff1a; import os from pdf2image import convert_from_path# 设置输出图像的 DPI&#xff08;分辨率&#xff09; DPI 300# 获取当前目录 current_directory os…...

解决 Spring Boot 中 `Ambiguous mapping. Cannot map ‘xxxController‘ method` 错误

前言 在使用 Spring Boot 开发 Web 应用时&#xff0c;经常会遇到各种各样的错误。其中一种常见的错误是 Ambiguous mapping. Cannot map ‘testController‘ method。本文将详细介绍这个错误的原因及解决方法&#xff0c;帮助开发者快速定位并解决问题。 错误解释 这个错误…...

C++ 函数返回值优化

本文中部分内容来自下面的文章&#xff0c;还有一部分来自智谱清言 C 返回值优化_c 局部变量返回优化-CSDN博客 elision:省略 copy elision&#xff1a;拷贝省略 RVO (Return Value Optimization)&#xff1a;返回值优化 ------ 我最近也遇到了上面博文中说到的问题&…...

c++源码阅读__ThreadPool__正文阅读

一. 简介 本章我们开始阅读c git 高星开源项目ThreadPool, 这是一个纯c的线程池项目, 并且代码量极小, 非常适合新手阅读 git地址: progschj / ThreadPool 二. 前提知识 为了面对不同读者对c掌握情况不同的情况, 这里我会将基本上稍微值得一说的前提知识点, 全部专门写成一篇…...

关于ES的查询

查询结果那么多字段都是什么&#xff1f; 为什么会提到这个问题呢&#xff0c;因为默认ES查询的结果会有很多信息&#xff0c;我们可能并不希望要那么多数据&#xff0c;所以你需要了解这些字段都表示什么&#xff0c;并正确的返回和使用它们。 took– Elasticsearch 运行查询…...

数据结构初识

目录 1.初识 2.时间复杂度 常见时间复杂度举例&#xff1a; 3.空间复杂度 4.包装类&简单认识泛型 4.1装箱和拆箱 5.泛型 6.泛型的上界 7.泛型方法 8.List接口 1.初识 1.多画图 2.多思考 3.多写代码 4.多做题 牛客网-题库/在线编程/剑指offer 算法篇&#xff1a…...

基于距离变化能量开销动态调整的WSN低功耗拓扑控制开销算法matlab仿真

目录 1.程序功能描述 2.测试软件版本以及运行结果展示 3.核心程序 4.算法仿真参数 5.算法理论概述 6.参考文献 7.完整程序 1.程序功能描述 通过动态调整节点通信的能量开销&#xff0c;平衡网络负载&#xff0c;延长WSN生命周期。具体通过建立基于距离的能量消耗模型&am…...

python/java环境配置

环境变量放一起 python&#xff1a; 1.首先下载Python Python下载地址&#xff1a;Download Python | Python.org downloads ---windows -- 64 2.安装Python 下面两个&#xff0c;然后自定义&#xff0c;全选 可以把前4个选上 3.环境配置 1&#xff09;搜高级系统设置 2…...

PPT|230页| 制造集团企业供应链端到端的数字化解决方案:从需求到结算的全链路业务闭环构建

制造业采购供应链管理是企业运营的核心环节&#xff0c;供应链协同管理在供应链上下游企业之间建立紧密的合作关系&#xff0c;通过信息共享、资源整合、业务协同等方式&#xff0c;实现供应链的全面管理和优化&#xff0c;提高供应链的效率和透明度&#xff0c;降低供应链的成…...

ESP32 I2S音频总线学习笔记(四): INMP441采集音频并实时播放

简介 前面两期文章我们介绍了I2S的读取和写入&#xff0c;一个是通过INMP441麦克风模块采集音频&#xff0c;一个是通过PCM5102A模块播放音频&#xff0c;那如果我们将两者结合起来&#xff0c;将麦克风采集到的音频通过PCM5102A播放&#xff0c;是不是就可以做一个扩音器了呢…...

VTK如何让部分单位不可见

最近遇到一个需求&#xff0c;需要让一个vtkDataSet中的部分单元不可见&#xff0c;查阅了一些资料大概有以下几种方式 1.通过颜色映射表来进行&#xff0c;是最正规的做法 vtkNew<vtkLookupTable> lut; //值为0不显示&#xff0c;主要是最后一个参数&#xff0c;透明度…...

MySQL中【正则表达式】用法

MySQL 中正则表达式通过 REGEXP 或 RLIKE 操作符实现&#xff08;两者等价&#xff09;&#xff0c;用于在 WHERE 子句中进行复杂的字符串模式匹配。以下是核心用法和示例&#xff1a; 一、基础语法 SELECT column_name FROM table_name WHERE column_name REGEXP pattern; …...

Unity | AmplifyShaderEditor插件基础(第七集:平面波动shader)

目录 一、&#x1f44b;&#x1f3fb;前言 二、&#x1f608;sinx波动的基本原理 三、&#x1f608;波动起来 1.sinx节点介绍 2.vertexPosition 3.集成Vector3 a.节点Append b.连起来 4.波动起来 a.波动的原理 b.时间节点 c.sinx的处理 四、&#x1f30a;波动优化…...

企业如何增强终端安全?

在数字化转型加速的今天&#xff0c;企业的业务运行越来越依赖于终端设备。从员工的笔记本电脑、智能手机&#xff0c;到工厂里的物联网设备、智能传感器&#xff0c;这些终端构成了企业与外部世界连接的 “神经末梢”。然而&#xff0c;随着远程办公的常态化和设备接入的爆炸式…...

【分享】推荐一些办公小工具

1、PDF 在线转换 https://smallpdf.com/cn/pdf-tools 推荐理由&#xff1a;大部分的转换软件需要收费&#xff0c;要么功能不齐全&#xff0c;而开会员又用不了几次浪费钱&#xff0c;借用别人的又不安全。 这个网站它不需要登录或下载安装。而且提供的免费功能就能满足日常…...

站群服务器的应用场景都有哪些?

站群服务器主要是为了多个网站的托管和管理所设计的&#xff0c;可以通过集中管理和高效资源的分配&#xff0c;来支持多个独立的网站同时运行&#xff0c;让每一个网站都可以分配到独立的IP地址&#xff0c;避免出现IP关联的风险&#xff0c;用户还可以通过控制面板进行管理功…...