C++ string类

目录

为什么要有string类?

auto和范围for

1. auto

2. 范围for

一. 标准库中string类的简单使用

二. string类的常用接口说明

1. string类对象的常见构造

2. string类对象的容量操作

 3. string类对象的访问及遍历操作

4. string类对象的修改与查找操作

 插入字符或字符串​编辑

字符串替换

字符串的交换

查找的使用

5. string类非成员函数


为什么要有string类?

在C语言中,字符串是以'\0'结尾的一些字符的集合,为了操作方便,C标准库中提供了一些str系列的库函数,但是这些库函数与字符串是分离开的,不太OOP的思想,而且底层空间需要用户自己管理,稍不留神可能还会越界访问。

auto和范围for

C++11的两个小语法

1. auto
  •  在早期C/C++中auto的含义是:: 使用auto修饰的变量,是具有自动存储器的局部变量,后来这个不重要了。C++11中,标准委员会变废为宝赋予了 auto全新的含义,即:auto不再是一个存储类型的指示符,而是作为一个新的类型指示符来指示编译器,auto声明的变量必须由编译器在编译时期推导而得
  • auto声明指针类型时,用auto和auto*没有任何区别,但用auto声明引用类型时则必须加&
#include<iostream>
#include<string>
using namespace std;
int main()
{int a = 100;auto b = a;auto c = 'c';//auto e;//报错,必须具有初始值设定cout << typeid(a).name() << endl;cout << typeid(b).name() << endl;cout << typeid(c).name() << endl;auto pa1 = &a;auto* pa2 = &a;auto& pa3 = a;//声明引用类型必须加&cout << a << endl;pa3 = 1;cout << typeid(pa1).name() << endl;cout << typeid(pa2).name() << endl;cout << pa3 << "  " << a<<endl;return 0;
}
  • 当在同一行声明多个变量时这些变量必须是相同的类型,否则编译器将会报错,因为编译器实际只对第一个类型进行推导,然后再用推导出来的类型定义其他变量
  • auto不能作为函数的参数,可以做返回值,但是建议谨慎使用
  • auto不能直接用来声明数组

如下代码

#include<iostream>
#include<string>
using namespace std;void fun1(auto a)
{return a;
}int main()
{auto a = 10, b = 11.11;auto c = 'x', d = 1.01;auto aa[] = { 1,2,3,4 };return 0;
}

报错分别为

  指在同一行第一个类型推导与后不一致

此处是指函数参数处

即auto不能直接声明数组类型

2. 范围for
  • 对于一个有范围的集合而言,由程序员来说明循环的范围是多余的,有时候还会容易犯错误。因此C++11中引入了基于范围的for循环。for循环后的括号由冒号" : " 分为两部分:第一部分是范围内用于迭代的变量,第二部分则表示被迭代的范围自动迭代,自动取数据,自动判断结束
  • 范围for可以作用到数组和容器对象上进行遍历。
  • 范围for的底层很简单,容器遍历实际就是替换为迭代器,这个从汇编层也可以看到。
#include<iostream>
#include<string>
using namespace std;
int main()
{int arr[] = { 1,2,3,74,6 };for (int i = 0; i < sizeof(arr) / sizeof(int); i++){arr[i] *= 2;}for (int i = 0; i < sizeof(arr) / sizeof(int); i++){cout << arr[i] << "  ";}cout << endl;for (auto v : arr){v *= 2;}for (auto v : arr){cout << v << "  ";}cout << endl;string s1 = "hello world";for (auto s : s1){cout << s << "  ";}cout << endl;return 0;
}

 我们发现遍历时并没有改变arr数组的值

我们只需将

	for (auto v : arr){v *= 2;}

改为

	for (auto& v : arr){v *= 2;}

即可 

而auto也不是一定的,只要我们知道要遍历的数组或容器对象的类型指定即可

如下代码

