【C++精华铺】9.STL string

目录

1. string类的优势

2. string类的常用接口

2.1 常用构造

1. 空串构造:string();

2. C串构造:string(const char* s);

3. 拷贝构造:string(const string& str);

4. 字符填充构造:string(size_t n, char c);

5. 迭代器构造:string(InputIterator first, InputIterator last);

2.2  string容量操作

1. size_t size(),size_t length()

 2. size_t capacity() const

 3. bool empty() const

 4. void clear()

 5. void reserve(size_t n = 0)

 6. void resize(size_t n), void resize(size_t char c)

2.3 类对象的访问

1. char& operator[](size_t pos),const char& operator[](size_t pos) const

 2. begin()、end()

 3. rbegin()、rend()

2.4 string对象处理函数

1. push_back()

2. append()

 3. operator+=

 4. c_str()

 5. insert()

6. erase()

7. find()、rfind()

8. substr()

9. getline()


1. string类的优势

         STL(Standard Template Library)库中的string类是一个字符串类,它提供了管理字符串的各种方法和功能。它是一个可变长度的字符序列,可以自动调整自身大小以适应字符串的长度变化。

        在之前我们操作字符串都是通过自己去实现相关的函数来进行操作,而且稍不留神就会内存泄漏,在C++中我们更倾向于使用string类来完成和字符串相关的操作。

        使用string类相比于使用C语言的字符串有以下优点:

  1. 更加安全:C语言的字符串没有自带长度信息,容易造成内存越界等安全问题。而string类包含有字符串长度信息,能够避免这类问题。

  2. 更加方便:C语言中处理字符串需要使用一系列函数,如strlen、strcmp、strcat等等,使用起来比较繁琐。而string类提供了一系列方法,如length、compare、append等等,使用起来更加直观方便。

  3. 更加灵活:string类可以动态地改变字符串的长度,而C语言的字符串长度通常是固定的。

  4. 更加高效:string类内部实现了很多优化,如字符串复制采用了引用计数和写时复制等技术,因此性能相对于C语言的字符串更高。

2. string类的常用接口

2.1 常用构造

        C++98中支持7种构造函数 ,如下:

  1. default (1)  string();
  2. copy(2) string(const string& str);
  3. substring(3) string(const string& str, size_t pos, size_t len = npos);
  4. from c - string(4) string(const char* s);
  5. from sequence(5) string(const char* s, size_t n);
  6. fill(6) string(size_t n, char c);
  7. range(7) template <class InputIterator>                                                   string(InputIterator first, InputIterator last);

         但是我们只对其中的五个进行讲解:

1. 空串构造:string();

        这个构造函数会构造一个空字符串,长度为0个字符,但是容量不一定为零,比如在vs下空string对象的容量是15,不同环境可能有所不同,主要为了避免频繁的扩容操作。

#include<iostream>
#include<string>
using namespace std;
int main()
{string s1; //创建一个空字符串cout << s1 << endl;cout << s1.capacity() << endl;
}

 输出:

2. C串构造:string(const char* s);

        以null结尾的字符序列(C串)构造string对象。

int main()
{string s1("hello world!");cout << s1 << endl;}

输出:

3. 拷贝构造:string(const string& str);

        利用一个已经存在的string对象构造对象。

int main()
{string s1("hello world!");cout << s1 << endl;string s2(s1);cout << s2 << endl;
}

 输出:

4. 字符填充构造:string(size_t n, char c);

         利用n个字符构造string对象。

int main()
{string s1(5, 'c');cout << s1 << endl;
}

输出:

5. 迭代器构造:template <class InputIterator>                                                                             string(InputIterator first, InputIterator last);

        使用迭代器区间来构造string对象。

int main()
{string s1(5, 'c');cout << s1 << endl;string s2(s1.begin(), s1.end());cout << s2 << endl;
}

输出:

2.2  string容量操作

1. size_t size(),size_t length()

        size()和length()都是返回字符串的有效字符长度,俩个函数的实现原理也完全相同,引入size()是为了和其他容器的接口保持一致。

int main()
{string s1(5, 'c');cout << s1.length() << endl;cout << s1.size() << endl;
}

