【C++程序员的自我修炼】string 库中常见用法(二)

制芰荷以为衣兮

集芙蓉以为裳

不吾知其亦已兮

苟余情其信芳


目录

 字符串的头部插入insert

<1>头部插入一个字符串:

 <2>头部插入一个字符:

<3>迭代器的插入 

 总结:

 字符串的头部删除 erase

 <1>头部插入删除一个字符:

 

<2>头部插入删除一个字符串: 

​编辑 

<3>指定位置的删除:

字符串的替换 replace

字符串中关于容量的库函数

计算一个字符串最大能开多大空间:max_size

查看一个字符串的当前容量:capacity

字符串容量的调整 reserve

总结

字符串大小的调整:resize

字符串的缩容:shrink_to_fit

 字符串的头部插入insert

还是老样子,我们先来参考一下文档在进行分析:

<1>头部插入一个字符串:

string& insert (size_t pos, const string& str);

根据文档我们可以得知是在目标字符串的 pos 位置(注意下标从 0 开始)插入字符串 str

不过我们经常用来头插

#include<iostream>
#include<string>
using namespace std;void TestString()
{string s1("hello");s1.insert(0, "world ");cout << s1 << endl;
}int main()
{TestString();return 0;
}

指定位置也是可以哒(后面都以实现头插为主)

void TestString()
{string s1("hello");s1.insert(2, "world ");cout << s1 << endl;
}

 <2>头部插入一个字符:

string& insert (size_t pos, size_t n, char c);

 这里我(疯狂吐槽):我们想在 pos 位置插入字符 c 还需要填写字符的数量 n

void TestString()
{string s1("hello");s1.insert(0, 1, 'x');cout << s1 << endl;
}

肯定有老铁跟我一样吐槽,这样设计实在是有些冗余,那能不能不写插入字符的个数呢?

--- 这就要借助我们的迭代器啦 ~

<3>迭代器的插入 

template <class InputIterator>void insert (iterator p, InputIterator first, InputIterator last);

 如果是迭代器的话,刚刚插入一个字符还需填写个数的形式就可以解决了

void TestString()
{string s1("hello");s1.insert(s1.begin(), 'x');cout << s1 << endl;
}

 我们还可以这么玩 ~

void TestString()
{string s1("hello");string s2("world ");s1.insert(s1.begin(), s2.begin(), s2.end());cout << s1 << endl;
}

 总结:

我们学过顺序表的都知道,数组在头部插入要挪动数据,效率是  O(N)

所以我们在使用 insert 还需要考虑时间复杂度的问题,因为这个效率属实不高 ~ 慎用 !!!


 字符串的头部删除 erase

string& erase (size_t pos = 0, size_t len = npos);

结合文档我们不难理解:

删除从 pos 位置开始的 len 个字符,如果 len 超过字符串剩下的值或者不写则全部删除

( pos 现在是缺省值,默认 pos 值是零,也就是从头开始删除,当然我们可以传其他参数)

 <1>头部插入删除一个字符:

#include<iostream>
#include<string>
using namespace std;void TestString()
{string s1("hello world");s1.erase(0,1);cout << s1 << endl;
}int main()
{TestString();return 0;
}

 

<2>头部插入删除一个字符串: 

void TestString()
{string s1("hello world");s1.erase(0,5);cout << s1 << endl;
}

 

<3>指定位置的删除:

void TestString()
{string s1("hello world");s1.erase(6,2);cout << s1 << endl;
}

注意:铁铁们如果我们 erase 什么也不传会发生什么呢?  -- 全部删完

因为 pos 缺省值默认从 0 开始,而默认的 npos 是一个 42 亿的数,所以全都删完噜 ~

总结:我们这里的头删与顺序表的也是一样,后一个覆盖前一个需要挪动数据,所以吃时间,我们用的是要还有斟酌一下(不要让程序跑的太慢哦)


字符串的替换 replace

string& replace (size_t pos,  size_t len,  const string& str);

结合文档我们不难理解:

把从 pos 开始的 len 个字符替换成字符串 str

我们可以来测试一下 ~ 我们将空格位置替换成 %20 

void TestString()
{string s1("hello world");s1.replace(5, 1, "%20");cout << s1 << endl;
}

注意:以上做法有没有什么弊端呢?-- 吃时间

因为我的一个字符的位置容不下 %20 ,所以位置要进行挪动,空间能容纳之后才能进行替换

我们替换最好控制在相等的长度内,不要让数据发生挪动