#include<iostream>
#include<string>
using namespace std;
int main()
{int arr[] = { 1,2,3,74,6 };for (int& v : arr){v *= 2;}for (int v : arr){cout << v << "  ";}cout << endl;string s1 = "hello world";for (char s : s1){cout << s << "  ";}cout << endl;return 0;
}

一. 标准库中string类的简单使用

在使用string类时,必须包含#include<string>头文件,由于string属于std标准库,所以通过域作用限定符可以使用string

#include<iostream>
#include<string>
int main()
{std::string ss1("56565");std::string ss2("12123");std::cout << ss1 << std::endl;std::cout << ss1 << std::endl;return 0;
}

也可以通过using 将其展开,然后使用

#include<iostream>
#include<string>
//using namespace std;
using std::cout;
using std::string;
using std::endl;
using std::cin;
int main()
{string ss1("56565");string ss2("12123");cout << ss1 << endl;cout << ss2 << endl;return 0;
}

二. string类的常用接口说明

1. string类对象的常见构造
函数名称功能说明
string()构造空的string类对象,即空字符串
string(const char* s)用s指向的字符序列(C字符串)构造string类对象
string(size_t n,char c)string类对象中包含n个字符c
string(const string& s)拷贝构造函数

如下代码所示

#include<iostream>
#include<string>
using namespace std;int main()
{//string s1();//不可以这样写string s2;//构造空的string类对象s2cout  << s2 << endl;string s3("hello world");//用C格式字符串构造string类对象s3cout << s3 << endl;string s4(11, 'x');//string类对象包含11个x字符cout << s4 << endl;string s5(s3);//拷贝构造s5cout << s5 << endl;return 0;
}
2. string类对象的容量操作
函数名称功能说明函数声明
size返回字符串有效字符长度
size_t size() const noexcept;
length返回字符串有效字符长度
size_t length() const noexcept;
capacity返回空间总大小
size_t capacity() const noexcept;
empty检测字符串是否为空串,是返回true,否则返回false
bool empty() const noexcept;
clear清空有效字符
void clear() noexcept;
reserve为字符串预留空间
void reserve (size_t n = 0);
resize将有效字符的个数改成n个,多出的空间用字符c填充
void resize (size_t n);
void resize (size_t n, char c);

size()与length() 方法底层实现原理完全相同,引入size()的原因是为了与其他容器的接口保持一致(比如非线性结构二叉树的length肯定不是全部节点的个数),一般情况下基本都是用size()

#include<iostream>
#include<string>
using namespace std;int main()
{string s2;cout << s2.size() << "  " << s2.length() << endl;string s3("hello world");cout << s3.size() << "  " << s3.length() << endl;string s4(13, 'x');cout << s4.size() << "  " << s4.length() << endl;return 0;
}

clear()只是将string中有效字符清空,不改变底层空间大小

 如以下代码

#include<iostream>
#include<string>
using namespace std;int main()
{string s4(13, 'x');cout << "字符串为:" << s4 << "\n容量为" << s4.capacity() << endl;s4.clear();if(s4.empty())cout << "字符串为:" << s4 << "\n容量为" << s4.capacity() << endl;return 0;
}

其结果为

 resize(size_t n)与resize(size_t  n,char  c)都是将字符串中有效字符个数改到n个,不同的是当字符个数增多时:resize(size_t  n) 用 0 来填充多出的元素空间,resize(size_t  n,char  c)用字符c来填充多出的元素空间。注意: resize在改变元素个数时,如果是将元素个数增多,可能会改变底层容量的大小,如果是将元素个数减少,底层空间总大小不变

 代码如下

