【C++】string类模拟实现过程中值得注意的点

👀樊梓慕:个人主页

 🎥个人专栏:《C语言》《数据结构》《蓝桥杯试题》《LeetCode刷题笔记》《实训项目》《C++》《Linux》

🌝每一个不曾起舞的日子,都是对生命的辜负


目录

前言

1.有关const的使用

(1)const修饰形参

(2)const修饰返回值

(3)const修饰成员函数

2.有关声明与定义分离可能出现的问题

(1)合并命名空间

(2) 指定类域

(3)当声明与定义分离时,缺省参数只需要在声明中体现

 (4)函数重定义问题

3.有关模拟实现的传统写法与现代写法

4.有关string类流插入和流提取操作符的实现思维定势陷阱


前言

本篇文章旨在记录博主在模拟实现string类中遇到的一些问题,希望与大家共勉,在模拟实现string类的过程中运用了之前所学的很多知识,包括一些问题的出现都是对之前所学知识的考验,为了更加深入的了解和掌握string类,大家不妨跟随博主的步伐一起,看看遇到这些问题时,你是否可以轻松应对,利用之前所学解决实际问题。


欢迎大家📂收藏📂以便未来做题时可以快速找到思路,巧妙的方法可以事半功倍。

=========================================================================

GITEE相关代码:🌟fanfei_c的仓库🌟

=========================================================================


1.有关const的使用

(1)const修饰形参

首先我们来看string类中这些函数的模拟实现:

//relational operators
bool operator>(const string& s);
bool operator==(const string& s);
bool operator<=(const string& s);
bool operator<(const string& s);
bool operator>=(const string& s);      
bool operator!=(const string& s);

 为什么用const修饰形参?

  • const修饰形参的意义主要体现在输入型参数上,尤其是指针类型和引用,const修饰某个变量,肯定是为了保护该变量的值不被改变。

那什么时候我们需要保护这个值不会被改变呢?

  • 当然是我们需要他作为函数的输入时,输出型参数我们需要他根据函数功能做出改变,所以肯定不需要用const修饰。

那我为什么又说const修饰输入型参数尤其在指针类型和引用呢?其他普通类型呢?

  • 因为我们知道形参是实参的一份临时拷贝,形参的改变不会影响实参,所以我们也就不需要保护它,因为它的改变不会影响到实参本身。

那为什么要用引用,我直接传不就得了,反正也只是拷贝,不会影响本身,这样也省得使用const来保护了。

  • 这里传递引用的价值就在于可以极大的优化节省调用函数的代价!

我们知道引用就是本身的一个别名,实际上就是本身,所以当我们传递引用时,就节省了拷贝的代价,为了效率我们值得做出这样的操作,但为了保护好我们本身不被修改,一定要用const修饰,这样可以增强程序的健壮性!防止我们的程序崩溃。


(2)const修饰返回值

 同样我们用模拟实现来引入。

//access
char& operator[](size_t index);
const char& operator[](size_t index)const;

这里为什么要实现两个版本的下标访问? 

类的有些对象可能可以改变,有的不能改变,举个例子来说,对于string对象来说,有的我们可能希望定义的string对象可以改变,有的则不能改变:

string str1= "hello world";

const string str2 = "hello world";

我们如果进行下标操作符运算,str1[0]='f'这是允许的,而str2[1]='f'则是不允许的。

并且这两个操作调用的成员函数也是不一样的,前一个调用的是string类的非const成员函数,后一个调用的是string类的const成员函数

如果我们没有定义const成员函数,则str2会调用相应的非const成员函数,从而导致我们本来不想其被修改的对象被意外修改。

const修饰返回值的意义何在? 

实际上,如果函数返回的是普通的变量,即值传递,那确实没有什么意义,因为根据前面的学习,函数是如何返回值的呢,实际上是拷贝,拷贝一份内容相同的常量返回,那它本来就是常量你再用const修饰那不多此一举了么。

是的,所以如果函数返回采用值传递的方式,那const这里就没有任何意义。

const修饰返回值真正的意义在于指针传递返回,用来赋值给const修饰的指针内容。

例如函数:

const char* string::c_str()const
{return _str;
}

接收这个函数的返回值我们需要:

