『C++成长记』string使用指南

news2025/1/31 8:06:41

🔥博客主页:小王又困了

📚系列专栏:C++

🌟人之为学,不日近则日退

❤️感谢大家点赞👍收藏⭐评论✍️

目录

一、string类介绍

二、string类的常用接口说明

📒2.1string类对象的常见构造

📒2.2string类对象的容量操作

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

📒2.4string类对象的修改操作

​编辑

📒2.5string类对象的查找接口

📒2.6string类非成员函数


🗒️前言:

C语言中,字符串是以'\0'结尾的一些字符的集合,为了操作方便,C标准库中提供了一些str系列的库函数, 但是这些库函数与字符串是分离开的,不太符合OOP的思想,而且底层空间需要用户自己管理,稍不留神可能还会越界访问。C++中引入了string类,给我们提供了便利。

一、string类介绍

  • string是表示字符串的字符串类
  • string类的接口与常规容器的接口基本相同,再添加了一些专门用来操作string的常规操作。
  • string在底层实际是:basic_string模板类的别名,typedef basic_string string;
  • 不能操作多字节或者变长字符的序列。

小Tips:在使用string类时,必须包含#include头文件以及using namespace std;

二、string类的常用接口说明

📒2.1string类对象的常见构造

🎀string()

    构造空的string类对象,即空字符串。

int main()
{
    string s1;
    cout << s1 << endl;
    return 0;
}

🎀string(const char* s)

    用一个C的字符串(或字符数组)来构造一个string类的对象。

int main()
{
    string s1("hellow world");
    cout << s1 << endl;
    return 0;
}

🎀string(size_t n, char c)

    用 个字符来构建string类对象。

int main()
{
    string s1(5, 'x');
    cout << s1 << endl;
    return 0;
}

🎀string(const string&s)

     string类的拷贝构造,用于构建一个和存在的s1一模一样的对象。

int main()
{
    string s1(5, 'x');
    string s2(s1);
    cout << s2 << endl;
    return 0;
}

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

    复制strpos位置开始len个字符的部分,使用一个已存在的string类对象的一部分来创建一个新的string类对象。

小Tips:nopsstring类里面的一个静态成员变量,它是size_t类型,初始化为-1,即表示整型的最大值。

int main()
{
    string s1("hellow world");
    string s2(s1, 7, 5);
    cout << s2 << endl;
    return 0;
}

🎀string(const char* s,size_t n)

    用 s 所指向字符串(或字符数组)的前 n 个字符来初始化创建一个string类对象。

int main()
{
    char s[] = "hellow ,world";
    string s1(s, 6);
    cout << s1 << endl;
    return 0;
}

📒2.2string类对象的容量操作

🎀size()

    返回字符串有效字符长度

int main()
{
    string s1("hellow world");
    string s2("hellow C++");
    cout << s1.size() << endl;
    cout << s2.size() << endl;
    return 0;
}

🎀length()

    返回字符串有效字符长度

int main()
{
    string s1("hellow world");
    string s2("hellow C++");
    cout << s1.length() << endl;
    cout << s2.length() << endl;
    return 0;
}

小Tips:size()length()方法底层实现原理完全相同,引入size()的原因是为了与其他容器的接口保持一致,一般情况下基本都是用size()

🎀capacity()

    返回string对象空间的总大小。

int main()
{
    string s1("hellow world");
    string s2("hellow world xxxxxxx");
    cout << s1.capacity() << endl;
    cout << s2.capacity() << endl;
    return 0;
}

小Tips:capacity()会返回对象空间的总大小,这个空间是存储有效字符的空间,实际上会多开辟一个空间,存储‘\0’。同一个对象在不同的环境开辟的空间不同,在不同环境下扩容机制也不同。

📖VS下的扩容机制

int main()
{
    string s1("hellow world");
    size_t old = s1.capacity();
    cout << old << endl;
    for (size_t i = 0; i < 100; i++)
    {
        s1.push_back('x');
        if (old != s1.capacity())
        {
            cout << s1.capacity() << endl;
            old = s1.capacity();
        }
    }
    return 0;
}

VS下默认开辟15个空间用来存储数据(本质上是开16个空间,因为还要存‘/0’),第一次扩容是2倍,后面都是以1.5倍的大小去扩容。 

📖Linux下的扩容机制

在Linux环境下最初对象需要多少空间就开多少,后面一次按照2倍的大小进行扩容。

🎀reserve(size_t n = 0)

    为字符串预留空间。直接一次开辟n个空间,可以用来存储n个有效字符,可以避免每次都要去扩容。

int main()
{
    string s1("hellow world");
    s1.reserve(100);
    size_t old = s1.capacity();
    cout << s1.capacity() << endl;
    for (size_t i = 0; i < 100; i++)
    {
        s1.push_back('x');
        if (old != s1.capacity())
        {
            cout << s1.capacity() << endl;
            old = s1.capacity();
        }
    }
    return 0;
}

