【C++要笑着学】编码的由来 | basic_string模板类 | string类的常用接口讲解 | 学会查文档(三)

简介: 好久不见!前段时间比较忙,更新频率有所减缓。好在现在快忙完了,我又有时间更文咯,还希望大伙能多多支持!我将会呈现出更多高质量的博客给大家!

Ⅴ.  string 类对象的修改操作


函数名称 功能说明
push_back 在字符串后尾插字符c
append 在字符后追加一个字符串
operator+=    (重点) 在字符串后追加字符串str
c_str              (重点) 返回c格式字符串
find  +  npos  (重点) 从字符串pos位置开始往后找字符c,返回该字符自字符串中的位置
rfind 从字符串pos位置开始往前找字符c,返回该字符在字符串中的位置
substr 在str中从pos位置开始,截取n个字符,然后将其返回

0x00  用来插入字符的 push_back()

string 用的最多的可能就是插入了。


📚 push_back:插入一个字符,但只能插入一个字符


int main(void)
{
  string s1("hello, ");
  s1.push_back('C');
  cout << s1 << endl;
  return 0;
}

06d35eb8f34057d153d58bb45d561696_e8e333a40d6a41379f21267b0fd198d6.png


0x01  用来插入字符串的 append()

📚 append:可以插入字符串,它就能干很多

ec62b7053729fbb1854500f2847c7e94_9f0fcb3d51be447694d1a28a26979501.png

(关于迭代器部分的知识我下面几章会说)


💬 这里面用的最多的就是 (1) 和 (3) ,插入一个完整字符串和插入一个对象。


int main(void)
{
  string s1("hello, ");
  s1.append("C++");
  cout << s1 << endl;
  return 0;
}

ad665c4750a161bb8bcaf7816f01a769_1b6258acb4a04592b55db254060c19ec.png

097ca31634dbab0f7588e9ff3c3ad8eb_01735f95ae054400a864026aff6973f5.png


0x02  用起来爽到飞起的 operator+=  


📚 还有用起来更舒服的 —— operator+=

int main(void)
{
  string s1("hello, ");
  s1 += 'C';    // 加个字符
  cout << s1 << endl;
  s1 += ' ';    // 加个空格
  cout << s1 << endl;
  s1 += "C++";  // 加字符串
  cout << s1 << endl;
  return 0;
}

a115b76c9162fdfec01033ccc70a6280_292a4a2813e44f2d8fba29a57aa144c4.png


0x03  返回C格式字符串的 c_str

📚 返回字符串指针


💬 代码演示:


void Test() {
  string s1("hello string");
  cout << s1 << endl;  // 调用重载的流插入运算符打印的
  cout << s1.c_str() << endl;  // 这是调字符串打印的,c_str 是遇到 \0 结束的,
}

371a6bd668d2aaf4a45bac7149b8bdd1_73e73e81c8cc47c8a1b30a24ecce196c.png

(结果看起来很像)


❓ 那这个 c_str 有什么意义呢?


// 获取file后缀
void Test() {
  string file("test.txt"); 
  FILE* pf = fopen(file.c_str(), "w");  
}

比如这里需要打开文件,fopen 第一个参数要求是 const char*,


所以这里怎么能直接放 string 是不行的,这时候可以用 .c_str()  就可以把字符串的地址返回出来。


0x04  追查字符位置的 find() 和砍字符串的 substr()

如果我们要取像这样的文件后缀名 .txt  或  .cpp,该怎么取?


使用 find() 就可以很好地解决!


📚 find:从字符串 pos 位置开始往后找字符 c ,返回该字符在字符串中的位置。


string file("test.txt");
FILE* pf = fopen(file.c_str(), "w");
size_t pos = file.find('.');  // 不写第二个参数,默认从0位置开始找。

6eb8ba1b46d0b1ea9ebba7b75dae1ad2_49f36b3cc9894c7aa430c151414abce1.png

通过文档,了解它的返回值后,我们就知道如何判断它了。


在这之前我们还要介绍一下可以砍字符串的 substr


📚 substr:在 str 中从 pos 位置开始,截取 n 个字符,然后将其返回。

5ac6165f8f5c767e704b4d032182388f_236047d8b54647c78cc857401834799e.png


💬 判断部分:


void Test() {
  string file("test.txt");
  FILE* pf = fopen(file.c_str(), "w");
  size_t pos = file.find('.');  // 不写第二个参数,默认从0位置开始找。
  if (pos != string::npos) {
  string suffix = file.substr(pos, file.size() - pos);
  cout << suffix << endl;
  }
}

aa13778d6dca66ad0854363bdeb1cfb3_951e7ccdaab0471892cef51b7284f85e.png


用 find 截取到 " . " 存入 pos,pos 的位置就是起始位置,就作为 substr() 的第一个参数 pos,


substr() 的第二个参数 len ,我们可以通过 size() - pos 来算出要截取多少个字符。


其实第二个参数你可以不给,看上面的文档可知,参数二 len 的缺省值是 npos,


