【C#学习笔记】引用类型(2)

news2024/11/15 5:58:24

在这里插入图片描述

文章目录

  • 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是如何动态修改字符串长度的?假设有一个字符串长度为16的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为字符B
      sb.Insert(sb.ToString().IndexOf("a ") + 2, "complete ");
      sb.Replace(",", "."); 
      Console.WriteLine(sb.ToString());
   }
}
// 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类型来与脚本语言进行对接。

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

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

相关文章

设计模式行为型——迭代器模式

什么是迭代器模式 迭代器模式&#xff08;Iterator Pattern&#xff09;属于行为型模式&#xff0c;其提供一种方法顺序访问一个聚合对象中的各种元素&#xff0c;而又不暴露该对象的内部表示&#xff0c;即不需要知道集合对象的底层表示。编程环境中非常常用的设计模式。 迭代…

leetcode每日一练-第102题-二叉树的层序遍历

一、思路 BFS 二、解题方法 通过广度优先搜索&#xff08;BFS&#xff09;的方式&#xff0c;按层遍历二叉树节点&#xff0c;并将每层的节点值保存在一个一维数组中&#xff0c;然后再将所有的一维数组存储在二维数组中&#xff0c;最后返回二维数组作为层序遍历的结果。 …

音视频--DTMF信号发送及检测

参考资料 https://zh.wikipedia.org/wiki/%E5%8F%8C%E9%9F%B3%E5%A4%9A%E9%A2%91https://www.cnblogs.com/lijingcheng/p/4454932.html 1. DTMF是什么 1.1 DTMF定义 双音多频信号&#xff08;英语&#xff1a;Dual-Tone Multi-Frequency&#xff0c;简称&#xff1a;DTMF&a…

当“国潮”遇见“双语” 以传承之心种下一颗文化的种子

看&#xff0c;活灵活现的纸片人在“跳舞”。光影的辉映下&#xff0c;两个形神兼备的“齐天大圣”究竟孰真孰假&#xff1f;舞台上&#xff0c;京西皮影非遗传承人王熙和5岁的Mona小朋友正在用双语为大家带来一段“真假美猴王”的好戏。生动的皮影造型和精彩的故事演绎看得台下…

【BEV感知】3-BEV开源数据集

3-BEV开源数据集 1 KITTI1.1 KITTI数据怎么采集?1.2 KITTI数据规模有多大?1.3 KITTI标注了哪些目标?1.4 转换矩阵1.5 标签文件 2 nuScenes2.1 nuScenes Vs KITTI2.2 标注文件 1 KITTI KITTI 1.1 KITTI数据怎么采集? 通过车载相机、激光雷达等传感器采集。 只提供了相机正…

bagging集成与boosting集成的区别是什么?

bagging集成与boosting集成的区别 区别一:数据方面 Bagging&#xff1a;对数据进行采样训练; Boosting&#xff1a;根据前一轮学习结果调整数据的重要性。 区别二:投票方面 Bagging&#xff1a;所有学习器平权投票; Boosting&#xff1a;对学习器进行加权投票。 区别三:…

接口相似数据结构复用率高?Apipost这招搞定!

在API设计和开发过程中&#xff0c;存在许多瓶颈&#xff0c;其中一个主要问题是在遇到相似数据结构的API时会产生重复性较多的工作&#xff1a;在每个API中都编写相同的数据&#xff0c;这不仅浪费时间和精力&#xff0c;还容易出错并降低API的可维护性。 为了解决这个问题&a…

排序八卦炉之冒泡、快排

文章目录 1.冒泡排序1.1代码实现1.2复杂度 2.快速排序2.1人物及思想介绍【源于百度】2.2hoare【霍尔】版本1.初识代码2.代码分析3.思其因果 3.相关博客 1.冒泡排序 1.1代码实现 //插入排序 O(N)~O(N^2) //冒泡排序 O(N)~O(N^2) //当数据有序 二者均为O(N) //当数据接近有序或…

linux大神Brendan Gregg 性能之巅 第二版(systems performance)阅读心得(第一章)

笔者从事某副省级市政务云系统运维7年&#xff0c;最近被下面这张图吸引开始阅读Brendan Gregg的《性能之巅 第二版》&#xff0c;下面将结合自己的运维经验一起来看看这本700多页的书。 1、绪论 1.1 系统性能 系统性能的影响因素涉及软件和硬件&#xff0c;CPU、内存、磁盘io…

