🔥个人主页: Forcible Bug Maker
🔥专栏: STL || C++
目录
- 前言
- 🌈关于string类
- 🌈string类的成员函数
- 🔥默认成员函数
- ==string类对象的构造(constructor)==
- ==string类对象的析构==
- ==string类对象的赋值运算符重载==
- 🔥迭代器接口(iterators)
- ==begin==
- ==end==
- ==rbegin==
- ==rend==
- ==cbgin,cend,erbegin和crend==
- 结语
前言
本篇博客主要内容:STL库中string类的默认成员函数和各种迭代器接口的介绍和使用。
在开始我们string类
使用接口的讲解之前,想先讲讲为什么我们要学习string类。在C语言
中,字符串是以’\0’结尾的一些字符的集合,为了操作方便,C标准库中提供了一些str系列的库函数,但是这些库函数与字符串是分离开的,不太符合OOP的思想,而且底层空间需要用户自己管理,稍不留神可能还会越界访问。
简单说,string就是C++STL标准库中便于维护和操作字符串的一个类。
学习STL过程中,我们主要参考这个网站中的文档:https://cplusplus.com,虽然里面的内容是全英的,但是不用担心,我会带着大家一一去翻译并讲解其中的接口。
🌈关于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,alloctor> string;
- 不能操作多字节或变长字节的序列。
在使用string类时,必须包含相应头文件
#include<string>
以及using namespace std;
🌈string类的成员函数
C++STL库的string类中重载的成员函数非常之多,但并不是每个接口都有着其必要性。string类是C++中最先被开发和实验的类,由于向前兼容等各种原因,其实其中也不免存在一些函数的冗余,在上百个重载的接口中,真正常用的也就是那十几二十个。对于一些相同道理的重载,我会做简单解释而不会展开讲,避免文章内容的冗余。
🔥默认成员函数
string类对象的构造(constructor)
string类提供了七种方式的重载用于构造string类对象。
(1)无参构造(默认构造)
string();
构造出一个空的string,里面存储0个字符元素。
(2)拷贝构造
string(const string& str);
构造出一个和str对象内容相同的拷贝对象。
(3)字串构造
string (const string& str, size_t pos, size_t len = npos);
拷贝str从pos位置开始跨越len个长度的元素到新创建的对象中(如果字符串太短或未提供第三个参数,直接从pos位置拷贝到字符串的末尾)。
(4)通过字符串构造
string (const char* s);
通过s指向的以 空字符(‘\0’) 结尾的字符串创建新的string类型的对象。
(5)通过部分字符串构造
string (const char* s, size_t n);
从s指向的字符串中拷贝前n个字符创建新的string类型的对象。
(6)字符填充构造
string (size_t n, char c);
用连续的n个c构成的串创建string类型的对象。
(7)迭代器区间构造
template <class InputIterator>
string (InputIterator first, InputIterator last);
通过迭代器区间 [first,last) 之间的元素,创建顺序相同的新string对象。
你现在可能不知道迭代器是什么,不过你可以先将它理解成是一种指针,指向string中的元素,可以通过++或–来改变这种指针的指向。
使用样例:
// string类的构造
#include <iostream>
#include <string>
// 由于下面使用了std::,所以此处不用命名空间展开
int main()
{
std::string s0("Initial string");
// constructors used in the same order as described above:
std::string s1;
std::string s2(s0);
std::string s3(s0, 8, 3);
std::string s4("A character sequence");
std::string s5("Another character sequence", 12);
std::string s6a(10, 'x');
std::string s6b(10, 42); // 42是字符'*'的ASCII码值
std::string s7(s0.begin(), s0.begin() + 7);
// 打印构造出来的字符串内容,这里重载了流插入>>和流提取<<运算符。
std::cout << "s1: " << s1 << "\ns2: " << s2 << "\ns3: " << s3;
std::cout << "\ns4: " << s4 << "\ns5: " << s5 << "\ns6a: " << s6a;
std::cout << "\ns6b: " << s6b << "\ns7: " << s7 << '\n';
return 0;
}
string类对象的析构
如你所见,析构只有一种,之前在类和对象篇已经讲过了,析构函数是不能重载的。
析构函数编译器会自动调用,释放资源,所以就不需要你操心了。
string类对象的赋值运算符重载
string类提供了三种赋值运算符重载,用于给已经创建好的对象赋值。
(1)string对象赋值
string& operator=(const string& str);
将str的内容拷贝到被赋值的对象中。
返回值:被赋值对象(*this)的引用。
(2)字符串赋值
string& operator=(const char* s);
通过一个已有的字符串给string赋值,其实不重载此函数也可以达到效果(隐式类型转换),但是考虑到隐式类型转换是有效率损耗的,故标准库中还是重载了此成员函数。
返回值:被赋值对象(*this)的引用。
(3)字符赋值
string& operator=(char c);
将字符直接赋值给串。
返回值:被赋值对象(*this)的引用。
使用样例:
#include <iostream>
#include <string>
using namespace std;
int main()
{
std::string str1, str2, str3;
str1 = "Test string. "; // c-string
str2 = str1; //string
str3 = 'x'; //char
cout << str1 << endl;
cout << str2 << endl;
cout << str3 << endl;
return 0;
}
🔥迭代器接口(iterators)
什么是迭代器接口,就是成员函数返回迭代器的接口,也就是获取迭代器的方法。
我们可以先将迭代器想象成指向容器元素的指针,而这些成员函数的作用就是让我们获取这些指针。同时我们可以通过+,-,++或–等运算符调整这种指针的指向。
begin
iterator begin();
const_iterator begin() const;
这两个成员函数都返回一个指向string串中第一个元素的正向迭代器。
end
iterator end();
const_iterator end() const;
这两个成员函数都返回一个指向string串中最后一个元素下一位的正向迭代器。
使用样例:
// string::begin/end
#include <iostream>
#include <string>
using namespace std;
int main()
{
std::string str("Test string");
string::iterator it = str.begin();
for (; it != str.end(); ++it)
cout << *it;
cout << endl;
return 0;
}
string的迭代器(iterator)是一个定义在string类内部的一个类型,它能指向string对象的元素,而迭代器的对象可以通过string提供的成员函数获取。上述代码案例中的迭代器it被我们称为正向迭代器,++使其向后移动指向下一元素,–使其向前移动指向上一元素。
而接下来我们要讲的rbegin和rend,是获取反向迭代器的两个接口函数。
rbegin
reverse_iterator rbegin();
const_reverse_iterator rbegin() const;
这两个成员函数都返回一个指向string串中最后一个元素的反向迭代器。
rend
reverse_iterator rend();
const_reverse_iterator rend() const;
这两个成员函数都返回一个指向string串中第一个元素上一位的反向迭代器。
使用样例:
// string::rbegin/rend
#include <iostream>
#include <string>
using namespace std;
int main()
{
string str("now step live...");
string::reverse_iterator rit = str.rbegin();
for (; rit != str.rend(); ++rit)
cout << *rit;
cout << endl;
return 0;
}
同理,string的反向迭代器(reverse_iterator)是一个定义在string类内部的一个类型,它能指向string对象的元素,而反向迭代器的对象可以通过string提供的成员函数获取。反向迭代器,++使其向前移动指向上一元素,–使其后前移动指向下一元素。
cbgin,cend,erbegin和crend
这几个作为C++11新增语法,专门提供了获取const类型的四种迭代器接口,使迭代器可以读,但不能通过迭代器解引用更改串中的内容。这里就不多做赘述。
结语
本篇博客讲了关于string类构造的七种重载,赋值运算符的三种重载,以及各种string迭代器接口的使用。后面博主会继续分享关于string类的使用以及STL更多的内容,感谢大家的支持。♥