C++ 泛编程 —— 嵌套使用模板类

嵌套使用模板类

  • 嵌套使用模板类最常见的场景
  • 数组容器中有栈容器
  • 栈容器中有数组容器
  • 递归使用模板类

嵌套使用模板类最常见的场景

容器中有容器
数组元素可以是中的元素可以是数组。先来看一下StackVector的基本代码,定长数组Array的代码也给出来,但是不会用到,代码如下:

#include <iostream>
using namespace std;// 定长数组 Array
template <class T, int len=10>
class Array {
private:T* items[len];
public:Array() {}~Array() {}T& operator[](int index) {return items[index];}const T& operator[] (int index) const {return items[index];}
};// 栈 Stack
template <class DataType>
class Stack {
private:DateType* items;int stacksize;   int top;
public:Stack(int size=3): stacksize(size), top(0) {items  = new DataType[stacksize];}~Stack() {delete [] items; items = nullptr;}bool isEmpty() {return top == 0;}bool isFull() {return top == stacksize;}bool push(const DateType& item) {if(top < stacksize) {items[top++] = item; return true;}return false;}bool pop() {if(top > 0) {item = items[--top]; return true;}return false;}
};// 动态数组 Vector
template <class T>
class Vector {
privatet:int len; // 数组元素的个数T* items; // 数组元素
public:Vector(int size=2): len(size) { // 元素个数的缺省值为 2items = new T[len];}~Vector() {delete [] items; items = nullptr;}void resize(int size) {if(size <= len) return; // 只能往更大的方向扩容T* tmp = new T[size]; // 分配更大的内存空间for(int i = 0; i < len; i++) tmp[i] = items[i]; // 复制旧数组元素到新数组delete [] items; // 释放旧数组items = tmp;len = size;}int size() {return len;}T& operator [] (int index) {if(index >= len) resize(index + 1);return items[index];}const T& operator [] (int index) const {return items[index];} // 重载操作符[],不能修改数组中的元素。
};int main() {return 0;
}

数组容器中有栈容器

目前的代码中,Vector容器的大小缺省值是2Stack容器的大小缺省值是3。现在演示数组容器Vector中有栈Stack容器的情况,代码如下:

#include <iostream>
using namespace std;// 栈 Stack
template <class DataType>
class Stack {
private:DataType* items;int stacksize;   int top;
public:Stack(int size=3): stacksize(size), top(0) {items  = new DataType[stacksize];}~Stack() {delete [] items; items = nullptr;}bool isEmpty() {return top == 0;}bool isFull() {return top == stacksize;}bool push(const DataType& item) {if(top < stacksize) {items[top++] = item; return true;}return false;}bool pop(DataType& item) {if(top > 0) {item = items[--top]; return true;}return false;}
};// 动态数组 Vector
template <class T>
class Vector {
private:int len; // 数组元素的个数T* items; // 数组元素
public:Vector(int size=2): len(size) { // 元素个数的缺省值为 2items = new T[len];}~Vector() {delete [] items; items = nullptr;}void resize(int size) {if(size <= len) return; // 只能往更大的方向扩容T* tmp = new T[size]; // 分配更大的内存空间for(int i = 0; i < len; i++) tmp[i] = items[i]; // 复制旧数组元素到新数组delete [] items; // 释放旧数组items = tmp;len = size;}int size() {return len;}T& operator [] (int index) {if(index >= len) resize(index + 1);return items[index];}const T& operator [] (int index) const {return items[index];} // 重载操作符[],不能修改数组中的元素。};int main() {// 创建Vector对象vs,vs是一个Vector,Vector中存储Stack<string>类型的元素。Vector<Stack<string>> vs;// 往容器中插入数据vs[0].push("hello world!"); vs[0].push("hihihi"); vs[0].push("123456"); // vs[0]是一个栈,往vs[0]中插入数据vs[1].push("你好!"); vs[1].push("abcdef"); vs[1].push("xxxxx"); // vs[1]也是一个栈,往vs[1]中插入数据// 用嵌套循环,遍历vs中的所有元素for(int i = 0; i < vs.size(); i++) {while(vs[i].isEmpty() == false) {string item;vs[i].pop(item);cout << "pop item = " << item << endl;}}return 0;
}

运行的结果如下:

pop item = 123456
pop item = hihihi
pop item = hello world!
pop item = xxxxx
pop item = abcdef
pop item = 你好!

容器中的容器就是二维容器,但不能简单把它说成二维数组。因为不同的容器可以实现不同的数据结构。像二维数组,但不是二维数组。目前的程序,Stack是没有扩展功能的,而Vector是有resize()扩展功能的,那就往Vector容器中再多加一个元素,只入栈两个元素,其他代码不变,main()函数代码如下:

int main() {Vector<Stack<string>> vs;vs[0].push("hello world!"); vs[0].push("hihihi"); vs[0].push("123456");vs[1].push("你好!"); vs[1].push("abcdef"); vs[1].push("xxxxx");vs[2].push("Happy Everyday!"); vs[1].push("吃好喝好,身体健康!"); // 往Vector中再增加一个Stack数据for(int i = 0; i < vs.size(); i++) {while(vs[i].isEmpty() == false) {string item;vs[i].pop(item);cout << "pop item = " << item << endl;}}return 0;
}

我的电脑编译运行的结果如下,你的电脑可能编译不通过,或者编译运行的结果与我的不同,总之是不正常的:

pop item = 123456
pop item = hihihi
pop item =
pop item = xxxxx
pop item = abcdef
pop item =
pop item = Happy Everyday!
free(): double free detected in tcache 2
Aborted (core dumped)

目前这个代码异常的原因在Vector类中,具体在扩展数组内存空间的resize()函数里面。 tmp[i] = items[i];这条语句的意思是:把原数组中的数据拷贝到新数组中去。如果复制的是C++内置的数据类型,则不存在问题;如果复制的是自定义的类,并且类中使用了堆区内存,就存在浅拷贝的问题。
具体解释一下,在Stack类中,根据语句:DataType* items;可以确定items这个成员变量是指针,使用了堆区内存。这样的话,对Stack类用浅拷贝是不行的,要用深拷贝。也就是说,对于Stack这种类,一定要重写拷贝构造函数和赋值函数。在此实例中,未用到Stack类的拷贝构造函数,无需在意;但这条语句:tmp[i] = items[i];用到了Stack类的赋值函数。所以,应该为Stack类重写赋值函数,实现深拷贝。具体的做法是,在Stack类中添加如下代码:

Stack& operator = (const Stack& v) {delete [] items;stacksize = v.stacksize;items = new DataType[stacksize];for(int i = 0; i < stacksize; i++) {items[i] = v.items[i];}top = v.top;return *this;
}

再次编译运行,结果如下:

pop item = 123456
pop item = hihihi
pop item = hello world!
pop item = xxxxx
pop item = abcdef
pop item = 你好!
pop item = Happy Everyday!

栈容器中有数组容器

Vector类也加上赋值运算符的重载函数,实现深拷贝,以便演示栈容器Stack中有数组容器Vector的情况,代码如下:

#include <iostream>
using namespace std;// 栈 Stack
template <class DataType>
class Stack {
private:DataType* items;int stacksize;   int top;
public:Stack(int size=3): stacksize(size), top(0) {items  = new DataType[stacksize];}~Stack() {delete [] items; items = nullptr;}Stack& operator = (const Stack& v) {delete [] items;stacksize = v.stacksize;items = new DataType[stacksize];for(int i = 0; i < stacksize; i++) {items[i] = v.items[i];}top = v.top;return *this;}bool isEmpty() {return top == 0;}bool isFull() {return top == stacksize;}bool push(const DataType& item) {if(top < stacksize) {items[top++] = item; return true;}return false;}bool pop(DataType& item) {if(top > 0) {item = items[--top]; return true;}return false;}
};// 动态数组 Vector
template <class T>
class Vector {
private:int len; // 数组元素的个数T* items; // 数组元素
public:Vector(int size=2): len(size) { // 元素个数的缺省值为 2items = new T[len];}~Vector() {delete [] items; items = nullptr;}Vector& operator = (const Vector& v) {delete [] items;len = v.len;items = new T[len];for(int i = 0; i < len; i++) items[i] = v.items[i];return *this;}void resize(int size) {if(size <= len) return; // 只能往更大的方向扩容T* tmp = new T[size]; // 分配更大的内存空间for(int i = 0; i < len; i++) tmp[i] = items[i]; // 复制旧数组元素到新数组delete [] items; // 释放旧数组items = tmp;len = size;}int size() {return len;}T& operator [] (int index) {if(index >= len) resize(index + 1);return items[index];}const T& operator [] (int index) const {return items[index];} // 重载操作符[],不能修改数组中的元素。};int main() {// 创建Stack对象,sv是一个栈,栈中存储Vector<string>类型的元素。Stack<Vector<string>> sv;// 先创建Vector<string>对象,再插入到Stack中Vector<string> tmp;// 第一次把字符串数据放到临时容器tmp中,再第一次入栈。tmp[0] = "hi ~"; tmp[1] = "hello ~"; sv.push(tmp);// 第二次把字符串数据放到临时容器tmp中,再第二次入栈。tmp[0] = "666"; tmp[1] = "888"; sv.push(tmp);// 第二次把字符串数据放到临时容器tmp中,再第二次入栈。tmp[0] = "qqqqq"; tmp[1] = "324fwre"; tmp[2] = "09ji"; sv.push(tmp);// 用嵌套循环,遍历sv中的所有元素while(sv.isEmpty() == false) {sv.pop(tmp);for(int i = 0; i < tmp.size(); i++) {cout << "sv[" << i << "] = " << tmp[i] << endl;}}return 0;
}

运行结果如下:

sv[0] = qqqqq
sv[1] = 324fwre
sv[2] = 09ji
sv[0] = 666
sv[1] = 888
sv[0] = hi ~
sv[1] = hello ~

递归使用模板类

递归使用模板类,属于嵌套使用模板类的特殊情况,自己嵌套自己。对于以上代码,仅需修改main函数中的调用代码即可,参考如下:

int main() {// Vector嵌套VectorVector<Vector<string>> vv; // 递归使用模板类vv[0][0] = "hello"; vv[0][1] = "world"; vv[0][2] = "!";vv[1][0] = "你好"; vv[1][1] = "世界";vv[2][0] = "Happy"; vv[2][1] = "Everyday"; vv[2][2] = "!"; vv[2][3] = "吃好喝好";for(int i = 0; i <vv.size(); i++){for(int j = 0; j < vv[i].size(); j++) {cout << "vv[" << i << "][" << j << "] = " << vv[i][j] << endl;}}return 0;
}

运行结果如下:

vv[0][0] = hello
vv[0][1] = world
vv[0][2] = !
vv[1][0] = 你好
vv[1][1] = 世界
vv[2][0] = Happy
vv[2][1] = Everyday
vv[2][2] = !
vv[2][3] = 吃好喝好

注意,这跟二维数组不同二维数组是一个矩阵第二维的大小是固定的。而vv的大小不是固定的。也可以调整一下输出格式,更能直观感受vv第二维不固定的特点,代码如下:

int main() {// Vector嵌套VectorVector<Vector<string>> vv; // 递归使用模板类vv[0][0] = "hello"; vv[0][1] = "world"; vv[0][2] = "xxx";vv[1][0] = "你好"; vv[1][1] = "世界";vv[2][0] = "Happy"; vv[2][1] = "Everyday"; vv[2][2] = "qqq"; vv[2][3] = "吃好喝好";for(int i = 0; i <vv.size(); i++){for(int j = 0; j < vv[i].size(); j++) {cout << vv[i][j] << " ";}cout << endl;}return 0;
}

运行的结果如下:

hello world xxx
你好 世界
Happy Everyday qqq 吃好喝好

由此可以看出,vv的第一行有3个元素、第二行有2个元素、第三行有4个元素。故Vector<Vector<string>> vv;的第二个纬度是不固定的。

感谢浏览,一起学习!

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

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

相关文章

【Ubuntu 20.4安装截图软件 flameshot 】

步骤一&#xff1a; 安装命令&#xff1a; sudo apt-get install flameshot 步骤二&#xff1a; 设置快捷方式&#xff1a; Ubuntu20.4 设置菜单&#xff0c;点击 号 步骤三&#xff1a; 输入软件名称&#xff0c; 软件快捷命令&#xff08;flameshot gui&#xff09;&am…

WordPress源码解析-数据库表结构

WordPress是一个功能强大的内容管理系统&#xff0c;它使用MySQL数据库来存储和管理网站的内容、用户和配置信息。作为WordPress开发者&#xff0c;了解WordPress数据库的结构和各表的作用至关重要&#xff0c;因为这将帮助您更好地开发插件和主题&#xff0c;以及执行高级数据…

Java中使用四叶天动态代理IP构建ip代理池,实现httpClient和Jsoup代理ip爬虫

在本次爬虫项目中&#xff0c;关于应用IP代理池方面&#xff0c;具体完成以下功能&#xff1a; 从指定API地址提取IP到ip池中&#xff08;一次提取的IP数量可以自定义更改&#xff09; 每次开始爬虫前&#xff08;多条爬虫线程并发执行&#xff09;&#xff0c;从ip池中获取一…

Python机器学习笔记(十三、k均值聚类)

聚类&#xff08;clustering&#xff09;是将数据集划分成组的任务&#xff0c;这些组叫作簇&#xff08;cluster&#xff09;。其目标是划分数据&#xff0c;使得一个簇内的数据点非常相似且不同簇内的数据点非常不同。与分类算法类似&#xff0c;聚类算法为每个数据点分配&am…

KNN分类算法 HNUST【数据分析技术】(2025)

1.理论知识 KNN&#xff08;K-Nearest Neighbor&#xff09;算法是机器学习算法中最基础、最简单的算法之一。它既能用于分类&#xff0c;也能用于回归。KNN通过测量不同特征值之间的距离来进行分类。 KNN算法的思想&#xff1a; 对于任意n维输入向量&#xff0c;分别对应于特征…

宝塔-firefox(Docker应用)-构建自己的Web浏览器

安装基础软件 宝塔中安装firefox(Docker应用) 。宝塔中需要先安装docker及docker-composefirefox配置安装 点击firefox应用&#xff0c;选择【安装配置】点击右边绿色按钮&#xff0c;进行安装&#xff0c;这一步等待docker-compose根据你的配置初始化docker应用 等待安装 …

如何从 0 到 1 ,打造全新一代分布式数据架构

导读&#xff1a;本文从 DIKW&#xff08;数据、信息、知识、智慧&#xff09; 模型视角出发&#xff0c;探讨数字世界中数据的重要性问题。接着站在业务视角&#xff0c;讨论了在不断满足业务诉求&#xff08;特别是 AI 需求&#xff09;的过程中&#xff0c;数据系统是如何一…

Docker部署GitLab服务器

一、GitLab介绍 1.1 GitLab简介 GitLab 是一款基于 Git 的开源代码托管平台&#xff0c;集成了版本控制、代码审查、问题跟踪、持续集成与持续交付&#xff08;CI/CD&#xff09;等多种功能&#xff0c;旨在为团队提供一站式的项目管理解决方案。借助 GitLab&#xff0c;开发…

芯片Tapeout power signoff 之IR Drop Redhawk Ploc文件格式及其意义

数字IC后端工程师在芯片流程最后阶段都会使用redhawk或voltus进行设计的IR Drop功耗signoff分析。必须确保静态&#xff0c;动态ir drop都符合signoff标准。 在做redhawk ir drop分析前&#xff0c;我们需要提供一个redhawk ploc供电点坐标。 数字IC设计后端实现前期预防IR D…

HarmonyOS NEXT 实战之元服务:静态案例效果---查看国内航班服务

背景&#xff1a; 前几篇学习了元服务&#xff0c;后面几期就让我们开发简单的元服务吧&#xff0c;里面丰富的内容大家自己加&#xff0c;本期案例 仅供参考 先上本期效果图 &#xff0c;里面图片自行替换 效果图1完整代码案例如下&#xff1a; Index代码 import { authen…

python+requests接口自动化测试框架实例详解

前段时间由于公司测试方向的转型&#xff0c;由原来的web页面功能测试转变成接口测试&#xff0c;之前大多都是手工进行&#xff0c;利用postman和jmeter进行的接口测试&#xff0c;后来&#xff0c;组内有人讲原先web自动化的测试框架移驾成接口的自动化框架&#xff0c;使用的…

前端:改变鼠标点击物体的颜色

需求&#xff1a; 需要改变图片中某一物体的颜色&#xff0c;该物体是纯色&#xff1b; 鼠标点击哪个物体&#xff0c;哪个物体的颜色变为指定的颜色&#xff0c;利用canvas实现。 演示案例 代码Demo <!DOCTYPE html> <html lang"en"><head>&l…

聊一聊 C#前台线程 如何阻塞程序退出

一&#xff1a;背景 1. 讲故事 这篇文章起源于我的 C#内功修炼训练营里的一位朋友提的问题&#xff1a;后台线程的内部是如何运转的 ? &#xff0c;犹记得C# Via CLR这本书中 Jeffery 就聊到了他曾经给别人解决一个程序无法退出的bug&#xff0c;最后发现是有一个 Backgrond…

Nmap基础入门及常用命令汇总

Nmap基础入门 免责声明&#xff1a;本文单纯分享技术&#xff0c;请大家使用过程中遵守法律法规~ 介绍及安装 nmap是网络扫描和主机检测的工具。作为一个渗透测试人员&#xff0c;必不可少的就是获取信息。那么nmap就是我们从互联网上获取信息的途径&#xff0c;我们可以扫描互…

Excel中一次查询返回多列

使用Excel或wps的时候&#xff0c;有时候需要一次查询返回多列内容&#xff0c;这种情况可以选择多次vlookup或者多次xlookup&#xff0c;但是这种做法费时费力不说&#xff0c;效率还有些低下&#xff0c;特别是要查询的列数过多时。我放了3种查询方法&#xff0c;效果图&…

NodeMCU驱动28BYJ-48型步进电机(Arduino)

NodeMCU NodeMCU开发板 此NodeMCU是在乐鑫公司&#xff08;Espressif Systems&#xff09;生产的ESP-12F模组的基础上封装好的具备WiFi功能的开源IoT开发板。本次选用安信可公司&#xff08;Ai-Thinker&#xff09;生产的CP2102版本的开发板。 28BYJ-48型步进电机 28BYJ-48型…

qt QZipReader详解

1、概述 QZipReader 是 Qt 中用于从 .zip 文件中读取和提取文件内容的类。它提供了便捷的方法来访问压缩包中的文件和目录&#xff0c;并允许你解压缩单个或多个文件。通过 QZipReader&#xff0c;你可以以编程方式读取 .zip 文件中的内容&#xff0c;并提取它们到目标目录中。…

html + css 淘宝网实战

之前有小伙伴说&#xff0c;淘宝那么牛逼你会写代码&#xff0c;能帮我做一个一样的淘宝网站吗&#xff0c;好呀&#xff0c;看我接下来如何给你做一个淘宝首页。hahh,开个玩笑。。。学习而已。 在进行html css编写之前 先了解下网页的组成和网页元素的尺寸吧 1.网页的组成 …

Ollama+OpenWebUI+llama3本地部署

引言 llama3在4月19日刚刚发布&#xff0c;官方的对比结果中在开源模型中堪称世界第一&#xff0c;整好周六日有时间&#xff0c;在魔搭社区上测试一下 2 安装Ollama 2.1 下载Ollama 登录Ollama官网下载Ollama安装包 GitHub&#xff1a;https://github.com/ollama/ollama?t…

【vue2父组件调用子组件方法之slot的使用】

父组件调用子组件方法之slot的使用 具体功能需求&#xff1a; 一个页面&#xff0c;点击按钮&#xff0c;打开一个弹窗。弹窗有自定义表单和公共表单&#xff0c;提交的时候要获取两个表单的数据以及复显表单数据 为什么使用插槽了&#xff0c;因为我需要在弹窗中复用公共表单…