【C#学习笔记】引用类型(2)
文章目录
- Object
- Equals
- GetType
- ToString
- GetHashCode
- string
- 逐字文本
- 复合格式字符串
- 字符串内插
- StringBuilder
- StringBuilder 的工作原理
- StringBuilder提供的方法
- 访问字符
- 迭代字符
- 查询字符
- dynamic
Object
支持 .NET 类层次结构中的所有类,并为派生类提供低级别服务。 这是所有 .NET 类的最终基类;它是类型层次结构的根。
简单来说,它是万能的,它是所有类型的父类。因为.NET 中的所有类都派生自 Object,因此 Object 类中定义的每个方法都可用于系统中的所有对象。
在 Object 类中提供了 4 个常用的方法,即 Equals、GetHashCode、GetType 以及 ToString 方法。
Equals
int a = 1;
int b = 1;
if (a.Equals(b))
{Debug.Log("111"); -- 输出111
}Person a = new Person();
Person b = 1;
if (a.Equals(b))
{Debug.Log("111"); -- 输出111
}
Equals是相等判断,与==相同,对于值类型的判断是否值相同,而对于引用类型则判断是否引用相同。
GetType
用于获取当前实例的类型,返回值为 System.Type 类型。GetType 方法不含任何参数,是非静态方法。
使用Object.GetType()来返回对象的类型x.GetType()和typeof()的区别是,GetType()是object的方法,可以用于所有变量后,以获取它们的类型。和typeof(xxx)内部只能放类型而不能放变量名。
ToString
将对象转换为其字符串表示形式,使其适合显示。
上述三种方法是可以重载,重写的。
GetHashCode
返回对象的哈希值,每个对象的哈希值都是固定的。该方法不含有任何参数,并且不是静态方法,因此需要使用实例来调用该方法。由于该方法是在 Object 类中定义的,因此任何对象都可以直接调用该方法。
string
字符串类型string
表示零个或多个 Unicode 字符的序列。 比较特殊的是,尽管string是引用类型,但是使用==
来比较时却是值类型的比较。需要注意的是,string是不可变类型,这意味着每次我们修改字符串string的操作实际上都是创建了一个新的字符串。
逐字文本
使用@
关键字标识该字符串为逐字文本,将不会识别转义符
@"""Ahoy!"" cried the captain." // "Ahoy!" cried the captain.
@"c:\Docs\Source\a.txt" // rather than "c:\\Docs\\Source\\a.txt"
复合格式字符串
使用String.Format方法将变量复合到格式项内,其中括号内的数字代表了对应的变量索引,而:
后面代表变量的指定类型。
String.Format("Name = {0}, hours = {1:hh}", name, DateTime.Now);
多个相同索引也可以同时存在
string multiple = String.Format("0x{0:X} {0:E} {0:N}",Int64.MaxValue);
Console.WriteLine(multiple);
// The example displays the following output:
// 0x7FFFFFFFFFFFFFFF 9.223372E+018 9,223,372,036,854,775,807.00
字符串内插
使用$
关键字标识字符串内插
string name = "Mark";
var date = DateTime.Now;
// Composite formatting:
Console.WriteLine("Hello, {0}! Today is {1}, it's {2:HH:mm} now.", name, date.DayOfWeek, date);
// String interpolation:
Console.WriteLine($"Hello, {name}! Today is {date.DayOfWeek}, it's {date:HH:mm} now.");
(更多用法请阅读官方文档,尤其是string类的方法,尴尬的是string类的方法特别多,但是String大多数情况下却被StringBuilder完爆)
StringBuilder
使用命名空间System.Text来引入该类型
StringBuilder可以说是String的完全上位替代,这是一个可变的字符串类。 可变性意味着,创建类的实例后,可以通过追加、删除、替换或插入字符对其进行修改。由于String每次修改时都需要产生一个新的字符串,这也就导致了对其进行操作可能产生的一些内存消耗问题。所以对于需要操作的字符串,我们会选择StringBuilder,不过性能取决于字符串的大小、要为新字符串分配的内存量、执行代码的系统以及操作的类型。
请考虑在以下情况下使用 String 类:
- 当代码对字符串进行的更改数较小时。 在这些情况下, StringBuilder 可能会提供微不足道的性能改进,因此可以直接使用String。
- 执行固定数量的串联操作时,尤其是字符串文本。 在这种情况下,编译器可能会将串联操作合并到单个操作中。
- 在生成字符串时必须执行广泛的搜索操作时。 类 StringBuilder 缺少搜索方法,例如 IndexOf 或 StartsWith。 对于这些操作,必须将 StringBuilder对象转换为String , 这可以抵消使用 StringBuilder的性能优势。 有关详细信息,请参阅 在 StringBuilder 对象中搜索文本 部分。
请考虑在以下情况下使用 StringBuilder 类:
- 如果预期代码在设计时对字符串进行未知数量的更改, 例如,使用循环连接包含用户输入的随机数量的字符串时。
- 预期代码对字符串进行大量更改时。
总的来说,一些短平快的操作,或者需要涉及串联字符串,或是查询整个字符串的时候更适合String。如果一些需要频繁更改,或者动态修改的字符串更适合StringBuilder
StringBuilder 的工作原理
StringBuilder是如何动态修改字符串长度的?假设有一个字符串长度为12的stringBuilder,如果我们需要Append成一个length =20的字符串,那么首先StringBuilder会把字符串丢入这个长度为16的缓冲区,当长度超过缓冲区的长度时,则意味着需要更大的缓冲区,则StringBuilder会找一个缓冲区大小翻一倍的新缓冲区(现在是32了)。然后把旧缓冲区的值复制过去,并加入那些超过原长度而未被存入旧缓冲区的新字符。如果还是超过就重复上述过程,直到存储完毕为止。(如果超出了设定最大容量或者内存没有多余空间了则报错)
public class Example
{public static void Main(){StringBuilder sb = new StringBuilder();ShowSBInfo(sb);sb.Append("This is a sentence.");ShowSBInfo(sb);for (int ctr = 0; ctr <= 10; ctr++) {sb.Append("This is an additional sentence.");ShowSBInfo(sb);} }private static void ShowSBInfo(StringBuilder sb){foreach (var prop in sb.GetType().GetProperties()) {if (prop.GetIndexParameters().Length == 0)Console.Write("{0}: {1:N0} ", prop.Name, prop.GetValue(sb));}Console.WriteLine();}
}
// The example displays the following output:
// Capacity: 16 MaxCapacity: 2,147,483,647 Length: 0
// Capacity: 32 MaxCapacity: 2,147,483,647 Length: 19
// Capacity: 64 MaxCapacity: 2,147,483,647 Length: 50
// Capacity: 128 MaxCapacity: 2,147,483,647 Length: 81
// Capacity: 128 MaxCapacity: 2,147,483,647 Length: 112
// Capacity: 256 MaxCapacity: 2,147,483,647 Length: 143
// Capacity: 256 MaxCapacity: 2,147,483,647 Length: 174
// Capacity: 256 MaxCapacity: 2,147,483,647 Length: 205
// Capacity: 256 MaxCapacity: 2,147,483,647 Length: 236
// Capacity: 512 MaxCapacity: 2,147,483,647 Length: 267
// Capacity: 512 MaxCapacity: 2,147,483,647 Length: 298
// Capacity: 512 MaxCapacity: 2,147,483,647 Length: 329
// Capacity: 512 MaxCapacity: 2,147,483,647 Length: 360
StringBuilder提供的方法
可以进行单独的方法调用并忽略返回值,如以下示例所示。
using System;
using System.Text;public class Example
{public static void Main(){StringBuilder sb = new StringBuilder();sb.Append("This is the beginning of a sentence, "); sb.Replace("the beginning of ", ""); //替换所有字符A为字符Bsb.Insert(sb.ToString().IndexOf("a ") + 2, "complete ");sb.Replace(",", "."); Console.WriteLine(sb.ToString()); //只有在必须接受string类型时再转化为string}
}
// The example displays the following output:
// This is a complete sentence.
可以在单个语句中调用一系列方法。 如果要编写链接连续操作的单个语句,这可以很方便。 以下示例将上一示例中的三个方法调用合并到一行代码中。
using System;
using System.Text;public class Example
{public static void Main(){StringBuilder sb = new StringBuilder("This is the beginning of a sentence, ");sb.Replace("the beginning of ", "").Insert(sb.ToString().IndexOf("a ") + 2, "complete ").Replace(",", ".");Console.WriteLine(sb.ToString());}
}
// The example displays the following output:
// This is a complete sentence.
访问字符
访问字符在我看来是StringBuilder的一大痛点。StringBuilder没有提供访问其内部字符的方法,而想要访问其中的某个字符却没那么简单,因为缓冲区的内存是不连续的,如果想要找到某个字符,我们就需要遍历整个缓冲区。因此较好的方法是通过ToString
方法将其转化为String
类型之后使用char[Index]
去访问对应字符。
迭代字符
当迭代StringBuilder的字符的时候,应当注意:即使对于大型“区块” StringBuilder 对象,使用 Chars[] 属性进行基于索引的访问一个或少量字符的性能影响可以忽略不计;通常,这是 一个 O (n) 操作。 当迭代 StringBuilder 对象中的字符时,会对性能造成显著的影响,这是 O(n^2) (遍历n个字符*缓存区容量n)操作。
因此当我们需要迭代StringBuilder的字符时,较好的方法还是通过ToString
方法将其转化为String
类型之后使用char[Index]
去访问对应字符。
查询字符
在StringBuilder中并没有提供查询字符的方法,想实现查询字符,也只能转为String
,然后用String.IndexOf
或者String.StartWith
方法查找对应的字符或字符串。
由于String和StringBuilder提供的方法太多,因此此处无法深入,推荐阅读官方文档
dynamic
最后稍微了解一下dynamic动态类型。dynamic本身也是一个Object。在大多数情况下,dynamic 类型与 object 类型的行为类似。 具体而言,任何非 Null 表达式都可以转换为 dynamic 类型。 dynamic 类型与 object 的不同之处在于,编译器不会对包含类型 dynamic 的表达式的操作进行解析或类型检查。 编译器将有关该操作信息打包在一起,之后这些信息会用于在运行时评估操作。 在此过程中,dynamic 类型的变量会编译为 object 类型的变量。 因此,dynamic 类型只在编译时存在,在运行时则不存在。
dynamic的特殊就在于它是不会在编译时被解析的,因此不会进行任何的安全检查。因此即使定义了错误的dynamic,编译器也不会报错。而只有在运行时出现错误才会报错。
其次,由于dynamic的本质是Object,它也可以和任何类型进行隐式转换。而这也是它的最大优点。例如:
dynamic x = 10; //此时x为int
x = x + "Hello"; // x 现在是一个字符串
x = x.ToUpper(); // x 仍然是一个字符串
x = 8; // x 现在保存的是一个int类型的变量
x = x * 8; // x现在仍然是int
在上例中,我们定义x为dynamic,可以看出来它对应应当是int型,然后我们将其转化为字符串,然后对字符串进行操作。随后又把x赋值为了int类型的。x在整个程序中动态的变动它的类型。而这是var做不到的,因为var关键字下编译器会自动地推断它的类型,那它就是一个静态类,一般没有显示转换的前提下我们是不能进行类型转换的。
如果只用静态类想实现则需要新创建其中设计到的静态类型对应的变量,而如果想让静态类型变量x实现上述过程只会更加复杂:
int x=10;
String y = x.ToString();
y=y+"Hello";
y = y.ToUpper();
x = 8;
x = x* 8
而dynamic的动态类型实际和某些脚本语言的变量性质是非常相似的。因此我们也使用dynamic类型来与脚本语言进行对接。
相关文章:

【C#学习笔记】引用类型(2)
文章目录 ObjectEqualsGetTypeToStringGetHashCode string逐字文本复合格式字符串字符串内插 StringBuilderStringBuilder 的工作原理StringBuilder提供的方法访问字符迭代字符查询字符 dynamic Object 支持 .NET 类层次结构中的所有类,并为派生类提供低级别服务。…...
【Rust 基础篇】Rust类函数宏:代码生成的魔法
导言 Rust是一门现代的、安全的系统级编程语言,它提供了丰富的元编程特性,其中类函数宏(Function-Like Macros)是其中之一。类函数宏允许开发者创建类似函数调用的宏,并在编译期间对代码进行生成和转换。在本篇博客中…...

Spring-1-透彻理解Spring XML的Bean创建--IOC
学习目标 上一篇文章我们介绍了什么是Spring,以及Spring的一些核心概念,并且快速快发一个Spring项目,实现IOC和DI,今天具体来讲解IOC 能够说出IOC的基础配置和Bean作用域 了解Bean的生命周期 能够说出Bean的实例化方式 一、Bean的基础配置 …...

【JAVA】类和对象
作者主页:paper jie的博客 本文作者:大家好,我是paper jie,感谢你阅读本文,欢迎一建三连哦。 本文录入于《JAVASE语法系列》专栏,本专栏是针对于大学生,编程小白精心打造的。笔者用重金(时间和精…...

jenkins准备
回到目录 jenkins是一个开源的、提供友好操作界面的持续集成(CI)工具,主要用于持续、自动的构建/测试软件项目、监控外部任务的运行。Jenkins用Java语言编写,可在Tomcat等流行的servlet容器中运行,也可独立运行。通常与版本管理工具(SCM)、构…...

【Rust】Rust学习
文档:Rust 程序设计语言 - Rust 程序设计语言 简体中文版 (bootcss.com) 墙裂推荐这个文档 第一章入门 入门指南 - Rust 程序设计语言 简体中文版 第二章猜猜看游戏 猜猜看游戏教程 - Rust 程序设计语言 简体中文版 (bootcss.com) // 导入库 use std::io; use s…...
Linux 常用命令之配置环境变量 PATH
PATH是系统环境变量中的一种,同时将一些二进制文件的绝对路径追加进去,则在系统终端中可以发现这些路径下的文件。 一. 环境变量设置 export PATH<二进制文件的绝对路径>:$PATH 以下为结合实际例子的操作 1、临时设置 打开一个终端执行如下命令 e…...

flask-----蓝图
1.引入蓝图 flask都写在一个文件中,项目这样肯定不行,会导致循环导入的问题,分目录,分包,使用蓝图划分目录。 2.使用蓝图 步骤如下: -1 实例化得到一个蓝图对象-order_blueBlueprint(order,__name__,tem…...

学习左耳听风栏目90天——第一天 1-90(学习左耳朵耗子的工匠精神,对技术的热爱)【洞悉技术的本质,享受科技的乐趣】
洞悉技术的本质,享受科技的乐趣 第一篇,我的感受就是 耗叔是一个热爱技术,可以通过代码找到快乐的技术人。 作为it从业者,我们如何可以通过代码找到快乐呢?这是一个问题? 至少目前,我还没有这种…...
后端登录安全的一种思路
PS:作者是小白能接触到的就只会这样写。勿喷。 前提 思路: 结合io流将登录token存储到配置文件中,不将token存储到浏览器端,从而避免盗取。 下面jwt的学习可以参考下这个: JWT --- 入门学习_本郡主是喵的博客-CSDN博客 JWT工具类 Component public class JWTtU…...

【深度学习_TensorFlow】激活函数
写在前面 上篇文章我们了解到感知机使用的阶跃函数和符号函数,它们都是非连续,导数为0的函数: 建议回顾上篇文章,本篇文章将介绍神经网络中的常见激活函数,这些函数都是平滑可导的,适合于梯度下降算法。 写…...

机器学习笔记之优化算法(七)线搜索方法(步长角度;非精确搜索;Wolfe Condition)
机器学习笔记之优化算法——线搜索方法[步长角度,非精确搜索,Wolfe Condition] 引言回顾: Armijo \text{Armijo} Armijo准则及其弊端 Glodstein \text{Glodstein} Glodstein准则及其弊端 Wolfe Condition \text{Wolfe Condition} Wolfe Condi…...

十四.redis哨兵模式
redis哨兵模式 1.概述2.测试3.哨兵模式优缺点 redis哨兵模式基础是主从复制 1.概述 主从切换的技术方法:当主节点服务器宕机后,需要手动把一台从服务器切换为主服务器,这就需要人工干预,费时费力,还会造成一段时间内服…...

采用UWB技术开发的智慧工厂人员定位系统源码【UWB定位基站、卡牌】
UWB (ULTRA WIDE BAND, UWB) 技术是一种无线载波通讯技术,它不采用正弦载波,而是利用纳秒级的非正弦波窄脉冲传输数据,因此其所占的频谱范围很宽。UWB定位系统依托在移动通信,雷达,微波电路,云计算与大数据…...

当你软件测试遇上加密接口,是不是就不能测了?
相信大家在工作中做接口测试的时候,肯定会遇到一个场景,那就是你们的软件,密码是加密存储的。 那么这样的话,我们在执行接口的时候,对于密码的处理就开始头疼了。 所以,本文将使用jmeter这款java开源的接…...
Flink
Flink(Apache Flink)是一个开源的分布式流处理引擎和批处理框架。它是由 Apache 软件基金会维护的项目,旨在处理大规模数据的实时流式处理和批处理任务。Flink 提供了强大的流处理和批处理功能,具有低延迟、高吞吐量和高容错性&am…...
python入门常用操作
python常用操作 1、ndarry数组的切片2、print用法2.1格式化输出format2.2字符串格式化输出 3、均值滤波函数 1、ndarry数组的切片 例如一个5列的ndarry数组,想要获取第2列和第3列数据,可以用 #(1)用法1 data[:,1:3],…...
SpringBoot复习:(21)自定义ImportBeanDefinitionRegistrar
要达到的目的:将某个包下使用了某个自定义注解(比如MyClassMapper)的类注册到Spring 容器。 一、自定义注解: package com.example.demo.service;import java.lang.annotation.Retention; import java.lang.annotation.RetentionPolicy;Rete…...

小黑子—JavaWeb:第五章 - JSP与会话跟踪技术
JavaWeb入门5.0 1. JSP1.1 JSP快速入门1.2 JSP原理1.3 JSP脚本1.3.1 JSP缺点 1.4 EL 表达式1.5 JSTL 标签1.5.1 JSTL 快速入门1.5.1 - I JSTL标签if1.5.1 - II JSTL标签forEach 1.6 MVC模式1.7 三层架构1.8 实现案例1.8.1 环境准备1.8.2 查询所有1.8.3 添加数据1.8.4 修改1.8.4…...

Python - 【socket】 客户端client重连处理简单示例Demo(一)
一. 前言 在Python中,使用socket进行网络通信时,如果连接断开,可以通过以下步骤实现重连处理 二. 示例代码 1. 定义一个函数,用于建立socket连接 import socketdef connect_socket(host, port):while True:try:# 建立socket连…...
day52 ResNet18 CBAM
在深度学习的旅程中,我们不断探索如何提升模型的性能。今天,我将分享我在 ResNet18 模型中插入 CBAM(Convolutional Block Attention Module)模块,并采用分阶段微调策略的实践过程。通过这个过程,我不仅提升…...

使用分级同态加密防御梯度泄漏
抽象 联邦学习 (FL) 支持跨分布式客户端进行协作模型训练,而无需共享原始数据,这使其成为在互联和自动驾驶汽车 (CAV) 等领域保护隐私的机器学习的一种很有前途的方法。然而,最近的研究表明&…...

376. Wiggle Subsequence
376. Wiggle Subsequence 代码 class Solution { public:int wiggleMaxLength(vector<int>& nums) {int n nums.size();int res 1;int prediff 0;int curdiff 0;for(int i 0;i < n-1;i){curdiff nums[i1] - nums[i];if( (prediff > 0 && curdif…...
iOS性能调优实战:借助克魔(KeyMob)与常用工具深度洞察App瓶颈
在日常iOS开发过程中,性能问题往往是最令人头疼的一类Bug。尤其是在App上线前的压测阶段或是处理用户反馈的高发期,开发者往往需要面对卡顿、崩溃、能耗异常、日志混乱等一系列问题。这些问题表面上看似偶发,但背后往往隐藏着系统资源调度不当…...

使用LangGraph和LangSmith构建多智能体人工智能系统
现在,通过组合几个较小的子智能体来创建一个强大的人工智能智能体正成为一种趋势。但这也带来了一些挑战,比如减少幻觉、管理对话流程、在测试期间留意智能体的工作方式、允许人工介入以及评估其性能。你需要进行大量的反复试验。 在这篇博客〔原作者&a…...
OD 算法题 B卷【正整数到Excel编号之间的转换】
文章目录 正整数到Excel编号之间的转换 正整数到Excel编号之间的转换 excel的列编号是这样的:a b c … z aa ab ac… az ba bb bc…yz za zb zc …zz aaa aab aac…; 分别代表以下的编号1 2 3 … 26 27 28 29… 52 53 54 55… 676 677 678 679 … 702 703 704 705;…...

WPF八大法则:告别模态窗口卡顿
⚙️ 核心问题:阻塞式模态窗口的缺陷 原始代码中ShowDialog()会阻塞UI线程,导致后续逻辑无法执行: var result modalWindow.ShowDialog(); // 线程阻塞 ProcessResult(result); // 必须等待窗口关闭根本问题:…...

热烈祝贺埃文科技正式加入可信数据空间发展联盟
2025年4月29日,在福州举办的第八届数字中国建设峰会“可信数据空间分论坛”上,可信数据空间发展联盟正式宣告成立。国家数据局党组书记、局长刘烈宏出席并致辞,强调该联盟是推进全国一体化数据市场建设的关键抓手。 郑州埃文科技有限公司&am…...

前端开发者常用网站
Can I use网站:一个查询网页技术兼容性的网站 一个查询网页技术兼容性的网站Can I use:Can I use... Support tables for HTML5, CSS3, etc (查询浏览器对HTML5的支持情况) 权威网站:MDN JavaScript权威网站:JavaScript | MDN...

Java后端检查空条件查询
通过抛出运行异常:throw new RuntimeException("请输入查询条件!");BranchWarehouseServiceImpl.java // 查询试剂交易(入库/出库)记录Overridepublic List<BranchWarehouseTransactions> queryForReagent(Branch…...