字符串中关于容量的库函数

计算一个字符串最大能开多大空间:max_size

max_size 计算的是字符串最大能开多少空间,但是真的能开那么大吗

我们可以先来看看这个空间有多大:

void TestString()
{string s1;cout << s1.max_size() << endl;
}

x64 环境下 :

x86 环境下:

我们虽然它的潜力空间很大(大概几个G),但是计算机真的愿意给一个字符串开那么大空间吗

-- 还真不愿意,就如一个富豪总资产才 8 个亿你叫他给你 4 个亿,你觉得他答应不答应

这个接口只需了解一下即可,平时基本没怎么用到


查看一个字符串的当前容量:capacity

根据文档我们可以得知:capacity 计算的是当前容量的大小,并以字节的形式返回

有了这个知识点,我们可以先来看一下 VS 下的扩容:

void TestString()
{string s1;size_t sz = s1.capacity();cout << "capacity changed:" << sz << endl;cout << "making s grow:" << endl;for (int i = 0; i < 100; i++){s1.push_back('c');if (sz != s1.capacity()){sz = s1.capacity();cout << "capacity changed:" << sz << endl;}}
}

我先来了解一下 VS 存储的逻辑:

首先我们在创建 对象s1 时用的是类中的空间  _Buf

然后我们了解到我们实际开的空间是 16 ,但是我们打印的 capacity 是15

说明 VS 编译器给 \0 预留了一个位置,这块空间不算在 capacity 里面

当 _Buf 不够用时就在堆上扩容并将其拷贝到新开辟的空间,事后不再使用这块 _Buf 空间

所以 s1 真正的扩容是在 making s grow 中开始的(之前的那块空间是类中存在的空间 _Buf)

属于 1.5 倍扩容

我们再看一下 Linux 下的扩容:

我们可以看到 Linux 环境下没有 _Buf 数组,而且显示了 \0 的空间,很直观 ~

属于 2 倍扩容

以后我们可以借助 capcacity 来了解编译器扩容的规则


字符串容量的调整 reserve

根据文档我们可以得知:reserve 为容器预留足够的空间,避免不必要的重复分配,分配空间大于等于函数的参数,影响 capacity

我们先来看一下 VS 下的  reserve:

void TestString()
{string s1;s1.reserve(100);cout << s1.capacity() << endl;
}

我们发现我们预留了 100 的空间,但是编译器开了 111

我们再来看看 Linux 环境下的 reserve

如果是 Linux 环境下则是设置多少就开多少

reserve 还有一点作用就是缩容,但是 VS 环境下是不缩容的:

可能有些老铁会疑惑 -- 为什么 reserve 小于15就会缩容到15呢?

 其实这不是缩容而是用了类中的 _Buf 空间

如果  reserve 小于 _Buf 则会释放在堆上开辟的空间,从而用 _Buf 的空间,这就说明为什么容量是 15 而不是 10 的问题

Linux 环境下是缩容的:

而且是按照我们指定的容量缩容

运用场景:

举个栗子 -- 栈的扩容,如果我们刚开始没有预留空间,数据量大的话就要频繁扩容,而频繁扩容对我们的程序是不好的,系统会产生大量的内存碎片 -- 导致空间的浪费

注意:

有些老铁以为开了空间就能对空间所有位置进行访问:

void TestString()
{string s1;s1.reserve(100);s1[55] = 'x';cout << s1.capacity() << endl;
}

这里就发生越界访问了 -- 原因是:虽然你的空间开好了,但是你的数据个数 size 依然是  0

我们数组访问的不是容量,而是你计入的数据量

总结

在使用 reserve、capacity 需要先了解编译器的环境(每个编译器的环境是不一样的),不然容易出错


字符串大小的调整:resize

根据文档我们可以得知:resize 可以改变容器的大小,使得其包含n个元素

我们这里讲讲常见 resize 三种用法 :

<1>直接改变 size 的大小,注意这里会间接改变 capacity 容量

 此时我们就可以访问我们 size 计入的空间(不会发生越界访问):

	s1[15] = 'x';

而其他没有被初始化的空间则全部用 \0 代替 ~

<2>改变 size 的大小的同时初始化

void TestString()
{string s1;s1.resize(10, 'x');cout << s1 << endl;
}

<3>假设 n 是 resize 的容器元素大小 -- 参数

(1)如果 n 小于当前的原字符串大小,那么则保留原字符串的前 n 个元素,去除超过的部分

void TestString()
{string s1("hello world");s1.resize(5);cout << s1 << endl;
}

