【C++心愿便利店】No.11---C++之string语法指南
文章目录
- 前言
- 一、 为什么学习string类
- 二、标准库中的string类
前言

👧个人主页:@小沈YO.
😚小编介绍:欢迎来到我的乱七八糟小星球🌝
📋专栏:C++ 心愿便利店
🔑本章内容:string
记得 评论📝 +点赞👍 +收藏😽 +关注💞哦~
提示:以下是本篇文章正文内容,下面案例可供参考
一、 为什么学习string类
C语言中,字符串是以’\0’结尾的一些字符的集合,为了操作方便,C标准库中提供了一些str系列的库函数,但是这些库函数与字符串是分离开的,不太符合OOP的思想,而且底层空间需要用户自己管理,稍不留神可能还会越界访问
开始学习string就需要开始学习读文档具体可以通过cplusplus.con网站去搜索
二、标准库中的string类
🌟string
- 字符串是表示字符序列的类
- 标准的字符串类提供了对此类对象的支持,其接口类似于标准字符容器的接口,但添加了专门用于操作单字节字符字符串的设计特性。
- string类是使用char(即作为它的字符类型,使用它的默认char_traits和分配器类型(关于模板的更多信息,请参阅basic_string)。
- string类是basic_string模板类的一个实例,它使用char来实例化basic_string模板类,并用char_traits和allocator作为basic_string的默认参数(根于更多的模板信息请参考basic_string)。
- 注意,这个类独立于所使用的编码来处理字节:如果用来处理多字节或变长字符(如UTF-8)的序列,这个类的所有成员(如长度或大小)以及它的迭代器,将仍然按照字节(而不是实际编码的字符)来操作。
总结:
- string是表示字符串的字符串类
- 该类的接口与常规容器的接口基本相同,再添加了一些专门用来操作string的常规操作
- string在底层实际是:basic_string模板类的别名,typedef basic_string<char, char_traits, allocator> string;
- 不能操作多字节或者变长字符的序列
在使用string类时,必须包含#include头文件以及using namespace std;
| (constructor)函数名称 | 功能说明 |
|---|---|
| string() (重点) | 构造空的string类对象,即空字符串 |
| string(const char* s) (重点) | 用C-string来构造string类对象 |
| string(size_t n, char c) | string类对象中包含n个字符c |
| string(const string&s) (重点) | 拷贝构造函数 |
- string() (重点) - - - 构造空的string类对象,即空字符串
void test_string1()
{string s1;cout<<s1<<endl;
}
int main()
{test_string1();return 0;
}
- string(const char* s) (重点)- - - 用C-string来构造string类对象
void test_string1()
{string s2("hello");cout << s2 << endl;
}
int main()
{test_string1();return 0;
}
- string(size_t n, char c) string类对象中包含n个字符c
int main()
{string s1(3, 'a');cout << s1 << endl;return 0;
}
- string(const string&s) (重点) 拷贝构造函数
int main()
{string s1("hello");string s2(s1);cout << s1 << endl;cout << s2 << endl;return 0;
}
| 函数名称 | 功能说明 |
|---|---|
| size(重点) | 返回字符串有效字符长度 |
| length | 返回字符串有效字符长度 |
| capacity | 返回空间总大小 |
| empty (重点) | 检测字符串释放为空串,是返回true,否则返回false |
| clear (重点) | 清空有效字符 |
| reserve (重点) | 为字符串预留空间(确定大概知道要多少空间,提前开好,减少扩容,提高效率) |
| resize (重点) | 将有效字符的个数改成n个,多出的空间用字符c填充 |
| shrink_to_fit | 将capacity容量缩至合适 |
- size(重点)- - - 返回字符串有效字符长度
int main()
{string s1("hello");string s2("aaaaaaaaaaaa");cout << s1.size() << endl;cout << s2.size() << endl;return 0;
}
- length - - - 返回字符串有效字符长度
int main()
{string s1("hello");string s2("aaaaaaaaaaaa");cout << s1.length() << endl;cout << s2.length() << endl;return 0;
}
- capacity - - - 返回空间总大小
int main()
{string s1("hello");string s2("aaaaaaaaaaaa");cout << s1.capacity () << endl;cout << s2.capacity ()<< endl;return 0;
}

同一个string对象,在不同平台下的capacity()(空间容量)可能不同,因为string在底层就是一个存储字符的动态顺序表,空间不够了要进行扩容,而不同平台底层的扩容机制有所不同,导致了最终capacity()的结果不同。例如下述展现的扩容机制:
🌟VS下的扩容机制: 第一次扩容是2倍,后面都是以1.5倍的大小去扩容。
void test_string1()
{string s;size_t old = s.capacity();cout << "初始" << s.capacity() << endl;for (size_t i = 0; i < 100; i++){s.push_back('a');if (s.capacity() != old){cout << "扩容:" << s.capacity() << endl;old = s.capacity();}}cout << s.capacity() << endl;
}

🌟Linux下的扩容机制: 一次按照2倍的大小进行扩容
void test_string1()
{string s;size_t old = s.capacity();cout << "初始" << s.capacity() << endl;for (size_t i = 0; i < 100; i++){s.push_back('a');if (s.capacity() != old){cout << "扩容:" << s.capacity() << endl;old = s.capacity();}}cout << s.capacity() << endl;
}

- empty (重点)- - - 检测字符串释放为空串,是返回true,否则返回false
int main()
{string s1;if (s1.empty()){cout << "s1是一个空字符串" << endl;}return 0;
}
- clear (重点)- - - 清空有效字符
void test_string8()
{string s1("hello world");cout << s1.size() << endl;//11cout << s1.capacity() << endl;//15s1.clear();cout << s1.size() << endl;//0cout << s1.capacity() << endl;//15
}
- reserve (重点)- - - 为字符串预留空间(确定大概知道要多少空间,提前开好,减少扩容,提高效率)
void test_string1()
{string s;s.reserve(100);size_t old = s.capacity();cout << "初始" << s.capacity() << endl;for (size_t i = 0; i < 100; i++){s.push_back('a');if (s.capacity() != old){cout << "扩容:" << s.capacity() << endl;old = s.capacity();}}s.reserve(10);cout << s.capacity() << endl;
}
如上当不写s.reserve(100);就会发生扩容,但是当写上s.reserve(100);提前开好空间就不会发生扩容,同时要注意s.reserve(10);并不会缩减空间(缩容)

- resize (重点)- - - 将有效字符的个数改成n个,多出的空间用字符c填充
void test_string2()
{string s1("hello world");cout << s1 << endl;cout << s1.size() << endl;cout << s1.capacity() << endl;//s1.resize(13, 'x');s1.resize(20, 'x');//这里有效字符个数改成20s1本来的有效字符hello world是11个超出部分用x补充,其次size()和capacity也会随之发生改变--->size()变成20;capacity()变成31s1.resize(5);cout << s1 << endl;cout << s1.size() << endl;cout << s1.capacity() << endl;string s2;s2.resize(10, '#');cout << s2 << endl;cout << s2.size() << endl;cout << s2.capacity() << endl;
}

- shrink_to_fit() - - - 将capacity容量缩至合适
void test_string1()
{string s;s.reserve(50);s += "hello world";cout << s.size() << endl;cout << s.capacity() << endl;s.shrink_to_fit();cout << s.size() << endl;cout << s.capacity() << endl;
}

注意:
- size()与length()方法底层实现原理完全相同,引入size()的原因是为了与其他容器的接口保持一致,一般情况下基本都是用size()。
- clear()只是将string中有效字符清空,不改变底层空间大小。
- resize(size_t n) 与 resize(size_t n, char c)都是将字符串中有效字符个数改变到n个,不同的是当字符个数增多时:resize(n)用0来填充多出的元素空间,resize(size_t n, char c)用字符c来填充多出的元素空间。注意:resize在改变元素个数时,如果是将元素个数增多,可能会改变底层容量的大小,如果是将元素个数减少,底层空间总大小不变。
- reserve(size_t res_arg=0):为string预留空间,不改变有效元素个数,当reserve的参数小于string的底层空间总大小时,reserver不会改变容量大小。
| 函数名称 | 功能说明 |
|---|---|
| operator[] (重点) | 返回pos位置的字符,const string类对象调用 |
| begin+ end | begin获取一个字符的迭代器 + end获取最后一个字符下一个位置的迭代器 |
| rbegin + rend | begin获取一个字符的迭代器 + end获取最后一个字符下一个位置的迭代器 |
| 范围for | C++11支持更简洁的范围for的新遍历方式 |
- operator[] (重点) 返回pos位置的字符,const string类对象调用
void test_string2()
{string s1("hello world\n");string s2 = "hello world";//单参数构造支持隐式类型转换//遍历stringfor (size_t i = 0; i < s1.size(); i++){//读cout << s1[i] << " ";}cout << endl;for (size_t i = 0; i < s1.size(); i++){//写s1[i]++;}cout << s1 << endl;;
}
- begin+ end - - - begin获取一个字符的迭代器 + end获取最后一个字符下一个位置的迭代器
void test_string3()
{string s4="hello world";string::iterator it = s4.begin();while (it != s4.end()){//读cout << *it << " ";it++;}cout << endl;it = s4.begin();//while (it < s4.end())可以这样写但是不建议while (it != s4.end()){//写*it='a';cout << *it << " ";it++;}cout << endl;
}
- rbegin + rend - - - begin获取一个字符的迭代器 + end获取最后一个字符下一个位置的迭代器
void test_string4()
{string s5 = "hello world";string::reverse_iterator rit = s5.rbegin();while (rit != s5.rend()){cout << *rit << " ";++rit;}cout << endl;
}
- 范围for - - - C++11支持更简洁的范围for的新遍历方式
void test_string5()
{string s6 = "hello world";//原理:编译器替换成迭代器for (auto ch : s6){//读cout << ch << " ";}cout << endl;//对于写---范围for 本质自动遍历是*it赋值给ch,ch是*it的拷贝所以要写的话要加&,这样ch就是*it的别名for (auto ch : s6)//错误写法for (auto& ch : s6){//写ch++;}cout << s6 << " ";cout << endl;}
void func(const string s)//不推荐传值传参,会进行拷贝调用拷贝构造string的底层不能用浅拷贝所以用引用+constvoid func(const string& s)
{//迭代器支持读写,但是这里是const不支持迭代器写所以C++设计出了cbegin() cend() crbegin() crend()也可以+const例如下面一行注释代码//string::const_iterator it = s.begin();//对比于上面代码+const和不+const还用修改可以直接使用auto直接推出类型auto it = s.begin();while (it != s.end()){//读cout << *it << " ";it++;}cout << endl;//string::const_reverse_iterator rit = s.rbegin();auto rit = s.rbegin();while (rit != s.rend()){cout << *rit << " ";++rit;}cout << endl;
}
void test_string6()
{string s7 = "hello world";func(s7);
}
| 函数名称 | 功能说明 |
|---|---|
| push_back | 在字符串后尾插字符c |
| append | 在字符串后追加一个字符串 |
| operator+= (重点) | 在字符串后追加字符串str |
| c_str(重点) | 返回C格式字符串 |
| find + npos(重点) | 从字符串pos位置开始往后找字符c,返回该字符在字符串中的位置 |
| rfind | 从字符串pos位置开始往前找字符c,返回该字符在字符串中的位置 |
| substr | 在str中从pos位置开始,截取n个字符,然后将其返回 |
- push_back - - - 在字符串后尾插字符c
void test_string3()
{string s;string ss("hello");s.push_back('#');cout <<s<< endl;
}

- append - - - 在字符串后追加一个字符串
void test_string3()
{string s;string ss("hello");s.append("hello world");s.append(ss);cout <<s<< endl;
}

- operator+= (重点) - - - 在字符串后追加字符串str
void test_string3()
{string s;string ss("hello");s += '#';s += "hello";s += ss;cout << s << endl;
}

- c_str(重点) - - - 返回C格式字符串
void test_string9()
{string filename;cin >> filename;FILE* fout = fopen(filename.c_str(), "r");
}
- find + npos(重点) - - - 从字符串pos位置开始往后找字符c,返回该字符在字符串中的位置
void test_string2()
{string s1("test.cpp");//读取文件后缀size_t i = s1.find('.');string s3 = s1.substr(i);cout << s3 << endl;
}

- rfind - - - 从字符串pos位置开始往前找字符c,返回该字符在字符串中的位置
void test_string2()
{string s2("test.cpp.tar.zip");//找文件后缀.zipsize_t j = s2.rfind('.');//倒着找string s4 = s2.substr(j);cout << s4 << endl;
}

- substr - - - 在str中从pos位置开始,截取n个字符,然后将其返回
void test_string2()
{string s1("test.cpp");//读取文件后缀size_t i = s1.find('.');string s3 = s1.substr(i);cout << s3 << endl;
}
注意:
- 在string尾部追加字符时,s.push_back© / s.append(1, c) / s += 'c’三种的实现方式差不多,一般情况下string类的+=操作用的比较多,+=操作不仅可以连接单个字符,还可以连接字符串。
- 对string操作时,如果能够大概预估到放多少字符,可以先通过reserve把空间预留好。
| 函数 | 功能说明 |
|---|---|
| operator+ | 尽量少用,因为传值返回,导致深拷贝效率低 |
| operator>> (重点) | 输入运算符重载 |
| operator<< (重点) | 输出运算符重载 |
| getline (重点) | 获取一行字符串 |
| relational operators (重点) | 大小比较 |
- operator+ - - - 尽量少用,因为传值返回,导致深拷贝效率低
void test_string3()//at失败后会抛异常
{string ss("hello");string ret=ss + '#';//+是一个传值返回,代价比较大string ret2 = ss + "hello";cout << ret << endl;cout << ret2 << endl;cout << endl;
}

- operator>> (重点)- - - 输入运算符重载
- operator<< (重点) - - - 输出运算符重载
- getline (重点) - - - 获取一行字符串
🌟cin>> 和getline的区别在于:>>遇到空格’ '和换行\n会截止,而getline默认只有遇到换行\n才截止,因此当我们需要从键盘读取一个含有空格的字符串是,只能用getline

void test_string3()
{string s1;getline(cin, s1, '!');cout << s1;
}

- relational operators (重点)- - - 大小比较
-
string类型转换成内置类型

-
内置类型转换成string

注意:下述结构是在32位平台下进行验证,32位平台下指针占4个字节
vs下string的结构
string总共占28个字节,内部结构稍微复杂一点,先是有一个联合体,联合体用来定义string中字
符串的存储空间:
当字符串长度小于16时,使用内部固定的字符数组来存放
当字符串长度大于等于16时,从堆上开辟空间
union _Bxty
{ // storage for small buffer or pointer to larger onevalue_type _Buf[_BUF_SIZE];pointer _Ptr;char _Alias[_BUF_SIZE]; // to permit aliasing
} _Bx;
这种设计也是有一定道理的,大多数情况下字符串的长度都小于16,那string对象创建好之后,内部已经有了16个字符数组的固定空间,不需要通过堆创建,效率高。
其次:还有一个size_t字段保存字符串长度,一个size_t字段保存从堆上开辟空间总的容量
最后:还有一个指针做一些其他事情。
故总共占16+4+4+4=28个字节

g++下string的结构
G++下,string是通过写时拷贝实现的,string对象总共占4个字节,内部只包含了一个指针,该指
针将来指向一块堆空间,内部包含了如下字段:
- 空间总大小
- 字符串有效长度
- 引用计数
struct _Rep_base
{size_type _M_length;size_type _M_capacity;_Atomic_word _M_refcount;
};
- 指向堆空间的指针,用来存储字符串。
相关文章:
【C++心愿便利店】No.11---C++之string语法指南
文章目录 前言一、 为什么学习string类二、标准库中的string类 前言 👧个人主页:小沈YO. 😚小编介绍:欢迎来到我的乱七八糟小星球🌝 📋专栏:C 心愿便利店 🔑本章内容:str…...
OpenCV检测圆(Python版本)
文章目录 示例代码示例结果调参 示例代码 import cv2 import numpy as np# 加载图像 image_path DistanceComparison/test_image/1.png image cv2.imread(image_path, cv2.IMREAD_COLOR)# 将图像转换为灰度 gray cv2.cvtColor(image, cv2.COLOR_BGR2GRAY)# 使用高斯模糊消除…...
轻量封装WebGPU渲染系统示例<15>- DrawInstance批量绘制(源码)
当前示例源码github地址: https://github.com/vilyLei/voxwebgpu/blob/main/src/voxgpu/sample/DrawInstanceTest.ts 此示例渲染系统实现的特性: 1. 用户态与系统态隔离。 细节请见:引擎系统设计思路 - 用户态与系统态隔离-CSDN博客 2. 高频调用与低频调用隔离。…...
E: 仓库 “http://cn.archive.ubuntu.com/ubuntu kinetic Release” 没有 Release 文件。
sudo apt-get update时报以下错误: E: 仓库 “http://cn.archive.ubuntu.com/ubuntu kinetic Release” 没有 Release 文件。 N: 无法安全地用该源进行更新,所以默认禁用该源。 N: 参见 apt-secure(8) 手册以了解仓库创建和用户配置方面的细节。 E: 仓库…...
【VR开发】【Unity】【VRTK】3-VR项目设置
任何VR避不开的步骤 如何设置VR项目,无论是PC VR还是安卓VR,我在不同的系列教程中都说过了,不过作为任何一个VR开发教程都难以避免的一环,本篇作为VRTK的开发教程还是对VR项目设置交代一下。 准备好你的硬件 头盔必须是6DoF的,推荐Oculus Quest系列,Rift系列,HTC和Pi…...
git log 用法
git log --format"%s" -n 1在 Git 中,您可以使用 git log 命令来查看提交历史,其中包含每个提交的详细信息,包括提交消息。如果您只想提取提交信息而不是完整的 git log 输出,可以使用 git log 命令的 --format 选项来指…...
Linux学习---有关监控系统zabbix的感悟
监控系统 监控系统就像咱们日常生活中小区监控(Monitor),用于及时发现问题(PROBLEM),根据相应的规则可以触发警告(Media),在后台显示屏(Dashboard)上以某种方面显示出来,高级的报警系统也许还能实现电话通知等功能,目的是为及时发…...
apollo云实验:定速巡航场景仿真调试
定速巡航场景仿真调试 概述启动仿真环境仿真系统修改默认巡航速度 实验目的福利活动 主页传送门:📀 传送 概述 自动驾驶汽车在实现落地应用前,需要经历大量的道路测试来验证算法的可行性和系统的稳定性,但道路测试存在成本高昂、…...
基于RK3568的新能源储能能量管理系统ems
新能源储能能量管理系统(EMS)是一种基于现代化技术的系统,旨在管理并优化新能源储能设备的能量使用。 该系统通过监测、调度和控制新能源储能设备来确保能源的高效利用和可持续发展。 本文将从不同的角度介绍新能源储能能量管理系统的原理、…...
dockerfile避坑笔记(VMWare下使用Ubuntu在Ubuntu20.04基础镜像下docker打包多个go项目)
一、docker简介 docker是一种方便跨平台迁移应用的程序,通过docker可以实现在同一类操作系统中,如Ubuntu和RedHat两个linux操作系统中,实现程序的跨平台部署。比如我在Ubuntu中打包了一个go项目的docker镜像(镜像为二进制文件&am…...
Qt 使用QtXlsx操作Excel表
1.环境搭建 QtXlsx是一个用于读写Microsoft Excel文件(.xlsx)的Qt库。它提供了一组简单易用的API,可以方便地处理电子表格数据。 Github下载:GitHub - dbzhang800/QtXlsxWriter: .xlsx file reader and writer for Qt5 官方文档…...
canal+es+kibana+springboot
1、环境准备 服务器:Centos7 Jdk版本:1.8 Mysql版本:5.7.44 Canal版本:1.17 Es版本:7.12.1 kibana版本:7.12.1 软件包下载地址:链接:https://pan.baidu.com/s/1jRpCJP0-hr9aI…...
【力扣】面试经典150题——双指针
文章目录 125. 验证回文串392. 判断子序列167. 两数之和 II - 输入有序数组11. 盛最多水的容器15. 三数之和 125. 验证回文串 如果在将所有大写字符转换为小写字符、并移除所有非字母数字字符之后,短语正着读和反着读都一样。则可以认为该短语是一个 回文串 。 字…...
6-8 最宽层次结点数 分数 10
文章目录 1.题目描述2.本题ac答案2.1法一: 代码复用2.2法二: 顺序队列实现层序遍历 3.C层序遍历求最大宽度3.1层序遍历代码3.2求最大宽度 1.题目描述 2.本题ac答案 2.1法一: 代码复用 //二叉树第i层结点个数 int LevelNodeCount(BiTree T, int i) {if (T NULL || i < 1)re…...
Linux学习第28天:Platform设备驱动开发(二): 专注与分散
Linux版本号4.1.15 芯片I.MX6ULL 大叔学Linux 品人间百味 思文短情长 三、硬件原理图分析 四、驱动开发 1、platform设备与驱动程序开发 53 /* 54 * 设备资源信息,也就是 LED0 所使用的所有寄存器 55 */ 56 static str…...
postgresql数组重叠(有共同元素)查询
直接上最终代码: select distinct id from a where string_to_array(in_area,,) && (select ARRAY_AGG( code) from areas where code like 11% or code 100000)::TEXT[] pg语法: 表 9.48显示了可用于数组类型的运算符。 表 9.48。数组运算符…...
ubuntu系统 生成RSA密钥对
在Ubuntu系统上生成密钥对通常指的是生成SSH密钥对,它常用于安全的远程登录、数据通信和其他安全网络操作。以下是如何在Ubuntu系统上生成SSH密钥对的步骤: 打开终端:你可以使用快捷键 Ctrl Alt T 在Ubuntu上打开一个终端窗口。 运行ssh-k…...
【RtpSeqNumOnlyRefFinder】webrtc m98: ManageFrameInternal 的帧决策过程分析
Jitterbuffer(FrameBuffer)需要组帧以后GOP内的参考关系 JeffreyLau 大神分析 了组帧原理而参考关系(RtpFrameReferenceFinder)的生成伴随了帧决策 FrameDecisionFrameDecision 影响力 帧的缓存。调用 OnAssembledFrame 传递已经拿到的RtpFrameObject 那么,RtpFrameObject…...
centos系统源码编译安装nginx,并编写服务脚本
1.安装编译所需的依赖项: yum install -y gcc pcre-devel openssl-devel zlib-devel2.下载 Nginx 源代码: wget http://nginx.org/download/nginx-1.21.3.tar.gz tar -xf nginx-1.21.3.tar.gz cd nginx-1.21.33.配置编译选项并进行编译和安装ÿ…...
2023下半年软考高项答题技巧!
2023下半年软考倒计时最后一天,一些软考高项答题技巧分享! 高项答题技巧 1、综合知识 (1)首先是分析试题的技巧 –先看清楚问题,再看选项; –判断题目到底考察的是什么知识点,排除干扰项。…...
循环冗余码校验CRC码 算法步骤+详细实例计算
通信过程:(白话解释) 我们将原始待发送的消息称为 M M M,依据发送接收消息双方约定的生成多项式 G ( x ) G(x) G(x)(意思就是 G ( x ) G(x) G(x) 是已知的)࿰…...
Objective-C常用命名规范总结
【OC】常用命名规范总结 文章目录 【OC】常用命名规范总结1.类名(Class Name)2.协议名(Protocol Name)3.方法名(Method Name)4.属性名(Property Name)5.局部变量/实例变量(Local / Instance Variables&…...
转转集团旗下首家二手多品类循环仓店“超级转转”开业
6月9日,国内领先的循环经济企业转转集团旗下首家二手多品类循环仓店“超级转转”正式开业。 转转集团创始人兼CEO黄炜、转转循环时尚发起人朱珠、转转集团COO兼红布林CEO胡伟琨、王府井集团副总裁祝捷等出席了开业剪彩仪式。 据「TMT星球」了解,“超级…...
Java-41 深入浅出 Spring - 声明式事务的支持 事务配置 XML模式 XML+注解模式
点一下关注吧!!!非常感谢!!持续更新!!! 🚀 AI篇持续更新中!(长期更新) 目前2025年06月05日更新到: AI炼丹日志-28 - Aud…...
Mac下Android Studio扫描根目录卡死问题记录
环境信息 操作系统: macOS 15.5 (Apple M2芯片)Android Studio版本: Meerkat Feature Drop | 2024.3.2 Patch 1 (Build #AI-243.26053.27.2432.13536105, 2025年5月22日构建) 问题现象 在项目开发过程中,提示一个依赖外部头文件的cpp源文件需要同步,点…...
JAVA后端开发——多租户
数据隔离是多租户系统中的核心概念,确保一个租户(在这个系统中可能是一个公司或一个独立的客户)的数据对其他租户是不可见的。在 RuoYi 框架(您当前项目所使用的基础框架)中,这通常是通过在数据表中增加一个…...
Spring是如何解决Bean的循环依赖:三级缓存机制
1、什么是 Bean 的循环依赖 在 Spring框架中,Bean 的循环依赖是指多个 Bean 之间互相持有对方引用,形成闭环依赖关系的现象。 多个 Bean 的依赖关系构成环形链路,例如: 双向依赖:Bean A 依赖 Bean B,同时 Bean B 也依赖 Bean A(A↔B)。链条循环: Bean A → Bean…...
mac:大模型系列测试
0 MAC 前几天经过学生优惠以及国补17K入手了mac studio,然后这两天亲自测试其模型行运用能力如何,是否支持微调、推理速度等能力。下面进入正文。 1 mac 与 unsloth 按照下面的进行安装以及测试,是可以跑通文章里面的代码。训练速度也是很快的。 注意…...
JDK 17 序列化是怎么回事
如何序列化?其实很简单,就是根据每个类型,用工厂类调用。逐个完成。 没什么漂亮的代码,只有有效、稳定的代码。 代码中调用toJson toJson 代码 mapper.writeValueAsString ObjectMapper DefaultSerializerProvider 一堆实…...
Linux基础开发工具——vim工具
文章目录 vim工具什么是vimvim的多模式和使用vim的基础模式vim的三种基础模式三种模式的初步了解 常用模式的详细讲解插入模式命令模式模式转化光标的移动文本的编辑 底行模式替换模式视图模式总结 使用vim的小技巧vim的配置(了解) vim工具 本文章仍然是继续讲解Linux系统下的…...