void test5()
{string s1 = "hello world!!!!!!!!!!!!!!!!";const char* tmp=s1.c_str();cout << tmp << endl;
}

 为什么要使用引用做返回值?

首先明确:

  • 之前我们说过,除了函数作用域,返回对象就会被销毁,不能用引用返回,否则结果是不确定的。 

那这里为什么可以这样使用呢?

  • 是因为string所使用的空间为动态内存new来的,我们说引用做返回值,可用静态变量或动态内存。

 引用做返回值的主要目的就是能够修改返回的内容,加const是为了适配const string str="abc"这种被const修饰了的字符串。


(3)const修饰成员函数

同样我们用string类的模拟实现引入:

// 返回c在string中第一次出现的位置
size_t find(char c, size_t pos = 0) const;
// 返回子串s在string中第一次出现的位置
size_t find(const char* s, size_t pos = 0) const;

 const修饰成员函数实际上修饰的是该成员函数隐含的this指针,表明在该成员函数中不能对类的任何成员进行修改。

任何不会修改数据成员的函数都应该声明为const 类型(这样const对象和非const对象都可以调用该函数)。如果在编写const 成员函数时,不慎修改了数据成员,或者调用了其它非const 成员函数,编译器将指出错误,这无疑会提高程序的健壮性

即,

  • const对象不可以调用非const成员函数;
  • 非const对象可以调用const成员函数;
  • const成员函数内部不能调用其他非const成员函数;
  • 非const成员函数内部可以调用其他const成员函数。

总结为:权限可以缩小,但是权限不可以放大!


2.有关声明与定义分离可能出现的问题

博主在实现string类时,统一的把函数定义都以内联的形式实现了,当然后期实现完成尝试着想要声明与定义分离下,但却出现了很多报错,这里我梳理一下,形成了以下内容。

(1)合并命名空间

首先分离后的第一个问题就是,由于我之前在头文件中类的实现部分都包含在了命名空间内部。

即,

所以首先需要解决这一问题,我们可以采用左图这种方式,当然右面的方式显然更好:


(2) 指定类域

由于声明定义分离,所以我们需要指定类域:


(3)当声明与定义分离时,缺省参数只需要在声明中体现


 (4)函数重定义问题

如果.h文件中存在函数定义(不包括成员函数),只要存在多个cpp文件,并且他们都包含了string.h头文件,那他们在展开时就会出现重定义的问题。


3.有关模拟实现的传统写法与现代写法

我们来观察一下两种实现方式的区别:

//传统写法
string::string(const string& s)
{_str = new char[s._capacity + 1];strcpy(_str, s._str);_size = s._size;_capacity = s._capacity;
}
//现代写法
string::string(const string& s)
{string tmp(s._str);swap(tmp);
}
//结合swap来观察
void string::swap(string& s)
{std::swap(_str, s._str);std::swap(_size, s._size);std::swap(_capacity, s._capacity);
}

那现代写法优越在哪呢?

现代写法非常巧妙,他完全利用了我们学习构造函数部分的知识。

首先完成用s._str完成构造tmp对象,然后交换*this与tmp,交换后tmp析构。


4.有关string类流插入和流提取操作符的实现思维定势陷阱

在之前的学习中,我们知道流插入与流提取的实现要放在全局从而保证第一个参数不为this指针,但当时面临一个问题,我们实现的是日期类,想要打印日期类就必须访问到类的成员变量,但我们知道成员变量一般都是私有的,所以我们引入了友元的概念用来解决这一问题。

=========================🐸这部分忘了的老铁可以戳🐸=========================

樊梓慕---类和对象(中)之拷贝构造与运算符、操作符重载【CSDN】icon-default.png?t=N7T8http://t.csdnimg.cn/2wbiI🐸🐸🐸🐸🐸🐸🐸🐸🐸🐸🐸🐸🐸🐸🐸🐸🐸🐸🐸🐸🐸🐸🐸🐸🐸🐸🐸🐸🐸🐸🐸


 但这里我们需不需要利用友元呢?