小Tips:当我们确定需要多少空间,就可以使用reserve()提前开辟好空间即可,可以减少扩容次数, 大部分的扩容都是异地扩容,扩容次数过多会影响效率。

🎀resize(size_t n = 0, char c)

    resize(size_t n)resize(size_t n, char c)都是将字符串中有效字符个数改变到n个,当n小于当前字符串的长度size(),会保留前n个字符,将第n个字符后面的所以字符删除;当n大于当前字符串的长度size(),先会进行扩容,不同的是当字符个数增多时:resize(size_t n)用‘\0’来填充多出的元素空间,resize(size_t n, char c)用字符c来填充多出的元素空间。

int main()
{
    string s1("hellow world");
    cout << s1 << endl;
    cout << s1.size() << endl;
    cout << s1.capacity() << endl;

    s1.resize(100,'x');
    cout << s1 << endl;
    cout << s1.size() << endl;
    cout << s1.capacity() << endl;
    return 0;
}	

小Tips:resize在改变元素个数时,如果是将元素个数增多,可能会改变底层容量的大小,如果是将元素个数减少,底层空间总大小不变。 reserve只会影响容量,不改变有效元素个数,当reserve的参数小于string的底层空间总大小时,reserver不会改变容量大小,即不会缩容。

  • reserve只影响容量,不影响数据
  • resize既影响容量,也影响数据

🎀clean()

    清理有效字符,不改变容量大小

int main()
{
	
    string s1("Hello C++!");
    cout << "清空之前的size:" << s1.size() << endl;
    cout << "清空之前的capacity:" << s1.capacity() << endl;
    s1.clear();
    cout << "清空之后的size:" << s1.size() << endl;
    cout << "清空之后的capacity:" << s1.capacity() << endl;
    return 0;
}	

小Tips:clear()只是将string中有效字符清空,不改变底层空间大小。

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

🎀operator()

    返回pos位置的字符

int main()
{
    string s1 = "hellow world";
    for (size_t i = 0; i < s1.size(); i++)
    {
        cout << s1[i];
    }
    cout << endl;
    return 0;
}

小Tips:如果字符串是const限定的,则该函数返回const char&。否则,它将返回char&at借口与operator[]借口功能类型,operator[]接口发生越界访问,程序会直接报错;at接口发生越界访问的时候会抛出异常。

🎀begin、end/rebegin、rend

    begin获取对象第一个字符的迭代器,end获取对象最后一个字符下一个位置的迭代器

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

    string::reverse_iterator it = s1.rbegin();//反向迭代器
	while (it != s1.rend())
	{
		cout << *it;
		it++;
	}
	cout << endl;
    return 0;
}

如上代码我们看到迭代器和指针的使用方法类似,都可以解引用和++,对象也可以修改。但迭代器的这些操作都是通过运算符重载来实现的。

小Tips:迭代器区间永远都是左闭右开,迭代器类型作为类的内置类型可以通过类名::itertator直接访问,例如:string::itertator就表示string类里面的迭代器类型。普通迭代器可读可写,const迭代器限制的是其指向的内容,只能读不能写,而const迭代器本身可以修改。

🎀范围for

int main()
{
    string s1("Hello world");
    for (auto it : s1)
    {
        cout << it;
    }
    cout << endl;
    return 0;
}

 小Tips:范围for就是基于迭代器实现的,在底层范围for会转化成正向迭代器。

📒2.4string类对象的修改操作

🎀push back

    在字符串后尾插字符c

int main()
{
    string s1("Hello world");
    cout << "追加前:" << s1 << endl;

    s1.push_back('x');
    cout << "追加后:" << s1 << endl;
    return 0;
}

🎀append

    在字符串后追加一个字符串,它有六种重载形式,如下图所示:

int main()
{
    string s1("Hello world");
    cout << "追加前:" << s1 << endl;
    s1.append("xxxxxxx");
    cout << "追加后:" << s1 << endl;

    string s2 = "abcdefg";
    cout << "追加前:" << s1 << endl;
    s1.append(s2, 2, 4);
    cout << "追加后:" << s1 << endl;
    return 0;
}

🎀operator+=

    通过重载运算符+=实现追加,该运算符重载有三种重载实现形式,如下图所示:

int main()
{
    string s1("Hello world");
    cout << "追加前:" << s1 << endl;
    s1 += "xxxxxx";
    cout << "追加后:" << s1 << endl;

    string s3 = "abcdefg";
    string s2("Hello world");
    cout << "追加前:" << s2 << endl;
    s2 += s3;
    cout << "追加后:" << s2 << endl;
    return 0;
}