#include<iostream>
#include<string>
using namespace std;int main()
{string s1(13, 'x');string s2 (s1);string s3 (s1);s2.resize(5);s3.resize(5, '$');cout << "字符串为:" << s2 << "\n容量为" << s2.capacity() << endl;cout << "字符串为:" << s3 << "\n容量为" << s3.capacity() << endl;s2.resize(10);s3.resize(10, '$');cout << "字符串为:" << s2 << "\n容量为" << s2.capacity() << endl;cout << "字符串为:" << s3 << "\n容量为" << s3.capacity() << endl;s2.resize(16);s3.resize(16, '@');cout << "字符串为:" << s2 << "\n容量为" << s2.capacity() << endl;cout << "字符串为:" << s3 << "\n容量为" << s3.capacity() << endl;return 0;
}

 reserve(size_t  res_arg=0)为string预留空间,不改变有效元素个数,当reserve的参数小于string的底层空间总大小时,reserve不会改变容量大小

 代码如下

#include<iostream>
#include<string>
using namespace std;int main()
{string s1("111");cout << "字符串为:" << s1 << "\n容量为" << s1.capacity() << endl;s1.reserve(18);cout << "字符串为:" << s1 << "\n容量为" << s1.capacity() << endl;s1.reserve(1);cout << "字符串为:" << s1 << "\n容量为" << s1.capacity() << endl;return 0;
}

 结果为

 3. string类对象的访问及遍历操作
函数名称功能说明函数声明
operator[]返回pos位置的字符,const string类对象调用
char& operator[] (size_t pos);
const char& operator[] (size_t pos) const;
begin与end

begin获取一个字符的迭代器

end获取最后一个字符的下一个位置的迭代器

 iterator begin() noexcept;
const_iterator begin() const noexcept; 
iterator end() noexcept;
const_iterator end() const noexcept;
rbegin与rend

rbegin返回一个指向容器最后一个元素的反向迭代器

rend返回一个反向迭代器,指向容器中第一个元素之前的位置

reverse_iterator rbegin() noexcept;
const_reverse_iterator rbegin() const noexcept;
reverse_iterator rend() noexcept;
const_reverse_iterator rend() const noexcept;
范围forC++11支持的更简洁的新遍历方式

代码如下

#include<iostream>
#include<string>
using namespace std;int main()
{string s1("135616549");for (int i = 0; i < s1.size(); i++)cout << s1[i];cout << endl;string::iterator i = s1.begin();for (; i != s1.end(); i++)cout << *i;cout << endl;string::reverse_iterator i1= s1.rbegin();for (; i1 != s1.rend();i1++)cout << *i1;cout << endl;return 0;
}

范围for此处就不做演示了,上文有

4. string类对象的修改与查找操作

函数名称

功能说明函数声明
push_back在字符串后尾插字符c
void push_back (char c);
append在字符串后追加一个字符串接口过多详见表格下图
operator+=在字符串后追加字符串str
insert在指定位置追加字符或字符串
assign使用指定字符串替换原字符串
replace使用字符串替换原字符串指定区间
erase删除字符串指定的部分
c_str返回C格式字符串
const char* c_str() const noexcept;
swap交换两个字符串
void swap (string& str);
find从字符串pos位置开始往后找字符c,找到返回该字符在字符串中的位置
rfind从字符串pos位置开始往前找字符c,找到返回该字符在字符串中的位置
npos(不是函数)静态成员常数值,size_t类型的最大可能值,find与rfind没找到返回npos
static const size_t npos = -1;
substr在str中从pos位置开始,截取n个字符,然后将其返回
string substr (size_t pos = 0, size_t len = npos) const;
 插入字符或字符串

 

 

 

 接口过多在使用时可以查阅文档使用

在string尾部追加字符时s.push_back(c) 与  s.append(1, c) 与 s += 'c'三种的实现方式差 不多,一般情况下string类的+=操作用的比较多,+=操作不仅可以连接单个字符,还可 以连接字符串。

 如下代码所示

#include<iostream>
#include<string>
using namespace std;int main()
{string s1("135616549");string s2(s1);string s3(s1);string s4(s1);string s5(s1);s2.push_back('x');s3.append("x");s4 += 'x';s5.insert(9,"x");cout << s2 << endl;cout << s3 << endl;cout << s4 << endl;cout << s5 << endl;s4 += "cccc";cout << s4 << endl;return 0;
}