ostream& operator<<(ostream& _cout, const F::string& s)
{for (auto e : s){_cout << e;}return _cout;
}
istream& operator>>(istream& _cin, F::string& s)
{s.clear();char buff[128];char ch = _cin.get();int i = 0;while (ch != ' ' && ch != '\n'){buff[i++] = ch;if (i == 127){buff[i] = '\0';s += buff;i = 0;}ch = _cin.get();}if (i > 0){buff[i] = '\0';s += buff;}return _cin;
}

我们并没有访问string类的成员变量,所以这里不需要利用友元!


🚨请不要形成思维定势,要活学活用🚨


=========================================================================

如果你对该系列文章有兴趣的话,欢迎持续关注博主动态,博主会持续输出优质内容

🍎博主很需要大家的支持,你的支持是我创作的不竭动力🍎

🌟~ 点赞收藏+关注 ~🌟

=========================================================================

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

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

相关文章

3.3 路由器的远程配置

实验3.3 路由器的远程配置 一、任务描述二、任务分析三、具体要求四、实验拓扑五、任务实施&#xff08;一&#xff09;、配置通过Telnet登录系统1.RA的基本配置2.RB的基本配置3.在RA上配置Telnet用户登录界面 &#xff08;二&#xff09;、配置通过STelnet登录系统1.RA的基本配…

熬夜会秃头——beta冲刺Day2

这个作业属于哪个课程2301-计算机学院-软件工程社区-CSDN社区云这个作业要求在哪里团队作业—beta冲刺事后诸葛亮-CSDN社区这个作业的目标记录beta冲刺Day2团队名称熬夜会秃头团队置顶集合随笔链接熬夜会秃头——Beta冲刺置顶随笔-CSDN社区 目录 一、团队成员会议总结 1、成员…

Trie字符串统计(字典树的插入与查找)

题目&#xff1a; 插入模拟&#xff1a;假如现在要依次插入cat,car,busy,cate,bus,car 查找&#xff1a; 代码&#xff1a; import java.util.Scanner;public class Main {public static int[][] chnew int[100010][26];public static int[] cntnew int[100010];public static…

java: 警告: 源发行版 17 需要目标发行版 17

这是一个编译期的报错提示 源发行版 17 , 即说明你的maven项目当前指定的编译版本是jdk17&#xff0c;需要目标发行版 17则是说明你的idea中实际选择的jdk版本并非17 检查你项目中的pom文件中的配置 <properties><java.version>17</java.version><prop…

Windows的常用cmd命令总结

文章目录 一.盘符切换二: cd命令(打开文件/文件夹)三:查看目录四.创建和删除文件夹五.查看本机ip地址六.清除当前屏幕七.复制文件到另一个地方八.移动文件到另一个地方九.删除文件&#xff08;不能删除文件夹&#xff09;十.测试网络连接十一.停止任务进程Windows快捷键总结大全…

KNN实战-图像识别

数据说明 是在循环0-9的数字一直循环500次所得到的数据&#xff0c;然后以手写照片的形式存在 识别的步骤 加载数据构建目标值构建模型参数调优可视化展示 加载数据 import numpy as np import matplotlib.pyplot as plt # 记载数据 data np.load(./digit.npy) data构建目…

webpack 使用打包报错 ERROR in node_modules\@types\node\ts4.8\assert.d.ts

报错如下&#xff1a; 解决方式&#xff0c;先查看自己的 node 版本 node -v然后再安装 types/node 对应版本&#xff0c;比如我的如下 npm i types/node14.10.0 -D然后再次打包&#xff0c;就没有报错了

Hdoop学习笔记(HDP)-Part.19 安装Kafka

目录 Part.01 关于HDP Part.02 核心组件原理 Part.03 资源规划 Part.04 基础环境配置 Part.05 Yum源配置 Part.06 安装OracleJDK Part.07 安装MySQL Part.08 部署Ambari集群 Part.09 安装OpenLDAP Part.10 创建集群 Part.11 安装Kerberos Part.12 安装HDFS Part.13 安装Ranger …

Day47力扣打卡

打卡记录 多边形三角剖分的最低得分&#xff08;区间DP&#xff09; 链接 class Solution:def minScoreTriangulation(self, values: List[int]) -> int:n len(values)f [[0] * n for _ in range(n)]for i in range(n - 3, -1, -1):for j in range(i 2, n):f[i][j] mi…

