【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.

相关文章
|
5月前
|
缓存 算法 程序员
C++STL底层原理:探秘标准模板库的内部机制
🌟蒋星熠Jaxonic带你深入STL底层:从容器内存管理到红黑树、哈希表,剖析迭代器、算法与分配器核心机制,揭秘C++标准库的高效设计哲学与性能优化实践。
C++STL底层原理:探秘标准模板库的内部机制
|
5月前
|
编解码 Java 开发者
Java String类的关键方法总结
以上总结了Java `String` 类最常见和重要功能性方法。每种操作都对应着日常编程任务,并且理解每种操作如何影响及处理 `Strings` 对于任何使用 Java 的开发者来说都至关重要。
378 5
|
9月前
|
存储 算法 安全
c++模板进阶操作——非类型模板参数、模板的特化以及模板的分离编译
在 C++ 中,仿函数(Functor)是指重载了函数调用运算符()的对象。仿函数可以像普通函数一样被调用,但它们实际上是对象,可以携带状态并具有更多功能。与普通函数相比,仿函数具有更强的灵活性和可扩展性。仿函数通常通过定义一个包含operator()的类来实现。public:// 重载函数调用运算符Add add;// 创建 Add 类的对象// 使用仿函数return 0;
292 0
|
9月前
|
人工智能 机器人 编译器
c++模板初阶----函数模板与类模板
class 类模板名private://类内成员声明class Apublic:A(T val):a(val){}private:T a;return 0;运行结果:注意:类模板中的成员函数若是放在类外定义时,需要加模板参数列表。return 0;
246 0
|
9月前
|
存储 编译器 C语言
关于string的‘\0‘与string,vector构造特点,反迭代器与迭代器类等的讨论
你真的了解string的'\0'么?你知道创建一个string a("abcddddddddddddddddddddddddd", 16);这样的string对象要创建多少个对象么?你知道string与vector进行扩容时进行了怎么的操作么?你知道怎么求Vector 最大 最小值 索引 位置么?
237 0
|
缓存 安全 Java
《从头开始学java,一天一个知识点》之:字符串处理:String类的核心API
🌱 **《字符串处理:String类的核心API》一分钟速通!** 本文快速介绍Java中String类的3个高频API:`substring`、`indexOf`和`split`,并通过代码示例展示其用法。重点提示:`substring`的结束索引不包含该位置,`split`支持正则表达式。进一步探讨了String不可变性的高效设计原理及企业级编码规范,如避免使用`new String()`、拼接时使用`StringBuilder`等。最后通过互动解密游戏帮助读者巩固知识。 (上一篇:《多维数组与常见操作》 | 下一篇预告:《输入与输出:Scanner与System类》)
334 11
|
编译器 C++
模板(C++)
本内容主要讲解了C++中的函数模板与类模板。函数模板是一个与类型无关的函数家族,使用时根据实参类型生成特定版本,其定义可用`typename`或`class`作为关键字。函数模板实例化分为隐式和显式,前者由编译器推导类型,后者手动指定类型。同时,非模板函数优先于同名模板函数调用,且模板函数不支持自动类型转换。类模板则通过在类名后加`&lt;&gt;`指定类型实例化,生成具体类。最后,语录鼓励大家继续努力,技术不断进步!
课时14:Java数据类型划分(初见String类)
课时14介绍Java数据类型,重点初见String类。通过三个范例讲解:观察String型变量、&quot;+&quot;操作符的使用问题及转义字符的应用。String不是基本数据类型而是引用类型,但使用方式类似基本类型。课程涵盖字符串连接、数学运算与字符串混合使用时的注意事项以及常用转义字符的用法。
362 9
|
存储 JavaScript Java
课时44:String类对象两种实例化方式比较
本次课程的主要讨论了两种处理模式在Java程序中的应用,直接赋值和构造方法实例化。此外,还讨论了字符串池的概念,指出在Java程序的底层,DOM提供了专门的字符串池,用于存储和查找字符串。 1.直接赋值的对象化模式 2.字符串池的概念 3.构造方法实例化
240 1
|
Java
【编程基础知识】(讲解+示例实战)方法参数的传递机制(值传递及地址传递)以及String类的对象的不可变性
本文深入探讨了Java中方法参数的传递机制,包括值传递和引用传递的区别,以及String类对象的不可变性。通过详细讲解和示例代码,帮助读者理解参数传递的内部原理,并掌握在实际编程中正确处理参数传递的方法。关键词:Java, 方法参数传递, 值传递, 引用传递, String不可变性。
345 1
【编程基础知识】(讲解+示例实战)方法参数的传递机制(值传递及地址传递)以及String类的对象的不可变性