9.物联网操作系统之软件定时器

一。软件定时器概念及应用 1.软件定时器定义 就是软件实现定时器。 2.FreeRTOS软件定时器介绍 如上图所示&#xff0c;Times的左边为设置定时器时间&#xff0c;设置方式可以为任务设置或者中断设置&#xff1b;Times的右边为定时器的定时相应&#xff0c;使用CalBack相应。 …

QMS质量管理系统是什么?

QMS质量管理系统是一种用于管理和优化企业质量管理的软件系统&#xff0c;在现代企业中&#xff0c;质量管理是非常重要的环节。 1. QMS系统的概念 QMS系统是一种用于管理和优化企业质量管理的软件系统。它可以帮助企业制定和实施质量管理策略、管理和控制质量过程、收集和分析…

RK356x Android11更换默认的Launcher

1、 开发环境 ubuntu版本&#xff1a;18.04 开发平台&#xff1a;RK356x Android版本&#xff1a;android11 2、目的 android11 系统自带了一个启动器Launcher3&#xff0c;在android源码路径下的packages/apps/Launcher3下&#xff0c;现需要将我们自己开发的Launcher放到a…

IOCP简单了解

1.IOCP是什么 IOCP是Input/Output Completion Ports的简称&#xff0c;中文翻译为完成端口&#xff0c;完成是应用程序向系统发起一个IO操作&#xff0c;系统会在操作结束后&#xff0c;将IO操作完成结果通知应用程序&#xff0c;端口指的是机制 2.重叠IO&#xff08;Overlappe…

Compose应用案例(利用docker compose安装lnmp实例)

目录 Compose应用案例 一、前提配置 &#xff08;一&#xff09;安装docker-ce&#xff08;Linux安装Docker&#xff09; &#xff08;二&#xff09;安装docker-compose 二、安装docker compose部署lnmp &#xff08;一&#xff09;目录结构&#xff1a; &#xff08;二…

SpringBoot、SpringCloud 版本查看

1、SpringBoot 官网地址 https://spring.io/projects/spring-boot#learn spring-boot-starter-parent 版本列表可查看&#xff1a; https://mvnrepository.com/artifact/org.springframework.boot/spring-boot-starter-parent 2、SpringCloud 官网地址 https://spring.io/pro…

pycharm安装

去官网下载安装包&#xff1a; 然后运行&#xff1a; &#xff08;左边第二个绿色字备注得有点子不对&#xff0c;这个勾选上的话&#xff0c;就是说在你的桌面上右击pycharm时会显示你的项目&#xff0c;你可以选择后直接打开。还是挺方便的一个功能&#xff0c;看自己需求要不…

Linux C++ 链接数据库并对数据库进行一些简单的操作

一.引言&#xff08;写在之前&#xff09; 在我们进行网络业务代码书写的时候&#xff0c;我们总是避免对产生的数据进行增删改查&#xff0c;为此&#xff0c;本小博主在这里简历分享一下自己在Linux中C语言与数据之间交互的代码的入门介绍。 二.代码书写以及一些变量和函数的…

Redis持久化两种方案以及对比差异

1.1.RDB持久化 RDB全称Redis Database Backup file&#xff08;Redis数据备份文件&#xff09;&#xff0c;也被叫做Redis数据快照。简单来说就是把内存中的所有数据都记录到磁盘中。当Redis实例故障重启后&#xff0c;从磁盘读取快照文件&#xff0c;恢复数据。快照文件称为R…

vmware网络配置

效果&#xff1a; 虚拟机和物理机网络互通&#xff1b; 虚拟机可以上外网 环境&#xff1a; vmware version 17.0.0 Centos 7.9 配置 1&#xff0c;vmware 菜单 - 编辑 - Virtual Network Edit 2&#xff0c; 选择VMnet8 VMnet information:NAT&#xff1b; 勾选2个…

第一百二十一天学习记录:线性代数:矩阵乘法运算(宋浩板书)

在编程和学习数据结构的过程中&#xff0c;发现有些算法会用到矩阵和矩阵的乘法运算&#xff0c;因此先将这一个知识点学习一下。 矩阵和行列式的区别 各种矩阵的概念 矩阵运算 乘法☆ 总结三条不满足