iris+vue上传到本地存储【go/iris】

iris部分 //main.go package mainimport ("fmt""io""net/http""os" )//上传视频文件部分 func uploadHandler_video(w http.ResponseWriter, r *http.Request) {// 解析上传的文件err : r.ParseMultipartForm(10 << 20) // 设置…

高并发下缓存失效问题-缓存穿透、缓存击穿、缓存雪崩、Redis分布式锁简单实现、Redisson实现分布式锁

文章目录 缓存基本使用范式暴露的几个问题缓存失效问题---缓存穿透缓存失效问题---缓存击穿一、单机锁正确的锁粒度不正确的锁粒度无法保证查询数据库次数是唯一 二、分布式锁getCatalogJsonData()分布式锁演进---基本原理分布式锁(加锁)演进一&#xff1a;删锁失败导致死锁分布…

牛客小白月赛82(A~C)

目录 A.谜题&#xff1a;质数 输入描述 输出描述 输入 输出 解析 B.Kevin逛超市 2 (简单版本) 输入描述 输出描述 输入 输出 思路 C.被遗忘的书籍 题目描述 输入描述 输出描述 输入 输出 输入 输出 思路 比赛链接 牛客小白月赛82_ACM/NOI/CSP/CCPC/ICPC算…

C++基础 -28- 友元

友元用于访问类中的所有数据成员 类中的私有成员&#xff0c;类外不可访问 定义友元的格式&#xff08;友元函数必须要在类内&#xff0c;声明&#xff09; friend void show(person &b); 使用友元访问类的所有成员 #include "iostream"using namespace std…

Istio可观测性

Istio可观测性 image-20231129072302901 前言 Istio 为网格内所有的服务通信生成详细的遥测数据。这种遥测技术提供了服务行为的可观测性&#xff0c;使运维人员能够排查故障、维护和优化应用程序&#xff0c;而不会给开发人员带来其他额外的负担。通过 Istio&#xff0c;运维…

好用的桌面管理软件推荐

随着电脑的普及&#xff0c;桌面管理软件已经成为我们日常生活和工作中不可或缺的一部分。一个好的桌面管理软件可以帮助我们更高效地组织和管理电脑上的文件和应用程序&#xff0c;提高我们的工作效率。下面&#xff0c;我将为大家推荐几款好用的桌面管理软件。 1、腾讯桌面整…

Nacos 注册中心下载到搭建详细步骤【微服务】

文章目录 一、下载与安装二、Nacos 服务注册1. 引入依赖2. 修改配置文件3. 开启 Nacos 注解4. 启动项目 三、Nacos 服务集群1. 模拟多实例部署2. 配置集群属性3. 服务权重配置 四、Nacos 环境隔离五、Nacos 注册中心原理1. Nacos 与 Eureka 比较2. 配置非临时实例 一、下载与安…

LeNet对MNIST 数据集中的图像进行分类--keras实现

我们将训练一个卷积神经网络来对 MNIST 数据库中的图像进行分类&#xff0c;可以与前面所提到的CNN实现对比CNN对 MNIST 数据库中的图像进行分类-CSDN博客 加载 MNIST 数据库 MNIST 是机器学习领域最著名的数据集之一。 它有 70,000 张手写数字图像 - 下载非常简单 - 图像尺…

MMdetection3.0 问题:DETR验证集上AP值全为0.0000

MMdetection3.0 问题&#xff1a;DETR验证集上AP值全为0.0000 条件&#xff1a; 1、选择的是NWPU-VHR-10数据集&#xff0c;其中训练集455张&#xff0c;验证、测试相同均为195张。 2、在Faster-rcnn、YOLOv3模型上均能训练成功&#xff0c;但是在DETR训练时&#xff0c;出现验…

《地理信息系统原理》笔记/期末复习资料(7. 空间分析)

目录 7. 空间分析 7.1 空间分析的内容与步骤 7.2 数据检索及表格分析 7.2.1 属性统计分析 7.2.2 布尔逻辑查询 7.2.3 空间数据库查询语言 7.2.4 重分类&#xff0c;边界消除与合并 7.3 叠置分析 7.3.1 栅格系统的叠加分析 7.3.2 矢量系统的叠加分析&#xff08;拓扑叠…