String类(三)

文章目录

  • string类(三)
    • string类的模拟实现:
    • 1.默认成员变量和函数
    • 2.string的长度和下表引用
    • 3.字符串拷贝构造
    • 4. +=赋值拷贝
    • 5.字符串比较
    • 6.字符串的增添操作
    • 7.insert插入操作
    • 8.遍历字符

string类(三)

string类的模拟实现:

1.默认成员变量和函数

首先自定义构造string类的默认成员变量和函数

声明和定义均在头文件(自己创建的string.h)中进行,主函数仅调用函数

#pragma once
#include <iostream>
using namespace std;
#include <string>namespace pzh   //pzh为命名空间名称 
{class string  //pzh::string为自定义的string类的名称{public:string(const char* str = "")  //构造函数   : _size(strlen(str)){_capacity = _size;_str = new char[_capacity + 1];strcpy(_str, str);}~string()  //析构函数{delete[] _str;_str = nullptr;_size = _capacity = 0;}const char* c_str()  //打印字符串函数{return _str;}private:char* _str;size_t _size;size_t _capacity;};void func_str1(){string s1;string s2("hello solity");cout << s1.c_str() << endl;cout << s2.c_str() << endl;}
}

运行结果:

在这里插入图片描述

2.string的长度和下表引用

定义size和**[ ]**:

size_t size() const
{return _size;
}char& operator[](size_t pos)
{assert(pos < _size);  //需要引入头文件 #inlcude <assert.h>return _str[pos];
}

测试功能调用:

void func_str()
{string s("hello solity");cout << s.size() << endl;for (size_t i = 0; i < s.size(); ++i){cout << s[i] << " ";}
}

运行结果:

在这里插入图片描述

3.字符串拷贝构造

此时如果需要进行字符串复制拷贝,便需要进行构造拷贝函数

string(const string& s)  //拷贝构造: _size(s._size)  //将s的size长度赋值给成员变量_size, _capacity(s._capacity)
{_str = new char[s._capacity + 1];  //加1是给'\0'准备的(深拷贝)strcpy(_str, s._str);
}

调用测试:

void func_str()
{string s1("hello solity");  //sstring s2(s1);  //thiscout << s1.c_str() << endl;cout << s2.c_str() << endl;
}

运行结果:

在这里插入图片描述

4. +=赋值拷贝

赋值不止有这种,当然还有**=**号进行赋值拷贝