并且 len 是 size_t 的,也就是说如果你不写,它默认会是我们所说的 "上亿的大数"


这种数肯定会超出字符串长度。而且要求是获取后缀,


因为考虑到 .txt  .cpp 这样的后缀,后面也没东西,这时候就会有多少出多少


💬 所以如果你懒的话甚至可以直接这样写:


void Test() {
  string file("test.txt");
  FILE* pf = fopen(file.c_str(), "w");
  size_t pos = file.find('.');
  if (pos != string::npos) {
  string suffix = file.substr(pos);  // 给个pos就完事了
  cout << suffix << endl;
  }
}

5f995c168b525d270c341b3be0eaba98_09ea9ca3d8cc4ddc9dc283472aa2d812.png


find 是很常用的!也是很重要的!我们还可以拿它取 url :


void Test() {
  // 创建字符串记录网址
  string url("http://www.cplusplus.com/reference/string/string/find/");
  // 取出协议
  size_t pos1 = url.find(':'); 
  string protocol = url.substr(0, pos1 - 0);
  cout << protocol << endl; 
  // 取出域名
  size_t pos2 = url.find('/', pos1 + 3);  // 冒号位置+3开始往后找(w位置) 
  string domain = url.substr(pos1 + 3, pos2 - (pos1 + 3));
  cout << domain << endl;
  // 取出路径
  string uri = url.substr(pos2 + 1);
  cout << uri << endl;
}

3a19de511c46c497773af5a72338876a_e837f3c5d7e84086b9cd3e7d8980d2fa.png


0x05  反向追查字符串的 rfind()  

但是,如果文件后缀名是 .txt.zip 呢?我们要取 .zip 啊。


如果还是按刚才的方式去取,会从第一个 " . " 开始选取·:


void Test() {
  string file("test.txt.zip");
  FILE* pf = fopen(file.c_str(), "w");
  size_t pos = file.find('.');
  if (pos != string::npos) {
  string suffix = file.substr(pos, file.size() - pos);
  cout << suffix << endl;
  }
}

fcdd20e3e8099b45d00845f0870745ca_959bdd39075d4c2e95c1fa0e7cc51ee3.png

(最后得到的结果不是我们想要的)


这种情况,就需要从右往左去找了。


📚 rfind:从字符串 pos 位置开始往前找字符 c ,返回该字符在字符串中的位置。


rfind,就是 right find —— 从右往左去找字符。

58c15021d3c12d3fcdf58b3b7bf67a48_8aa128a3085444afa56df7fd4e69b6a4.png

💬 这里我们只需要改成 rfind 就可以了:


void Test() {
  string file("test.txt.zip");
  FILE* pf = fopen(file.c_str(), "w");
  size_t pos = file.rfind('.');
  if (pos != string::npos) {
  string suffix = file.substr(pos, file.size() - pos);
  cout << suffix << endl;
  }
}

2f931b8b2aca39a5e61f3ab1d01ca0f4_856c71da4cee4a1398f754ad7c9cbed9.png


Ⅵ.  string 类非成员函数

函数 功能说明
operator+ 尽量少用,因为传值返回,导致深拷贝效率低
operator>>   (重点) 输入运算符重载
operator<<   (重点) 输出运算符重载
getline          (重点) 获取一行字符串
relational operators  (重点) 大小比较


string类中还有很多操作,这里就不一一列举了,需要用到时不明白可以查文档。

……

End.

相关文章
|
4天前
|
算法 C++ 容器
【C++】string模拟实现
【C++】string模拟实现
12 1
|
3天前
|
存储 算法 搜索推荐
C++|STL简介-string-vector基础运用
C++|STL简介-string-vector基础运用
|
3天前
|
存储 编译器 C++
|
4天前
|
C语言 C++ 容器
C++ string类
C++ string类
9 0
|
4天前
|
编译器 C++
【C++】继续学习 string类 吧
首先不得不说的是由于历史原因,string的接口多达130多个,简直冗杂… 所以学习过程中,我们只需要选取常用的,好用的来进行使用即可(有种垃圾堆里翻美食的感觉)
9 1
|
4天前
|
算法 安全 程序员
【C++】STL学习之旅——初识STL,认识string类
现在我正式开始学习STL,这让我期待好久了,一想到不用手撕链表,手搓堆栈,心里非常爽
16 0
|
4天前
|
存储 安全 测试技术
【C++】string学习 — 手搓string类项目
C++ 的 string 类是 C++ 标准库中提供的一个用于处理字符串的类。它在 C++ 的历史中扮演了重要的角色,为字符串处理提供了更加方便、高效的方法。
18 0
【C++】string学习 — 手搓string类项目
|
4天前
|
存储 算法 编译器
C++的模板与泛型编程探秘
C++的模板与泛型编程探秘
11 0
|
4天前
|
存储 C++ 容器
【C++从练气到飞升】09---string语法指南(二)
【C++从练气到飞升】09---string语法指南(二)
|
4天前
|
存储 Linux C语言
【C++从练气到飞升】09---string语法指南(一)
【C++从练气到飞升】09---string语法指南(一)