输出:

 2. size_t capacity() const

        capacity()是返回空间的总大小,在vs下对象的初始容量是15,当存储的对象超过这个大小的时候才会去开空间,在新开的空间存储数据。

int main()
{string s1(5, 'c');cout << s1.capacity() << endl;string s2(16, 'c');cout << s2.capacity() << endl;
}

输出:

 3. bool empty() const

        检测字符串是否是空串,是返回true,不是返回false。

int main()
{string s1(5, 'c');if (s1.empty()){cout << "true" << endl;cout << "空" << endl;}else{cout << "false" << endl;cout << "非空" << endl;}
}

输出:

 4. void clear()

        清空对像中的有效字符。

int main()
{string s1("hello world!");cout << s1 << endl;cout << "--------------------------------" << endl;s1.clear();cout << s1 << endl;
}

输出:

 5. void reserve(size_t n = 0)

        reserve()是为了给对象预留空间,如果我们提前得知字符串需要的空间我们就可以提前开好,避免频繁扩容带来的性能消耗。当reserve的参数小于string底层空间大小的时候,reserve就不会对容量进行处理。(一般为了保障程序的安全性,reserve()会多开出一部分空间)。

int main()
{string s1(10,'x');s1.reserve(20);cout << s1.capacity() << endl;s1.reserve(2);cout << s1.capacity() << endl;
}

输出:

 6. void resize(size_t n), void resize(size_t char c)

        修改字符有效个数为n,多出的空间用字符c填充。如果n小于有效字符数,本质就是删字符操作。如果容量不够会扩容。

int main()
{string s1(10, 'x');cout << s1 << endl;s1.resize(20, 'c');cout << s1 << endl;s1.resize(2);cout << s1 << endl;
}

输出:

2.3 类对象的访问

1. char& operator[](size_t pos),const char& operator[](size_t pos) const

        返回pos位的字符,并且可以对非const对象进行修改。

int main()
{string s1(10, '1');cout << s1 << endl;s1[4] = '0';cout << s1 << endl;for (int i = 0; i < 10; i++){cout << s1[i] << ' ';}
}

输出:

 2. begin()、end()

        begin()返回第一个字符位置的迭代器,end()返回最后一个字符的下一个位置的迭代器。begin()和end()都返回了自己的const版本。

int main()
{string s1(10, '1');auto it = s1.begin();for (; it < s1.end(); it++){cout << *it;}
}

输出:

 3. rbegin()、rend()

        rbegin()获取一个反向迭代器,指向其反向开头,rend()获取一个反向迭代器,指向其反向结尾的前一个理论元素。

int main()
{string s1("123456789");cout << s1 << endl;auto it = s1.rbegin();for (; it < s1.rend(); it++){cout << *it;}
}

输出:

2.4 string对象处理函数

1. push_back()

        在字符串后尾插字符c。

int main()
{string s1;s1.push_back('1');cout << s1;
}

输出:

2. append()

        append()函数在在C++98有六种重载形式,用于在字符串后追加一个字符串。

  1. string(1)       string& append(const string& str);
  2. substring(2) string& append(const string& str, size_t subpos, size_t sublen);
  3. c - string(3)  string & append(const char* s);
  4. buffer(4)       string& append(const char* s, size_t n);
  5. fill(5)             string& append(size_t n, char c);
  6. range(6)       template <class InputIterator>                                                                                           string& append(InputIterator first, InputIterator last);
int main()
{string s1("hello");string s2(" world");cout << s1 << endl;s1.append(s2); //在s1结尾追加一个s1cout << s1 << endl;s1.append(s2, 1, 2); //在s1结尾追加s1的下标为1往后的俩个字符cout << s1 << endl;s1.append("C串");   //在s1后追加一个C串cout << s1 << endl;s1.append("123456", 2);//在s1后追加字符串的前 n 个字符。cout << s1 << endl;s1.append(5, 'c');  //在s1后追加5个字符‘c’cout << s1 << endl;string s3("CSDN");s1.append(s3.begin(), s3.end());//在s1后追加一个迭代器区间cout << s1 << endl;
}

 输出:

 3. operator+=

        operator+=()和append()功能类似,都是在字符串后追加内容,但是我们在实际开发中更愿意使用operator+=。