小Tips:string尾部追加字符的三种方式中,+=操作用的比较多,+=操作不仅可以连接单个字符,还可以连接字符串。

📒2.5string类对象的查找接口

🎀find

    从字符串的pos位置开始往后查找字符或字符串,返回其在当前字符串中的位置。

int main()
{
    string s1 = "hellow world";
    string s2 = "world";
    size_t pos = s1.find(s2);
    cout << pos << endl;
    return 0;
}

 🎀substr

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

int main()
{
    string s1 = "test.txt.cpp";
    size_t pos = s1.rfind('.');
    if (pos != string::npos)
    {
        string s2 = s1.substr(pos);
        cout << s2 << endl;
    }
    return 0;
}

📒2.6string类非成员函数

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

小Tipsoperator>>getline的区别在于,前者遇到空格‘  ’和换行‘\n’会截止,而后者默认只有遇到换行‘\n’才截止,因此当我们需要从键盘读取一个含有空格的字符串是,只能用getline


🎁结语: 

     本次的内容到这里就结束啦。希望大家阅读完可以有所收获,同时也感谢各位读者三连支持。文章有问题可以在评论区留言,博主一定认真认真修改,以后写出更好的文章。你们的支持就是博主最大的动力。

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

线下店铺的商品如何查价

品牌渠道主要分为线上和线下&#xff0c;线上的价格查价方式可以通过系统完成&#xff0c;系统筛选不同平台的数据&#xff0c;然后做价格的比较&#xff0c;输出是否破价的监测报表&#xff0c;这个过程多依赖系统去操作&#xff0c;但线下店铺的产品价格查价方式则没有固定的…

Windows下Cloudreve WebDAV挂载失败解决方法

如果你正在使用Windows的资源管理器挂载WebDAV&#xff0c;并且发现挂载失败&#xff0c;无论你如何去配置他 首先&#xff0c;Cloudreve 中可以为每个用户组单独设置“是否允许使用WebDAV访问”的开关&#xff0c;但是这玩意儿默认是开启的&#xff0c;所以如果不小心手残关掉…

机器学习系列——(六)数据降维

引言 在机器学习领域&#xff0c;数据降维是一种常用的技术&#xff0c;旨在减少数据集的维度&#xff0c;同时保留尽可能多的有用信息。数据降维可以帮助我们解决高维数据带来的问题&#xff0c;提高模型的效率和准确性。本文将详细介绍机器学习中的数据降维方法和技术&#…

解锁文档处理的全新维度:ONLYOFFICE 文档开发者版

前言 相信大家对于 ONLYOFFICE 这款办公软件可能已经有所耳闻&#xff0c;最近因工作需要&#xff0c;我在众多办公协作工具中选择了 ONLYOFFICE&#xff0c;原因主要是它开源经济实惠&#xff0c;可以部署在自己的服务器上并且能够轻松集成到我们的平台中。在数字化信息时代&…

10、数据结构与算法——堆

一、什么是堆 堆是一种特殊的树形数据结构&#xff0c;通常实现为完全二叉树或满二叉树。堆又分为两种类型最大堆&#xff08;Max Heap&#xff09; 和 最小堆&#xff08;Min Heap&#xff09; 1.1、什么是二叉树 二叉树是一种数据结构&#xff0c;它是由n&#xff08;n≥0&a…

Python 计算两个时间之间的耗时

博主按照自己的需求写的一个可以计算程序耗时的函数&#xff0c;各位有需要的自行改写哈 实现的大致功能 功能1 记录开始时间&#xff0c;返回一个时间字典。数据包括&#xff1a;开始时间的日期格式、文本格式 功能2 记录结束时间和耗时&#xff0c;返回一个时间字典。数据包…

Web3智能合约:重新定义商业合作的未来

随着区块链技术的飞速发展&#xff0c;Web3时代正逐渐到来&#xff0c;而其中的智能合约成为推动商业合作变革的关键力量。本文将深入探讨Web3智能合约的概念、特点以及对商业合作未来的巨大影响。 什么是Web3智能合约&#xff1f; 智能合约是一种以代码形式编写、自动执行合同…

Oracle 面试题 | 05.精选Oracle高频面试题

&#x1f90d; 前端开发工程师、技术日更博主、已过CET6 &#x1f368; 阿珊和她的猫_CSDN博客专家、23年度博客之星前端领域TOP1 &#x1f560; 牛客高级专题作者、打造专栏《前端面试必备》 、《2024面试高频手撕题》 &#x1f35a; 蓝桥云课签约作者、上架课程《Vue.js 和 E…

Flask 入门4:Flask 模板

1. 前言 Flask 拥有丰富的扩展方法&#xff0c;且都有统一的特点&#xff1a;简单和即学即用。当我们要实现某个功能之前&#xff0c;可以提前去搜一搜这个功能包是否已经存在&#xff0c;这样也能帮助我剩下很多时间。那么要去哪里找到这些扩展包呢&#xff0c;这里推荐两个方…