对string操作时,若可以提前知道放多少字符,可以先通过reserve把空间预留好

字符串替换

 演示代码如下

#include<iostream>
#include<string>
using namespace std;int main()
{string s1("@123456@");string s2("555555555");string s3(s1);cout << s1 << endl;s1.assign(s2);cout << s1 << endl;s1.assign("666666665");cout << s1 << endl;s1.replace(1, 5, s3);cout << s1 << endl;return 0;
}

输出结果如下

字符串的交换

string::swap通常直接交换两个string对象的内部数据(如字符数组指针和长度信息),而不需要复制字符数据。这使得交换非常高效。在C++98中优于全局swap(而在C++11中,全局的swap检测到是两个string交换也会调用string的swap)

#include<iostream>
#include<string>
using namespace std;int main()
{string s1("@123456@");string s2("55555555");cout << "字符串为:  " << s1 << endl;cout << "字符串为:  " << s2 << endl;cout << endl;s1.swap(s2);cout << endl;cout << "字符串为:  " << s1 << endl;cout << "字符串为:  " << s2 << endl;cout << endl;swap(s1, s2);cout << endl;cout << "字符串为:  " << s1 << endl;cout << "字符串为:  " << s2 << endl;return 0;
}
查找的使用

可以与多种需要坐标定位的函数联合使用

#include<iostream>
#include<string>
using namespace std;int main()
{string s1("@123456@");size_t pos1=s1.find('@');size_t pos2 = s1.rfind('@');for (int i = pos1; i < s1.size(); i++)cout << s1[i];cout << endl;for (int i = pos2; i < s1.size(); i++)cout << s1[i];cout << endl;size_t pos3 = s1.find('@', 3);for (int i = pos3; i < s1.size(); i++)cout << s1[i];cout << endl;size_t pos4 = s1.find('&');size_t pos5 = s1.rfind('&');cout << pos4 << "  " << pos5 << "  " << string::npos << endl;//与substr的联合使用size_t pos = s1.find('4');cout << s1.substr(pos, 3) << endl;//与insert的联合使用s1.insert(pos, "xxxx");cout << s1 << endl;return 0;
}
5. string类非成员函数
函数功能

operator+

链接字符串返回一个新构造的字符串对象(尽量少用)

,因为传值返回导致深拷贝效率低

getline获取一行字符串
operator>>输入运算符重载
operator<<输出运算符重载
relational operators大小比较

 我们这里只介绍getline

代码:

#include<iostream>
#include<string>
using namespace std;int main()
{string s1;string s2;cin >> s1;cout << s1<<endl;getline(cin, s2);cout << s2<<endl;string s3;//遇到'&'才会结束getline(cin, s3, '&');cout << s3 << endl;return 0;
}

结果为

 getline与流插入都是读取字符串,但是流插入遇见空格就停止,并且不把空格读进去而getline会遇见空格不会停止,由上面我们的第一个输入

hello hello first

可得:流插入读取到第一个空格就结束了,第一个空格被getline读进去了(从输出也可以看出来)

普通getline遇见回车结束,我们可以指定遇见什么结束


这篇文章就到这里啦~喜欢的可以点一下赞

(๑′ᴗ‵๑)I Lᵒᵛᵉᵧₒᵤ❤

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

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

【Web 前端开发】vue3开发环境部署

1、安装 Node.js 和 npm 访问 Node.js 官网 下载并安装最新的 LTS 版本。 安装完成后&#xff0c;打开命令行工具&#xff0c; 输入 node -v 和 npm -v 检查安装是否成功。 node -vnpm -v 如下图&#xff1a; 2、安装 Vue CLI 在命令行工具中输入以下命令安装 Vue CLI&…