(2)如果 n 大于当前的原字符串大小,则通过在原字符串结尾插入适合数量的元素使得整个字符串的大小达到 n

void TestString()
{string s1("hello world");s1.resize(15, 'x');cout << s1 << endl;
}

 


字符串的缩容:shrink_to_fit

C++11提出了 shrink_to_fit 函数目的是缩容:让容量 capacity 大小跟 size 大小保持一致

void TestString()
{string s1;s1.reserve(100);cout << "capacity:" << s1.capacity() << endl;s1.resize(20);cout << "capacity:" << s1.capacity() << endl;s1.shrink_to_fit();cout << "capacity:" << s1.capacity() << endl;
}

说到这里,我们来讲一下缩容的原理:
错误示范(在没讲之前是不是有很多老铁是这种缩容观念)

将一整条空间中的一部分释放再保留一部分

其实这是错误的 ~  因为我们释放空间是必须从数组的首地址开始释放:

正确的缩容方式为:找一块更小的空间再将数据拷贝进去,并释放原来的空间

这样就导致他的效率很低 ~ 老铁使用的时候要注意


 

此身天地一虚舟

                何处江山不自由

先介绍到这里啦~

有不对的地方请指出💞

 

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

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

相关文章

nodejs实战——搭建websocket服务器

本博客主要介绍websocket服务器库安装&#xff0c;并举了一个简单服务器例子。 服务器端使用websocket需要安装nodejs websocket。 cd 工程目录 # 此刻我们需要执行命令&#xff1a; sudo npm init上述命令创建package.json文件&#xff0c;系统会提示相关配置。 我们也可以使…

数据结构十:哈希表

本次将从概念上理解什么是哈希表&#xff0c;理论知识较多&#xff0c;满满干货&#xff0c;这也是面试笔试的一个重点区域。 目录 一、什么是哈希表 1.0 为什么会有哈希表&#xff1f; 1.1 哈希表的基本概念 1.2 基本思想 1.3 举例理解 1.4 存在的问题 1.5 总结 二、…

基于JSP的人才公寓管理系统

目录 背景 技术简介 系统简介 界面预览 背景 随着互联网的广泛推广和应用&#xff0c;人才公寓管理系统在网络技术的推动下迅速进步。该系统的设计初衷是满足住户的实际需求&#xff0c;通过深入了解住户的期望&#xff0c;开发出高度定制化的人才公寓管理系统。利用互联网…

Django关于ORM的增删改查

Django中使用orm进行数据库的管理&#xff0c;主要包括以下步骤 1、创建model&#xff0c; 2、进行迁移 3、在视图函数中使用 以下的内容可以先从查询开始看&#xff0c;这样更容易理解后面删除部分代码 主要包括几下几种&#xff1a; 1、增 1&#xff09;实例例化model,代…

SpringTask定时任务

SpringBoot项目定时任务 首先在启动类引入注解EnableScheduling然后在方法中加注解Scheduled(cron“”)cron表达式 生成cron https://www.pppet.net/

流畅的Python阅读笔记

五一快乐的时光总是飞快了&#xff0c;不知多久没有拿起键盘写文章了&#xff0c;最近公司有Python的需求&#xff0c;想着复习下Python吧&#xff0c;然后就买了本Python的书籍 书名&#xff1a; 《流畅的Python》 下面是整理的一个阅读笔记&#xff0c;大家自行查阅&#xf…

【Qt QML】Frame组件

Frame&#xff08;框架&#xff09;包含在&#xff1a; import QtQuick.Controls继承自Pane控件。用于在可视框架内布局一组逻辑控件。简单来说就是用来包裹和突出显示其他可视元素。Frame不提供自己的布局&#xff0c;但需要自己对元素位置进行设置和定位&#xff0c;例如通过…

Numerical Analysis(byRichard.L..Burden)【pdf高清英文原版】

专栏导读 作者简介&#xff1a;工学博士&#xff0c;高级工程师&#xff0c;专注于工业软件算法研究本文已收录于专栏&#xff1a;《有限元编程从入门到精通》本专栏旨在提供 1.以案例的形式讲解各类有限元问题的程序实现&#xff0c;并提供所有案例完整源码&#xff1b;2.单元…

基于TL431基准电压源的可调恒压恒流源的Multisim电路仿真设计