int main()
{string s1("hello ");string s2("world!");s1 += s2;cout << s1 << endl;
}

输出:

 4. c_str()

        c_str()会返回string对象存储字符串的首地址,用户可以通过这个接口来适配c语言中的某些场景。

int main()
{string s1("hello world!");printf("%s", s1.c_str());
}

输出:

 5. insert()

        insert()支持在pos位对字符串进行插入操作,但是由于pos位插入涉及到元素的挪动,而挪动元素会使性能下降,所以insert()要少用。insert()有多种重载形式,如下:

  1. string(1)       string& insert(size_t pos, const string& str);
  2. substring(2) string& insert(size_t pos, const string& str, size_t subpos, size_t sublen);
  3. c - string(3)  string & insert(size_t pos, const char* s);
  4. buffer(4)       string& insert(size_t pos, const char* s, size_t n);
  5. fill(5)             string& insert(size_t pos, size_t n, char c);                                                                       void insert(iterator p, size_t n, char c);
  6. single character(6) iterator insert(iterator p, char c);
  7. range(7)        template <class InputIterator>                                                                                          void insert(iterator p, InputIterator first, InputIterator last);
int main()
{string s1("string");cout << s1 << endl;s1.insert(2, "*********");cout << s1 << endl;
}

输出:

6. erase()

        erase()是删除字符串的一部分,但是不会改变string对象的容量,如果给的删除字符个数大于pos位后的字符个数,会将后面的全部删除。如果没有给pos位,则默认清空字符串。如果不给删除个数则默认是无符号整型npos(-1)即约等于42亿,就会将pos位后面的字符序列全部删除。

  1. sequence(1) string& erase(size_t pos = 0, size_t len = npos);
  2. character(2) iterator erase(iterator p);
  3. range(3)       iterator erase(iterator first, iterator last);
int main()
{string s1("1234567890");s1.erase(2, 3);//从pos位2开始依次删除三个字符,即删除"345"cout << s1 << endl;s1.erase(s1.begin() + 5);cout << s1 << endl;//删除迭代器位置的字符s1.erase(s1.begin(), s1.begin() + 2);//删除beigin()位置和后面1位字符cout << s1 << endl;
}

 输出:

7. find()、rfind()

        find()函数用于查找某字符串中是否包含某个字符或者字符串,会从pos位往后查找,rfind()则是从pos位往前查找。找到后会返回字符首次出现的pos位或者字符串首次出现的pos位。如果没有找到则会返回npos(无符号整型‘-1’)。

  1. string(1)       size_t find(const string& str, size_t pos = 0) const;
  2. c - string(2)  size_t find(const char* s, size_t pos = 0) const;
  3. buffer(3)       size_t find(const char* s, size_t pos, size_t n) const;
  4. character(4) size_t find(char c, size_t pos = 0) const;
int main()
{string s1("hello world");size_t pos = s1.find("llo", 0);cout << pos << endl;
}

输出:

8. substr()

        substr()会获取pos后包括pos位的n个字符并且返回一个string对象,如果没有给获取长度n,则会默认将后面的包括pos位的所有字符全部获取并且返回一个string对象。

  • string substr(size_t pos = 0, size_t len = npos) const;
int main()
{string s1("123456789");cout << s1 << endl;string s2(s1.substr(2, 5));cout << s2 << endl;
}

 输出:

9. getline()

         从 istream 中提取字符并将其存储到 str 中,直到找到分隔字符 delim(或 (2) 的换行符“\n”)。如果在 is 中到达文件末尾或在输入操作期间发生其他错误,则提取也会停止。如果找到分隔符,则将其提取并丢弃(即不存储它,下一个输入操作将在它之后开始)。请注意,调用之前的任何内容都将替换为新提取的序列。每个提取的字符都追加到字符串中,就像调用其成员push_back一样。(不以空格作为分隔符,可以读取空格符)

  • (1)istream& getline(istream& is, string& str, char delim);
  • (2)istream& getline(istream& is, string& str);

 

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

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