Android:RecyclerView跨行跨列的LayoutManager:Spannedgridlayoutmanager

前言&#xff1a; RecyclerView可以使用GridLayoutManager实现跨行&#xff0c;但是不能跨列&#xff1b;瀑布流布局可以跨列但是又不能跨行。原生自带的各个LayoutManager中并没有可以又跨行又能跨列的。网上搜寻了一番&#xff0c;找到了一个亲测可行好用的三方库&#xff1…

微信小程序可以一个人开发么?

随着移动互联网的飞速发展&#xff0c;微信小程序已经成为了人们日常生活中不可或缺的一部分。它以其便捷性和高效性赢得了广大用户的青睐。对于许多有志于进入小程序领域的开发者来说&#xff0c;一个常见的问题是&#xff1a;微信小程序可以一个人开发吗&#xff1f;答案是肯…

2024 Flutter 重大更新,Dart 宏(Macros)编程开始支持,JSON 序列化有救

说起宏编程可能大家并不陌生&#xff0c;但是这对于 Flutter 和 Dart 开发者来说它一直是一个「遗憾」&#xff0c;这个「遗憾」体现在编辑过程的代码修改支持上&#xff0c;其中最典型的莫过于 Dart 的 JSON 序列化。 举个例子&#xff0c;目前 Dart 语言的 JSON 序列化高度依…

【笔记】React-Native Navigation页面导航

/** * 须知&#xff1a;a bare React Native Project(not an Expo managed project) * * React官方文档&#xff1a;https://reactnative.dev/docs/navigation * * 当前文档基于6.x * React Navigation文档&#xff1a;https://reactnavigation.org/docs/getting-started * …

C++ Webserver从零开始:基础知识(七)——多进程编程

前言 在学习操作系统时&#xff0c;我们知道现代计算机往往都是多进程多线程的&#xff0c;多进程和多线程技术能大大提高了CPU的利用率&#xff0c;因此在web服务器的设计中&#xff0c;不可避免地要涉及到多进程多线程技术。 这一章将简要讲解web服务器中的多进程编程&#x…

AI与数字孪生

源自&#xff1a;译文 “人工智能技术与咨询”发布 AI和数字孪生 预测分析&#xff1a;网络安全水晶球 面对不断变化的威胁&#xff0c;提供自适应安全防护 自然语言处理&#xff1a;解码威胁语言 先进技术&#xff1a;人工智能作为数字孪生安全的基础 道德考量 …

关于v8垃圾回收机制以及与其相关联的知识点--还没整理版本

对于值类型b来说&#xff0c;就直接释放了其占用的内存&#xff0c;对于引用类型obj来说&#xff0c;销毁的只是变量obj对堆内存地址 1001 的引用&#xff0c;obj的值 { c: 3 } 依然存在于堆内存中。那么堆内存中的变量如何进行回收呢&#xff1f; V8的垃圾回收策略主要是基于…

git将项目的某次签入遴选(Cherry-Pick)另一个项目

需求&#xff1a;将项目Product&#xff0c;分支feature/platform&#xff0c;签入959294ce6b75ee48c5cb22c46d7398654628a896&#xff0c;遴选到项目BRP&#xff0c;分支dev 第一步&#xff1a;使用原签入生成patch文件&#xff08;git format-patch -1 <commit_hash>&a…

西部数据DX4000数据恢复案例

西部数据DX4000是一款NAS&#xff08;网络附加存储&#xff09;产品&#xff0c;它集成了存储、备份、共享和远程访问企业数据等多种功能。然而&#xff0c;和任何电子设备存在一样通病&#xff0c;DX4000也可能会出现故障&#xff0c;导致数据无法访问。下面是一个客户的西部数…

Python OpenCV实现图片像素区域缩放

Python OpenCV实现图片像素区域缩放 前言项目安装OpenCV和Pillow思路代码编写 前言 遇到一个要将大量图片缩放成统一规格的难题&#xff0c;并且这些图片周围还有很多空白像素&#xff0c;所以用Python实现一下。 项目 安装OpenCV和Pillow pip install opencv-python pip …

基于YOLOv8的水下生物检测,多种优化方法---DCNv4结合SPPF,效果秒杀DCNv3,涨点两个点(四)

&#x1f4a1;&#x1f4a1;&#x1f4a1;本文主要内容:详细介绍了水下生物检测整个过程&#xff0c;从数据集到训练模型到结果可视化分析&#xff0c;以及如何优化提升检测性能。 &#x1f4a1;&#x1f4a1;&#x1f4a1;加入DCNv4_SPPF mAP0.5由原始的0.522提升至0.543 1.水…