【C++String类使用】万字详解保姆级教学,手把手教你使用string类。1

简介: 【C++String类使用】万字详解保姆级教学,手把手教你使用string类。

什么是string类?

C++中的string类是一个字符串容器类,它提供了一系列操作字符串的方法,例如连接、查找、删除、替换等。与C语言中的字符串不同,使用string类可以避免许多重复繁琐的操作,使得代码更加简洁和易于维护。另外,string类支持自动内存管理和动态扩容,可以根据需要动态地调整字符串的大小,这也是其与C语言中的字符数组相比的一个优势。

1.字符串是表示字符序列的类

2.标准的字符串类提供了对此类对象的支持,其接口类似于标准字符容器的接口,但添加了专门用于操作单字节字符字符串的设计特性。

3.string类是使用char(即作为它的字符类型,使用它的默认char_traits和分配器类型(关于模板的更多信息,请参阅basic_string)。

4.string类是basic_string模板类的一个实例,它使用char来实例化basic_string模板类,并用char_traits和allocator作为basic_string的默认参数(根于更多的模板信息请参考basic_string)。

5.注意,这个类独立于所使用的编码来处理字节:如果用来处理多字节或变长字符(如UTF-8)的序列,这个类的所有成员(如长度或大小)以及它的迭代器,将仍然按照字节(而不是实际编码的字符)来操作。


总结:


1.string是表示字符串的字符串类

2.该类的接口与常规容器的接口基本相同,再添加了一些专门用来操作string的常规操作。

3.string在底层实际是:basic_string模板类的别名,typedef basic_string<char, char_traits, allocator>string;

4.不能操作多字节或者变长字符的序列。在使用string类时,必须包含#include头文件以及using namespace std;


我们学习string类需要通过查文档来学习:string类文档介绍


string构造

前面提到了我们学习string类需要结合文档来学习,这里我们在文档中查询string类的构造。

ae7167dc988f4ba5913a9a8fce50e092.png


文档下面也对这几个构造函数有解释>


f2697d0fc33c4da6ad6eb68417e36967.png



73f28af7bedf40708deafe5e64699f35.pngGoogle翻译>


029a4a483b5d4b47bcbb089b7124c475.png


cdd1fecc460a41439456ccb5ef28b3cb.png


string();

创建一个空的字符串,长度为零个字符。

eg:


int main()
{
  //构造一个空的字符串
  string s1;
  cout << s1 << endl;
  return 0;
}


7d2a40ac3ab94d7896a40f4438b778d8.png


可以看到打印出来是空白的。


string (const char* s);

构造一个s字符串的拷贝。

eg:

int main()
{
  //构造一个“hello C++”的字符串
  string s0("hello C++");
  cout << s0 << endl;
  return 0;
}


8444c60b43064f1fb260b2b94f161149.png

string (const string& str);

拷贝构造函数,构造一个str字符串的副本。

eg:


int main()
{
  //构造一个“hello C++”的字符串
  string s0("hello C++");
  //通过s0字符串拷贝构造出s2
  string s2(s0);
  cout << s0 << endl;
  cout << s2 << endl;
  return 0;
}

ba8fb2d0b0844a249e79738b9a59ea20.png


string (const string& str, size_t pos, size_t len = npos);

复制从字符位置pos开始并跨越len个字符的str部分(或者直到str的末尾,如果str太短或者len是string::npos)

eg:


int main()
{
  //构造一个“hello World”的字符串
  string s0("hello World");
  string s3(s0, 8, 3);
  cout << s0 << endl;
  cout << s3 << endl;
  return 0;
}


4dd2986b49104dbd917c1a0e37ae541f.png


这里我们可以发现这个构造函数中有一个缺省值npos,我们可以查看这个npos的定义:


245a78f8cc164f18a620e1135f59c4e0.png

这里给npos赋值-1,其补码是全1,npos的类型又是size_t(无符号整型)发生了类型转换,把-1强转为size_t类型,所以将-1的补码全1转换为原码,又是无符号类型,所以原反补都相同,此时原码就成为了全1(整型最大值)。

Google翻译>


b96260f359774bc78a5352c32c2b5360.png


string (const char* s, size_t n);

这个构造函数的作用是从s字符串中拷贝前n个字符赋给构造的对象。

eg:


int main()
{
  string s5("C++ is very Good!", 12);
  cout << s5 << endl;
  return 0;
}


1009d406f11d4719860a5f89e566b9dd.png

string (size_t n, char c);

用字符c的n 个连续拷贝填充字符串。

eg:

int main()
{
  string s6(10, 'c');
  cout << s6 << endl;
  return 0;
}

cdb30f57b3384178b00ea1bf6900b5f7.png


template < class InputIterator >string (InputIterator first, InputIterator last);

范围构造函数

以相同的顺序复制[first,last)范围内的字符序列。

eg:


int main()
{
  string s0("hello C++");
  string s7(s0.begin(), s0.begin() + 7);
  cout << "s0:" << s0 << endl;
  cout << "s7:" << s7 << endl;
  return 0;
}


a21c26f7d7c048628984f525c5918c98.png


string类对象的容量操作

size


498771d720b44a25b7dea32dbb5cff25.png


size的功能返回字符串的有效长度,以字节为单位。这是符合字符串内容的实际字节数,不一定等于它的存储容量。

eg:


int main()
{
  string str("hello C++!");
  cout << str.size() << endl;
  return 0;
}

e1a0dd6efcdc4dd9908e2f3f2c0d6134.png


length


7ca34c756cbf4a67a2c9e0afbe59edfe.png


length的功能与size的功能相同,都是返回字符串的有效长度。

eg:

int main()
{
  string str("hello C++!");
  cout << str.size() << endl;
  cout << str.length() << endl;
  return 0;
}


86bbec01c0b24ed5b297d790936fa54b.png


capacity


28b13767e508497b988eb62b354b9462.png

返回当前为string 分配的存储空间的大小,以字节表示。

容量不一定等于字符串长度。它可以等于或更大,额外的空间允许对象在新字符添加到 string 时优化其操作。

我们在vs2019编译器下编译运行以下代码>


int main()
{
  string str("hello C++!");
  cout << str.size() << endl;
  cout << str.length() << endl;
  cout << str.capacity() << endl;
  cout << str.max_size() << endl;
  return 0;
}


e159468c07df4d058c5ac0a5398182d7.png

可以看到capacity比实际的大小要大一点。

我们再来在g++编译器来运行上面的代码>


11bdb79462914a479f58aaa82287e786.png


我们能看到g++编译器下的capacity和size的大小是相同的,两款编译器下的max_size也是不一样的,所以我们在平时几乎不会用max_size()。(max_size返回的是字符串可达到的最大长度)


empty


983eeee4cabc4321a2904aa044c41ae3.png

测试字符串是否为空
返回字符串是否为空(即它的长度是否为0)
如果字符串长度为0则为true,否则为false。

clear


5ce920295818444d98fad5a3ec3ec77d.png


清除字符串
擦除string 的内容,它变成一个空字符串(长度为0个字符)。

reserve

3e090b0e04b04c2abdfcc757d135b494.png


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


resize

bc3fc0ba1c544371b05b908c332e7020.png

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


注意:

  1. size()与length()方法底层实现原理完全相同,引入size()的原因是为了与其他容器的接口保持一致,一般情况下基本都是用size()。
  2. clear()只是将string中有效字符清空,不改变底层空间大小。


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

operator[]


6acbb09baf1947288f6e5e20370aa5e6.png


他的功能是可以返回pos位置的字符,也就是说可以通过 [ ] 来访问字符串中的某一个字符。

eg:


int main()
{
  string str("hello C++");
  cout << str << endl;
  for (int i = 0; i < str.size(); i++)
  {
    cout << str[i];
  }
  cout << endl;
  return 0;
}


42dbf0482fa8434c953384f3f724d193.png

可以看到我们通过str[i]也可以对str字符串进行遍历输出。

他也有函数重载,一个是被const修饰的一个不能被修改,一个没有被const修饰可以进行修改。这里我们举个栗子>

我们对一个字符串的每个字符都进行++处理,再来打印:


int main()
{
  string str("hello C++");
  cout << str << endl;
  //对字符串的每个字符进行++
  for (int i = 0; i < str.size(); i++)
  {
    str[i]++;
  }
  //通过[]对字符串进行遍历
  for (int i = 0; i < str.size(); i++)
  {
    cout << str[i];
  }
  cout << endl;
  return 0;
}

b617956116274407a32969596395c951.png



可以看到我们把原本的字符串进行了修改,当然如果是一个const修饰的对象调用operator[ ],他就会调用const修饰的operator[ ]函数,被const修饰的对象当然也不能被改变。


iterator迭代器

我们来看这样一段代码>

int main()
{
  string s1("hello C+++");
  string::iterator it = s1.begin();
  while (it != s1.end())
  {
    cout << *it << " ";
    ++it;
  }
  cout << endl;
  return 0;
}


其中s1.begin()是指向的字符串的第一个字符,s1.end()指向的是最后一个字符的下一个。


83a108bca1324bc983d43656a57fdab3.png


可以看到这个函数的功能是:

将迭代器返回到开头
返回指向字符串第一个字符的迭代器。


2bd689b3ae604a47822e6b8e179b4b37.png

相关文章
|
15天前
|
存储 C语言 C++
【C++数据结构——栈与队列】顺序栈的基本运算(头歌实践教学平台习题)【合集】
本关任务:编写一个程序实现顺序栈的基本运算。开始你的任务吧,祝你成功!​ 相关知识 初始化栈 销毁栈 判断栈是否为空 进栈 出栈 取栈顶元素 1.初始化栈 概念:初始化栈是为栈的使用做准备,包括分配内存空间(如果是动态分配)和设置栈的初始状态。栈有顺序栈和链式栈两种常见形式。对于顺序栈,通常需要定义一个数组来存储栈元素,并设置一个变量来记录栈顶位置;对于链式栈,需要定义节点结构,包含数据域和指针域,同时初始化栈顶指针。 示例(顺序栈): 以下是一个简单的顺序栈初始化示例,假设用C语言实现,栈中存储
132 77
|
15天前
|
存储 C++
【C++数据结构——树】哈夫曼树(头歌实践教学平台习题) 【合集】
【数据结构——树】哈夫曼树(头歌实践教学平台习题)【合集】目录 任务描述 相关知识 测试说明 我的通关代码: 测试结果:任务描述 本关任务:编写一个程序构建哈夫曼树和生成哈夫曼编码。 相关知识 为了完成本关任务,你需要掌握: 1.如何构建哈夫曼树, 2.如何生成哈夫曼编码。 测试说明 平台会对你编写的代码进行测试: 测试输入: 1192677541518462450242195190181174157138124123 (用户分别输入所列单词的频度) 预
52 14
【C++数据结构——树】哈夫曼树(头歌实践教学平台习题) 【合集】
|
15天前
|
C++ 芯片
【C++面向对象——类与对象】Computer类(头歌实践教学平台习题)【合集】
声明一个简单的Computer类,含有数据成员芯片(cpu)、内存(ram)、光驱(cdrom)等等,以及两个公有成员函数run、stop。只能在类的内部访问。这是一种数据隐藏的机制,用于保护类的数据不被外部随意修改。根据提示,在右侧编辑器补充代码,平台会对你编写的代码进行测试。成员可以在派生类(继承该类的子类)中访问。成员,在类的外部不能直接访问。可以在类的外部直接访问。为了完成本关任务,你需要掌握。
56 19
|
15天前
|
存储 人工智能 算法
【C++数据结构——图】最短路径(头歌教学实验平台习题) 【合集】
任务描述 本关任务:编写一个程序,利用Dijkstra算法,实现带权有向图的最短路径。 相关知识 为了完成本关任务,你需要掌握:Dijkst本关任务:编写一个程序,利用Dijkstra算法,实现带权有向图的最短路径。为了完成本关任务,你需要掌握:Dijkstra算法。带权有向图:该图对应的二维数组如下所示:Dijkstra算法:Dijkstra算法是指给定一个带权有向图G与源点v,求从v到G中其他顶点的最短路径。Dijkstra算法的具体步骤如下:(1)初始时,S只包含源点,即S={v},v的距离为0。
46 15
|
15天前
|
存储 编译器 数据安全/隐私保护
【C++面向对象——类与对象】CPU类(头歌实践教学平台习题)【合集】
声明一个CPU类,包含等级(rank)、频率(frequency)、电压(voltage)等属性,以及两个公有成员函数run、stop。根据提示,在右侧编辑器补充代码,平台会对你编写的代码进行测试。​ 相关知识 类的声明和使用。 类的声明和对象的声明。 构造函数和析构函数的执行。 一、类的声明和使用 1.类的声明基础 在C++中,类是创建对象的蓝图。类的声明定义了类的成员,包括数据成员(变量)和成员函数(方法)。一个简单的类声明示例如下: classMyClass{ public: int
39 13
|
15天前
|
Java C++
【C++数据结构——树】二叉树的基本运算(头歌实践教学平台习题)【合集】
本关任务:编写一个程序实现二叉树的基本运算。​ 相关知识 创建二叉树 销毁二叉树 查找结点 求二叉树的高度 输出二叉树 //二叉树节点结构体定义 structTreeNode{ intval; TreeNode*left; TreeNode*right; TreeNode(intx):val(x),left(NULL),right(NULL){} }; 创建二叉树 //创建二叉树函数(简单示例,手动构建) TreeNode*create
39 12
|
15天前
|
C++
【C++数据结构——树】二叉树的性质(头歌实践教学平台习题)【合集】
本文档介绍了如何根据二叉树的括号表示串创建二叉树,并计算其结点个数、叶子结点个数、某结点的层次和二叉树的宽度。主要内容包括: 1. **定义二叉树节点结构体**:定义了包含节点值、左子节点指针和右子节点指针的结构体。 2. **实现构建二叉树的函数**:通过解析括号表示串,递归地构建二叉树的各个节点及其子树。 3. **使用示例**:展示了如何调用 `buildTree` 函数构建二叉树并进行简单验证。 4. **计算二叉树属性**: - 计算二叉树节点个数。 - 计算二叉树叶子节点个数。 - 计算某节点的层次。 - 计算二叉树的宽度。 最后,提供了测试说明及通关代
38 10
|
15天前
|
编译器 数据安全/隐私保护 C++
【C++面向对象——继承与派生】派生类的应用(头歌实践教学平台习题)【合集】
本实验旨在学习类的继承关系、不同继承方式下的访问控制及利用虚基类解决二义性问题。主要内容包括: 1. **类的继承关系基础概念**:介绍继承的定义及声明派生类的语法。 2. **不同继承方式下对基类成员的访问控制**:详细说明`public`、`private`和`protected`继承方式对基类成员的访问权限影响。 3. **利用虚基类解决二义性问题**:解释多继承中可能出现的二义性及其解决方案——虚基类。 实验任务要求从`people`类派生出`student`、`teacher`、`graduate`和`TA`类,添加特定属性并测试这些类的功能。最终通过创建教师和助教实例,验证代码
43 5
|
15天前
|
存储 算法 搜索推荐
【C++面向对象——群体类和群体数据的组织】实现含排序功能的数组类(头歌实践教学平台习题)【合集】
1. **相关排序和查找算法的原理**:介绍直接插入排序、直接选择排序、冒泡排序和顺序查找的基本原理及其实现代码。 2. **C++ 类与成员函数的定义**:讲解如何定义`Array`类,包括类的声明和实现,以及成员函数的定义与调用。 3. **数组作为类的成员变量的处理**:探讨内存管理和正确访问数组元素的方法,确保在类中正确使用动态分配的数组。 4. **函数参数传递与返回值处理**:解释排序和查找函数的参数传递方式及返回值处理,确保函数功能正确实现。 通过掌握这些知识,可以顺利地将排序和查找算法封装到`Array`类中,并进行测试验证。编程要求是在右侧编辑器补充代码以实现三种排序算法
34 5
|
15天前
|
Serverless 编译器 C++
【C++面向对象——类的多态性与虚函数】计算图像面积(头歌实践教学平台习题)【合集】
本任务要求设计一个矩形类、圆形类和图形基类,计算并输出相应图形面积。相关知识点包括纯虚函数和抽象类的使用。 **目录:** - 任务描述 - 相关知识 - 纯虚函数 - 特点 - 使用场景 - 作用 - 注意事项 - 相关概念对比 - 抽象类的使用 - 定义与概念 - 使用场景 - 编程要求 - 测试说明 - 通关代码 - 测试结果 **任务概述:** 1. **图形基类(Shape)**:包含纯虚函数 `void PrintArea()`。 2. **矩形类(Rectangle)**:继承 Shape 类,重写 `Print
41 4