微信小程序开发:宿主环境—组件

✨✨ 欢迎大家来访Srlua的博文&#xff08;づ&#xffe3;3&#xffe3;&#xff09;づ╭❤&#xff5e;✨✨ &#x1f31f;&#x1f31f; 欢迎各位亲爱的读者&#xff0c;感谢你们抽出宝贵的时间来阅读我的文章。 我是Srlua小谢&#xff0c;在这里我会分享我的知识和经验。&am…

详解DDR3原理以及使用Xilinx MIG IP核(app 接口)实现DDR3读写测试

系列文章目录 &#xff08;1&#xff09;详解SDRAM基本原理以及FPGA实现读写控制 文章目录 系列文章目录一、DDR简介1.1 什么是 SDRAM、DDR、DDR2、DDR31.2 SDRAM、DDR、DDR2、DDR3核心频率、工作频率以及等效频率的计算1.3 DDR3带宽以及容量的计算 二、MIG IP核的介绍三、MIG…

机器学习 第8章-集成学习

机器学习 第8章-集成学习 8.1 个体与集成 集成学习(ensemble learning)通过构建并结合多个学习器来完成学习任务&#xff0c;有时也被称为多分类器系统(multi-classifersystem)、基于委员会的学习(committee-based learning)等。 图8.1显示出集成学习的一般结构:先产生一组“…

【附安装包】CentOS7(Linux)详细安装教程(手把手图文详解版)

目前流行的虚拟机软件有VMware、Virtual Box和Virtual PC等等&#xff0c;其中最常用的就是VMware。 而centos是Linux使用最广泛的版本之一。 教程开始教程有许多不完备之处&#xff0c;大佬请忽略。。。 1.安装VMware 首先需要准备VMware的安装包以及Ubuntu的ISO镜像&#…

鸿蒙系统开发【设备安全服务-应用设备状态检测】安全

设备安全服务-应用设备状态检测 介绍 本示例向您介绍如何在应用中获取DeviceToken用于对应用的设备状态进行检测。 需要使用设备安全服务接口 kit.DeviceSecurityKit。 效果预览 Sample工程的配置与使用 在DevEco中配置Sample工程的步骤如下 [创建项目]及[应用]。打开Sam…

数据结构与算法 - 递归

一、递归 1. 概述 定义&#xff1a;在计算机科学中&#xff0c;递归是一种解决计算问题的方法&#xff0c;其中解决方案取决于同一类问题的更小子集。 比如单链表递归遍历的例子&#xff1a; void f(Node node) {if(node null) {return;}println("before:" node…

Animate软件基础:在时间轴中添加或插入帧

FlashASer&#xff1a;AdobeAnimate2021软件零基础入门教程https://zhuanlan.zhihu.com/p/633230084 FlashASer&#xff1a;实用的各种Adobe Animate软件教程https://zhuanlan.zhihu.com/p/675680471 FlashASer&#xff1a;Animate教程及作品源文件https://zhuanlan.zhihu.co…

软件测试--python基础

一、python基础 (1)第一个python (2)python解释器 (3)基础语法 ①字面量 什么是字面量 常用的值类型 字符串 ②注释 ③变量 什么是变量 变量的特征 变量的目的是存储运行过程的数据 存储的目的是为了&#xff1a;重复使用 ④数据类型 type()语句 变量有类型吗&#xff1f;…

在欧拉系统中安装数据库

在欧拉系统中的安装 &#xff08;禁止超级用户root登录&#xff09; yum install mariadb-server -y #下载命令 systemctl enable --now mariadb #设置为开机自启&#xff0c;并立即启动该服务 mysql_secure_installation #安全设置&#xff0c;初始化 修…

C语言数据在内存中的存储超详解