string& operator=(const string& s)
{if (this != &s){char* tmp = new char[s._capacity + 1];  //先开新空间strcpy(tmp, s._str); //进行拷贝delete[] _str; //释放旧空间(this)_str = tmp;  //把新开空间给旧空间_size = s._size;_capacity = s._capacity;}return *this;
}

测试运行:

void func_str()
{string s1; string s2("hello solity");cout << "Before s1 = " << s1.c_str() << endl;cout << "Before s2 = " << s2.c_str() << endl;s1 = s2;cout << "After s1 = " << s1.c_str() << endl;cout << "After s2 = " << s2.c_str() << endl;
}

输出结果:

在这里插入图片描述

5.字符串比较

当需要进行字符串比较的操作时:

bool operator>(const string& s) const
{return strcmp(_str, s._str) > 0;
}bool operator==(const string& s) const
{return strcmp(_str, s._str) == 0;
}bool operator>=(const string& s) const
{return *this > s || *this == s;
}bool operator<(const string& s) const
{return !(*this >= s);
}bool operator<=(const string& s) const
{return !(*this > s);
}bool operator!=(const string& s) const
{return !(*this == s);
}

测试运行:

void func_str()
{string s1("hello solity");string s2("hello pzh");string s3("abc");cout << (s1 < s2) << endl;cout << (s1 == s2) << endl;cout << (s1 >= s2) << endl; 
}

输出结果:

在这里插入图片描述

6.字符串的增添操作

void reserve(size_t n)
{char* tmp = new char[n + 1];strcpy(tmp, _str);delete[] _str;_str = tmp;_capacity = n;
}void push_back(char ch)  //添加字符
{if (_size + 1 > _capacity){reserve(_capacity * 2);}_str[_size] = ch;++_size;_str[_size] = '\0';
}void append(const char* str)  //添加字符串
{size_t len = strlen(str);if (_size + len > _capacity){reserve(len + _size);}strcpy(_str + _size, str);_size += len;
}

测试运行:

void func_str()
{string s1("hello solity");s1.push_back(' ');cout << s1.c_str() << endl;s1.append("#####################");cout << s1.c_str() << endl;;
}

输出结果:

在这里插入图片描述

当然,追加操作这样就显得很臃肿,我们可以尝试更简介的**+=**操作符来进行:

string& operator+=(const char* str)
{append(str);return *this;
}

测试运行:

void func_str()
{string s1("hello solity");s1.push_back(' ');cout << s1.c_str() << endl;s1.append("#####################");cout << s1.c_str() << endl;s1 += "pzh solity";cout << s1.c_str() << endl;
}

输出结果:

在这里插入图片描述

7.insert插入操作

void insert(size_t pos, char ch)
{assert(pos <= _size);if (_size + 1 > _capacity){reserve(2 * _capacity);}size_t end = _size;while (end >= pos){_str[end + 1] = _str[end];--end;}_str[pos] = ch;++_size;
}

测试运行:

void func_str()
{string s1("hello solity");cout << "Before s1 = " << s1.c_str() << endl;s1.insert(2, 'a');cout << "After s1 = " << s1.c_str() << endl;
}

输出结果:

在这里插入图片描述

8.遍历字符

测试运行:

void func_str()
{string s1("hello solity");for (size_t i = 0; i < s1.size(); ++i){cout << s1[i] << " ";  //输出运算后结果}cout << endl;for (size_t i = 0; i < s1.size(); ++i){s1[i]++;  //对每个字符进行加1操作}for (size_t i = 0; i < s1.size(); ++i){cout << s1[i] << " ";  //输出运算后结果}cout << endl;
}

输出结果:

在这里插入图片描述

如果使用范围for进行输出,便需要用到迭代器:

typedef char* iterator;
typedef const char* const_iterator;iterator begin()
{return _str;
}
iterator end()
{return _str + _size;
}const_iterator begin() const
{return _str;
}
const_iterator end() const
{return _str + _size;
}

测试运行:

void func_str()
{string s("hello solity");string::iterator it = s.begin();while (it != s.end()){cout << *it << " ";++it;}cout << endl;for (auto ch : s){cout << ch << " ";}}

输出结果:

iterator end()
{
return _str + _size;
}

const_iterator begin() const
{
return _str;
}
const_iterator end() const
{
return _str + _size;
}


测试运行:```c++
void func_str()
{string s("hello solity");string::iterator it = s.begin();while (it != s.end()){cout << *it << " ";++it;}cout << endl;for (auto ch : s){cout << ch << " ";}}

输出结果:

在这里插入图片描述
综上,完整的string.h头文件如下:

#pragma once
#include <iostream>
using namespace std;
#include <string>
#include <assert.h>namespace pzh  
{class string{public:typedef char* iterator;typedef const char* const_iterator;iterator begin(){return _str;}iterator end(){return _str + _size;}const_iterator begin() const{return _str;}const_iterator end() const{return _str + _size;}string(const char* str = "")  //构造函数   : _size(strlen(str)){_capacity = _size;_str = new char[_capacity + 1];strcpy(_str, str);}~string()  //析构函数{delete[] _str;_str = nullptr;_size = _capacity = 0;}const char* c_str()  //打印字符串函数{return _str;}size_t size() const  //字符串长度{return _size;}char& operator[](size_t pos)  //字符串下表引用{assert(pos < _size);return _str[pos];}string(const string& s)  //拷贝构造: _size(s._size)  //将s的size长度赋值给成员变量_size, _capacity(s._capacity){_str = new char[s._capacity + 1];  //加1是给'\0'准备的(深拷贝)strcpy(_str, s._str);}string& operator=(const string& s){if (this != &s){char* tmp = new char[s._capacity + 1];strcpy(tmp, s._str);delete[] _str;  _str = tmp;_size = s._size;_capacity = s._capacity;delete[] tmp;}return *this;}//不修改成员变量的函数,保险起见统统加constbool operator>(const string& s) const{return strcmp(_str, s._str) > 0;}bool operator==(const string& s) const{return strcmp(_str, s._str) == 0;}bool operator>=(const string& s) const{return *this > s || *this == s;}bool operator<(const string& s) const{return !(*this >= s);}bool operator<=(const string& s) const{return !(*this > s);}bool operator!=(const string& s) const{return !(*this == s);}void reserve(size_t n){char* tmp = new char[n + 1];strcpy(tmp, _str);delete[] _str;_str = tmp;_capacity = n;}void push_back(char ch){if (_size + 1 > _capacity){reserve(_capacity * 2);}_str[_size] = ch;++_size;_str[_size] = '\0';}void append(const char* str){size_t len = strlen(str);if (_size + len > _capacity){reserve(len + _size);}strcpy(_str + _size, str);_size += len;}string& operator+=(const char* str){append(str);return *this;}void insert(size_t pos, char ch){assert(pos <= _size);if (_size + 1 > _capacity){reserve(2 * _capacity);}size_t end = _size;while (end >= pos){_str[end + 1] = _str[end];--end;}_str[pos] = ch;++_size;}private:char* _str;size_t _size;size_t _capacity;};void func_str(){//...测试功能代码}
}

main.cpp主文件如下:

#define _CRT_SECURE_NO_WARNINGS
#include "string.h"int main()
{pzh::func_str();return 0;
}

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

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

相关文章

Django(二)-搭建第一个应用(1)

一、项目环境和结构 1、项目环境 2、项目结构 二、编写项目 1、创建模型 代码示例: import datetimefrom django.db import models from django.utils import timezone# Create your models here.class Question(models.Model):question_text models.CharField(max_length2…

ES6 基础

文章目录 1. 初识 ES62. let 声明变量3. const 声明常量4. 解构赋值 1. 初识 ES6 ECMAScript6.0(以下简称ES6)是JavaScript语言的下一代标准&#xff0c;已经在2015年6月正式发布了。它的目标&#xff0c;是使得」JavaScript语言可以用来编写复杂的大型应用程序&#xff0c;成为…

【环境配置】Ubuntu MySQL 8.0.28 安装并允许外部客户端连接

文章目录 MySQL 安装步骤配置 MySQL Server 允许外部连接 MySQL 安装步骤 步骤一&#xff1a;在 MySQL 官网找到 apt 仓库&#xff0c;下载最新的仓库 点击 Download&#xff1a; 输入如下命令&#xff1a; sudo wget -c https://dev.mysql.com/get/mysql-apt-config_0.8…

Java拆装箱及128陷阱

有以下一段代码&#xff1a; Integer a 123; Integer b 123; int c 123; int d 123; System.out.println(c d); System.out.println(a b); System.out.println(a c); 这段代码运行的结果是什么呢&#xff1f; c d 一定为True。 由于Java中存在自动拆装箱&#xff0…

【Spring】Spring框架中的一个核心接口ApplicationContext 简介,以及入口 Run() 的源码分析

一、简介 ApplicationContext 是Spring框架中的一个核心接口&#xff0c;它是Spring IoC容器的实现之一&#xff0c;用于管理和组织应用程序中的各种Bean&#xff0c;同时提供了一系列功能来支持依赖注入、AOP等特性。 简单来说&#xff0c;ApplicationContext 是一个大型的、…

如何备考2025年AMC8竞赛?吃透2000-2024年600道真题(免费送题)

最近有家长朋友问我&#xff0c;现在有哪些类似于奥数的比赛可以参加&#xff1f;我的建议可以关注下AMC8的竞赛&#xff0c;类似于国内的奥数&#xff0c;但是其难度要比国内的奥数低一些&#xff0c;而且比赛门槛更低&#xff0c;考试也更方便。比赛的题目尤其是应用题比较有…

FPGA之组合逻辑与时序逻辑

数字逻辑电路根据逻辑功能的不同&#xff0c;可以分成两大类&#xff1a;组合逻辑电路和时序逻辑电路&#xff0c;这两种电路结构是FPGA编程常用到的&#xff0c;掌握这两种电路结构是学习FPGA的基本要求。 1.组合逻辑电路 组合逻辑电路概念&#xff1a;任意时刻的输出仅仅取决…

2015年认证杯SPSSPRO杯数学建模B题(第一阶段)替换式密码全过程文档及程序

2015年认证杯SPSSPRO杯数学建模 B题 替换式密码 原题再现&#xff1a; 历史上有许多密码的编制方法。较为简单的是替换式密码&#xff0c;也就是将文中出现的字符一对一地替换成其它的符号。对拼音文字而言&#xff0c;最简单的形式是单字母替换加密&#xff0c;也就是以每个…

小迪安全48WEB 攻防-通用漏洞Py 反序列化链构造自动审计 bandit魔术方法

#知识点&#xff1a; 1、Python-反序列化函数使用 2、Python-反序列化魔术方法 3、Python-反序列化 POP 链构造&#xff08;payload构造&#xff09; 4、Python-自动化审计 bandit 使用 #前置知识&#xff1a; 函数使用&#xff1a; pickle.dump(obj, file) : 将对…

pycharm使用远程服务器的jupyter环境

1、确保服务器上安装了jupyter,如果没有&#xff0c;执行下面命令安装 pip install jupyter2、启动jupyter notebook服务 jupyter notebook --no-browser --port8888 --ip0.0.0.0 --allow-root表明在服务器的8888 端口上启动 Jupyter Notebook&#xff0c;并允许从任何 IP 地…

caffe | 使用caffe SSD制作VOC07112 lmdb数据集

git clone -b ssd https://github.com/weiliu89/caffe.git caffe_ssdcd caffe_ssdcp caffe/Makefile.config caffe_ssd/# 把 cuda 和 cudnn 关了&#xff0c;用 cpu 版本的就好了 make -j32 make pycaffemake test -j8 make runtest -j8 vim ~/.bashrc# 加入 export LD_LIBRAR…

亮数据代理IP轻松解决爬虫数据采集痛点

文章目录 一、爬虫数据采集痛点二、为什么使用代理IP可以解决&#xff1f;2.1 爬虫和代理IP的关系2.2 使用代理IP的好处 三、亮数据代理IP的优势3.1 IP种类丰富3.1.1 动态住宅代理IP3.1.2 静态住宅代理IP3.1.3 机房代理IP3.1.4 移动代理IP 3.2 高质量IP全球覆盖3.3 超级代理服务…

达梦DEM部署说明-详细步骤-DM8达梦数据库

DMDEM部署说明-详细步骤-DM8达梦数据库 环境介绍1 部署DM8 数据库 1.1 创建一个数据库作为DEM后台数据库1.2 创建数据库用户 DEM1.3 使用DEM用户导入dem_init.sql 2 配置tomcat 2.1 配置/tomcat/conf/server.xml2.2 修改jvm启动参数 3 配置JAVA 1.8及以上版本的运行时环境 3.1…

线性数据结构----(数组,链表,栈,队列,哈希表)

线性数据结构 数组链表栈使用场景 队列应用场景 哈希表特点哈希函数&#xff0c;哈希值&#xff0c;哈希冲突键值对 Entry 开放寻址法和拉链法 参考文档 数组 数组(Array) 是一种很常见的数据结构。由相同类型的元素组成&#xff0c;并且是使用一块连续的内存来存储的。 在数组…

【二】TensorFlow神经网络模型构建之卷积函数

卷积函数是构建神经网络的重要支架&#xff0c;是在一批图像上扫描的二维过滤器。 tf.nn.convolution(input,filter,padding,stridesNone,dilation_rateNone,nameNone,data_formatNone)该函数计算N维卷积的和。tf.nn.conv2d(input,filter,padding,strides,use_cudnn_on_gpuNon…

java算法第31天 | 贪心算法 part01 ● 理论基础 ● 455.分发饼干 ● 376. 摆动序列 ● 53. 最大子序和

理论基础 贪心算法没有固定的套路&#xff0c;贪心的本质是选择每一阶段的局部最优&#xff0c;从而达到全局最优。 贪心算法一般分为如下四步&#xff1a; 将问题分解为若干个子问题 找出适合的贪心策略 求解每一个子问题的最优解 将局部最优解堆叠成全局最优解 这个四步其…

分布式系统面试全集通第一篇(dubbo+redis+zookeeper----分布式+CAP+BASE+分布式事务+分布式锁)

目录 分布式系统面试全集通第一篇什么是分布式?和微服务的区别什么是分布式分布式与微服务的区别 什么是CAP?为什么不能三者同时拥有分区容错性一致性可用性 Base理论了解吗基本可用软状态最终一致性 什么是分布式事务分布式事务有哪些常见的实现方案?2PC&#xff08;Two Ph…

如何查询电脑是否被锁定了IP地址?锁定IP会出现什么问题?

前言 电脑刚到手的时候&#xff0c;基本上是通过路由器DHCP进行IP分配的。路由器DHCP分配IP给电脑的好处是网络不会出现IP冲突&#xff0c;网络能正常使用。 有些电脑可能在DHCP自动获取IP时出现错误&#xff0c;所以小伙伴就会通过手动设置IP让电脑可以正常上网。 这样的操…

3-iperf3 使用什么工具可以检测网络带宽、延迟和数据包丢失率等网络性能参数呢?

(1)iperf3简介 1.iperf3简介 2.用途&#xff08;特点&#xff09; 3.下载iperf3地址 &#xff08;2&#xff09;实战 1.iperf3参数 &#xff08;1&#xff09;通用参数&#xff08;客户端和服务器端都是适用的&#xff09; &#xff08;2&#xff09;客户端参数 实验1&…

35.基于SpringBoot + Vue实现的前后端分离-在线考试系统(项目 + 论文)

项目介绍 本站是一个B/S模式系统&#xff0c;采用SpringBoot Vue框架&#xff0c;MYSQL数据库设计开发&#xff0c;充分保证系统的稳定性。系统具有界面清晰、操作简单&#xff0c;功能齐全的特点&#xff0c;使得基于SpringBoot Vue技术的在线考试系统设计与实现管理工作系统…