1、线性电源的工作原理 在我们日常应用里&#xff0c;直流电是从市电或电网中的交流电获取的。例如15V直流电压源、24V直流电压源等等。交流电变为直流电的过程大概分为一下几步&#xff1a; 首先&#xff0c;交流电通过变压器降低其电压幅值。接着&#xff0c;经过整流电路进…

基于机器学习的网络流量识别分类

1.cicflowmeter的目录框架&#xff1a; 各部分具体代码 FlowMgr类&#xff1a; package cic.cs.unb.ca.flow;import cic.cs.unb.ca.Sys; import org.slf4j.Logger; import org.slf4j.LoggerFactory;import java.time.LocalDate;public class FlowMgr {protected static final…

项目管理【人】概述

系列文章目录 【引论一】项目管理的意义 【引论二】项目管理的逻辑 【环境】概述 【环境】原则 【环境】任务 【环境】绩效 【人】概述 一、项目涉及到的人 1.1 项目发起人、项目指导委员会和变更控制委员会 项目发起人&#xff08;Sponsor&#xff09; 项目指导委员会&…

JVM笔记4-虚拟机类加载机制

1、概述 Java虚拟机把描述类的数据从Class文件加载到内存中&#xff0c;并对数据进行检验、转换解析和初始化&#xff0c;最终形成可以被虚拟机直接使用的Java类型。这个过程称为虚拟机的类加载机制。 2、类加载的时机 一个类型从被加载到内存中开始&#xff0c;到卸载出内存…

使用socket+Python实现ping

import os import socket import struct import select import time# 计算校验和&#xff0c;用于确保数据的完整性 def checksum(source_string):sum 0count 0max_count len(source_string)# 处理成对的字节while count < max_count - 1:val source_string[count 1] *…

5.5代码

目录 1.内存空间 1.内存空间 真的要吐了&#xff0c;人都麻了题还没看完&#xff0c;看样子就是要想办法提取出来想要的东西2022第十三届蓝桥杯决赛C/C大学A组-C题内存空间_蓝桥杯a组c语言题目-CSDN博客 这个是一个非常清晰的代码&#xff0c;好几个帖子都管这个题叫大模拟题…

开源版本管理系统的搭建二:SVN部署及使用

作者&#xff1a;私语茶馆 1. Visual SVN Server部署 SVN Server部署包括&#xff1a; 创建版本仓库创建用户 这些部署是通过VisualSVN Server Manager实现的&#xff0c;如下图&#xff1a; VisualSVN Server Manager&#xff08;安装后自带&#xff09; 1.1.SVN 初始化配…

ROS2学习——Docker环境下安装于使用(1)

目录 一、简要 二、ROS2和ROS1区别 三、环境搭建与安装 &#xff08;2&#xff09;拉取ubuntu22.04镜像 &#xff08;2&#xff09;安装ROS2 1. 基本设置 2.设置源 3.安装ROS2功能包 4.测试 四、相关指令学习 1.小海龟测试 2.ros2 node等指令 3.rqt 一、简要 随着R…

Git常用(持续更新)

常用场景&#xff1a; 初始化&#xff1a; git config --global user.name "codelabs" git config --global user.email mycodelabs.com git init git remote add origin https://github.com/username/repository.git git pull origin master 提交&#xff1a; gi…

踏春正当时!VELO Prevail Ride带你探索多元骑行潮流体验~

嘿&#xff0c;朋友&#xff01;踏春正当时嘞&#xff01;在这个追求个性化与多元化的新时代&#xff0c;骑行爱好者们也开始寻找能适应各种骑行场景的理想坐垫。从悠闲自在的日常通勤&#xff0c;到热血沸腾的公路竞速&#xff0c;再到勇攀高峰的山地探险&#xff0c;维乐VELO…

HNU-人工智能-实验3-贝叶斯分类器

人工智能-实验3 计科210x 甘晴void 【感悟】本实验值得自己完成一遍 文章目录 人工智能-实验3一、实验目的二、实验平台三、实验内容3.0 基础知识3.1 条件概率&#xff08;选择题&#xff09;3.2 贝叶斯公式&#xff08;选择题&#xff09;3.3 朴素贝叶斯分类算法流程3.3.1 算…

多线程系列(三) -synchronized 关键字使用详解

一、简介 在之前的线程系列文章中&#xff0c;我们介绍了线程创建的几种方式以及常用的方法介绍。 今天我们接着聊聊多线程线程安全的问题&#xff0c;以及解决办法。 实际上&#xff0c;在多线程环境中&#xff0c;难免会出现多个线程对一个对象的实例变量进行同时访问和操…