相关文章

数据结构之哈希

哈希 1. 哈希概念2. 哈希冲突3. 哈希冲突解决3.1 哈希表的闭散列3.2 哈希表的开散列 2. 哈希的应用2.1 位图2.2 布隆过滤器 哈希&#xff08;Hash&#xff09;是一种将任意长度的二进制明文映射为较短的二进制串的算法。它是一种重要的存储方式&#xff0c;也是一种常见的检索方…

【Linux网络】TCP UDP socket HTTP webSocket之间的区别

目录 一、OSI & TCP/IP模型 二、几者之间的关系 三、HTTP 四、Socket 五、WebSocket 5.1、WebSocket 优点 一、OSI & TCP/IP模型 首先我们要了解OSI七层模型&#xff0c;和预支对应的TCP/IP 四层的模型。 用下面的图可以看出&#xff0c;TCP UDP 工作在传输层&…

Android Studio开发之路 (五)导入OpenCV以及报错解决

一、步骤 官网下载opencv包&#xff08;我下的是4.7.0&#xff09;并解压&#xff0c;openvc官网 先创建一个空项目&#xff0c;简单跑一下能正常输出helloworld 点击file->new->Import Module选择解压之后的opencv-android-sdk文件夹中的SDk文件夹&#xff0c; modu…

Visual Studio 2022的MFC框架——theApp全局对象

我是荔园微风&#xff0c;作为一名在IT界整整25年的老兵&#xff0c;今天我们来重新审视一下Visual Studio 2022下开发工具的MFC框架知识。 MFC中的WinMain函数是如何与MFC程序中的各个类组织在一起的呢&#xff1f;MFC程序中的类是如何与WinMain函数关联起来的呢&#xff1f…

【LeetCode 】数组简介

集合列表和数组 本文中介绍的概念为适用于所有编程语言的抽象理论&#xff0c;具体实现会由编程语言的不同而稍有差别。 具体介绍数组之前&#xff0c;我们先来了解一下集合、列表和数组的概念之间的差别。 集合 集合一般被定义为&#xff1a;由一个或多个确定的元素所构成的…

Vue中使用element-plus中的el-dialog定义弹窗-内部样式修改-v-model实现-demo

效果图 实现代码 <template><el-dialog class"no-code-dialog" v-model"isShow" title"没有收到验证码&#xff1f;"><div class"nocode-body"><div class"tips">请尝试一下操作</div><d…

把Android手机变成电脑摄像头

一、使用 DroidCam 使用 DroidCam&#xff0c;你可以将手机作为电脑摄像头和麦克风。一则省钱&#xff0c;二则可以在紧急情况下使用&#xff0c;比如要在电脑端参加一个紧急会议&#xff0c;但电脑却没有摄像头和麦克风。 DroidCam 的安卓端分为免费的 DroidCam 版和收费的 …

《软件开发的201个原则》阅读笔记 120-161条

目录 使用有效的测试完成度标准 原则122 达成有效的测试覆盖 原则123 不要在单元测试之前集成 原则 124 测量你的软件 原则125 分析错误的原因 对错不对人 原则127 好的管理比好的技术更重要 使用恰当的方法 原则 129 不要相信你读到的一切 原则130 理解客户的优先级 原…

用QT实现MVP模式

近些天用qt 作项目,遇到参数界面.偷闲写个mvp模式示例. mvp模式重要的有两点 1 低耦合: 界面与后端数据类,不直接引用,可方便替换. 2 形成界面驱动-界面更新的闭环.:通过函数指针类技术,让数据自动回流. MVP (Model-View-Presenter) 视图&#xff08;View&#xff09;: 接…

git Update failed cannot lock ref

报错详情 解决方案 百度了很多方案&#xff0c;过滤出了有效方案 去该项目下的.git文件里找到报错文件&#xff0c;本例中即为&#xff1a;.git/refs/tags/pre-RELEASE-PRE-20230817-03 删除该文件&#xff0c;重新pull&#xff0c;pull成功问题解决

openapi中job提交