文章目录 1. 整数在内存中的存储2. 大小端字节序和字节序判断2. 1 什么是大小端&#xff1f;2. 2 为什么会有大小端&#xff1f;2. 3 练习 3. 浮点数在内存中的存储3. 1 一个代码3. 2 浮点数的存储3. 2. 1 浮点数存的过程3. 2. 2 浮点数取的过程3. 3 题目解析 1. 整数在内存中的…

ctfshow-web入门-sql注入(web171-web175)

目录 1、web171 2、web172 3、web173 4、web174 5、web175 1、web171 单引号测一下&#xff0c;报错 -- 闭合后回显正常 也可以用 # &#xff0c;不过需要 URL 编码 成功闭合之后&#xff0c;先判断下字段数&#xff1a; 1 order by 3-- 3 的时候正常 4 的时候报错&am…

C++必修:STL之vector的了解与使用

✨✨ 欢迎大家来到贝蒂大讲堂✨✨ &#x1f388;&#x1f388;养成好习惯&#xff0c;先赞后看哦~&#x1f388;&#x1f388; 所属专栏&#xff1a;C学习 贝蒂的主页&#xff1a;Betty’s blog 1. C/C中的数组 1.1. C语言中的数组 在 C 语言中&#xff0c;数组是一组相同类型…

泰迪智能科技AI大模型某医院合作案例介绍

泰迪智能科技AI大模型支持以ChatGLM2-6B、Baichuan-13B、Qwen14B和文心一言等多种大语言模型为底座&#xff0c;实现基于特定领域数据、面向智能客服、问答系统、自动摘要、智能打标、内容创作、信息抽取等应用场景的模型微调、评估和推理&#xff0c;为业务智能升级和价值挖掘…

【C++从小白到大牛】类和对象

目录 一、面向过程和面向对象初步认识 二、类的引入 三、类的定义 类的成员函数两种定义方式&#xff1a; 1. 声明和定义全部放在类体中 2. 类声明放在.h文件中&#xff0c;成员函数定义放在.cpp文件中 成员变量命名规则的建议&#xff1a; 四、类的访问限定符 【访问限…

本地部署持续集成工具Jenkins并配置公网地址实现远程自动化构建

文章目录 前言1. 安装Jenkins2. 局域网访问Jenkins3. 安装 cpolar内网穿透软件4. 配置Jenkins公网访问地址5. 公网远程访问Jenkins6. 固定公网地址 前言 本文主要介绍如何在Linux CentOS 7中安装Jenkins并结合cpolar内网穿透工具实现远程访问管理本地部署的Jenkins服务. Jenk…

大模型算法面试题(十八)

本系列收纳各种大模型面试题及答案。 1、P-tuning v2 思路、优缺点是什么 P-tuning v2是清华大学自然语言处理实验室&#xff08;THUDM&#xff09;等研究机构提出的一种新的预训练模型优化方法&#xff0c;主要关注如何通过动态构建任务相关的提示序列来引导预训练模型进行更…

【数据结构进阶】手撕红黑树

&#x1f525;个人主页&#xff1a; Forcible Bug Maker &#x1f525;专栏&#xff1a; C || 数据结构 目录 &#x1f308;前言&#x1f525;红黑树的概念&#x1f525;手撕红黑树红黑树结点的定义红黑树主体需要实现的成员函数红黑树的插入findEmpty和Size拷贝构造析构函数和…

Redis和Mysql如何保持数据一致性

一般情况下&#xff0c;Redis是用来实现应用和数据库之间读操作得缓存层&#xff0c;主要目的是减少数据库IO&#xff0c;还可以提升数据的IO性能。 当应用程序需要去读取某个数据时&#xff0c;会首先尝试去Redis里面加载&#xff0c;如果命中就直接返回&#xff0c;如果没有…

C++ 操作Git仓库

代码 #include "common.h" #include "args.c" #include "common.c"enum index_mode {INDEX_NONE,INDEX_ADD };struct index_options {int dry_run;int verbose;git_repository* repo;enum index_mode mode;int add_update; };/* Forward declar…