openapi中job提交 简介创建job查看job查看job 的描述查看job 的日志 简介 这里使用微软OpenPAI, 在nvidia的GPU设备上进行job测试。 创建job protocolVersion: 2 name: lenet_gpu_pytorch112_jiaxiaolei_20230825_1013 type: job jobRetryCount: 0 prerequisites:- type: doc…

大红喜庆版UI猜灯谜小程序源码/猜字谜微信小程序源码

今天给大家带来一款UI比较喜庆的猜灯谜小程序&#xff0c;大家看演示图的时候当然也是可以看得到那界面是多么的喜庆&#xff0c;而且新的一年也很快就来了,所以种种的界面可能都比较往喜庆方面去变吧。 这款小程序搭建是免服务器和域名的&#xff0c;只需要使用微信开发者工具…

Spring Cloud Nacos详解

目录 1、Spring Cloud Nacos详细介绍2、Spring Cloud Nacos具体案列 Spring Cloud Nacos 是一个由阿里巴巴集团开发的开源分布式系统服务发现、配置管理和服务管理的平台。Nacos 支持多种服务发现方式&#xff0c;包括 DNS 方式、HTTP 和 RPC 方式&#xff0c;同时提供了灵活的…

vue-element-admin最新版4.4实现多个url路由匹配到一个路径时,左侧菜单保持高亮状态

文章目录 环境&#xff1a;需求&#xff1a;原因分析&#xff1a;如何解决&#xff1a; 环境&#xff1a; vue-admin-template-4.4版本&#xff08;vue2&#xff09; 需求&#xff1a; 当我访问申请开户时&#xff0c;也希望支付菜单能保持高亮状态。 原因分析&#xff1a; …

【UniApp开发小程序】私聊功能uniapp界面实现 (买家、卖家 沟通商品信息)【后端基于若依管理系统开发】

文章目录 效果显示WebSocket连接使用全局变量WebSocket连接细节 最近和自己聊天的用户信息界面效果界面代码最近的聊天内容太长日期时间显示未读消息数量显示 私聊界面界面展示代码实现英文长串不换行问题聊天区域自动滑动到底部键盘呼出&#xff0c;聊天区域收缩&#xff0c;聊…

lnmp架构-nginx

1 Nginx服务的部署 前面高可用负载均衡做流量均摊只是用到 iso 前四层 下面针对某个请求 在七层模型 上做 apache也是可以的 但是 apache一般吞吐性能没有nigix 强 nigix可以做负载均衡器 也可以做常规的web服务器 作为网站的发布服务器 官网下载好之后 直接拖到软件中 源…

Qt 自定义提示框 右下角冒泡

网页右下角上经常会出现一些提示性的信息&#xff0c;B/S有的东西&#xff0c;C/S当然也可以有&#xff0c;就像QQ的消息提示一样&#xff01; 实现一个类似的东西并不困难&#xff0c;只要想明白原理实现起来就很简单了&#xff01; 实现原理&#xff1a; &#xff08;1&#…

yo!这里是Linux权限入门理解

目录 前言 权限概念 权限管理 分类 1.用户 2.文件&&目录 表示 设置 1.chmod指令 2.chown指令 3.chgrp指令 4.umask指令 粘滞位 后记 前言 对于Linux基本指令&#xff0c;基本上就是操作文件或者目录&#xff0c;但是&#xff0c;是谁可以操作文件或目录&…

Python requests实现图片上传接口自动化测试

最近帮别人写个小需求&#xff0c;需要本地自动化截图&#xff0c;然后图片自动化上传到又拍云&#xff0c;实现自动截图非常简单&#xff0c;在这里就不详细介绍了&#xff0c;主要和大家写下&#xff0c;如何通过Pythonrequests实现上传本地图片到又拍云服务器。 话不多说&a…

政府网站定期巡检:构建高效、安全与透明的数字政务

在数字时代&#xff0c;政府网站已不仅仅是一个信息发布窗口&#xff0c;更是政府与公众互动的桥梁、政务服务的主要渠道以及数字化治理的重要平台。因此&#xff0c;确保政府网站的高效运行、信息安全与透明公开就显得尤为重要。在此背景下&#xff0c;定期的网站巡检与巡查成…