【C++】泛型编程 ⑭ ( 类模板示例 - 数组类模板 | 容器思想 | 自定义类可拷贝 - 深拷贝与浅拷贝 | 自定义类可打印 - 左移运算符重载 )

文章目录

  • 一、容器思想
    • 1、自定义类可拷贝 - 深拷贝与浅拷贝
    • 2、自定义类可拷贝 - 代码示例
    • 3、自定义类可打印 - 左移运算符重载
  • 二、代码示例
    • 1、Array.h 头文件
    • 2、Array.cpp 代码文件
    • 3、Test.cpp 主函数代码文件
    • 4、执行结果






一、容器思想




1、自定义类可拷贝 - 深拷贝与浅拷贝


上一篇博客 【C++】泛型编程 ⑬ ( 类模板示例 - 数组类模板 | 构造函数和析构函数 的 声明与实现 | 普通成员函数 的 声明与实现 | 外部友元函数 的 声明与实现 ) 中 , 实现了一个 数组 类模板 , 数组 中的 数据元素 是 泛型类型 , 可以是任意类型 ;

也就是说 , 该数组可以存储 任意类型 的数据 , 包括 自定义类对象 ;

该数组 就是一个 数据的容器 ;

数组中 每个元素 插入数据时 , 其本质是一个 拷贝操作 , 数组 的 内存空间 在 声明实际类型 以及 创建 时 , 就已经确定了 , 向数组中插入元素 , 就是将 已有的 数据 拷贝到 已经分配好的内存中 ;

向 数据容器 ( 数组 ) 中插入的数据 , 必须可以被 拷贝 , 如果 不能被拷贝 , 就会出现插入数据失败的问题 ;


容器 中的 类型 可拷贝 , 就是要求 容器中的 数据类型 都是 值语义 , 不是 引用语义 ,

向 容器 中插入元素 , 就是拷贝 数据内容 到容器中 , 要将真实的值拷贝进去 , 不是将 引用地址 拷贝进去 ,

就是 深拷贝 和 浅拷贝 的问题 ;


下面的示例中 , 自定义类中的成员变量 char m_name[32] 是 在定义时 , 直接分配好的 ,

如果 自定义类 中有 指针类型的成员变量 , 如 char* m_name , 涉及到 动态分配内存 , 如果没有定义 拷贝构造函数 , 默认的 拷贝构造函数 是 浅拷贝 函数 , 直接将 指针地址 简单拷贝 , 这就是 不可被拷贝的情况 ;

那么多个 数组元素 就会共享 相同的 堆内存 数据 , 此时就会出现问题 ;


如果遇到了上述问题 , 定义了 char* m_name 成员变量 , 涉及到 动态分配内存 , 那么 该自定义类 必须自己实现 深拷贝 的 拷贝构造函数 ;


编写的类 , 可以存储到 数组类模板 容器 中 , 那么 该类 必须 支持 拷贝工作 , 具体一些就是 深拷贝 工作 ;


2、自定义类可拷贝 - 代码示例


下面简单实现一个类 , 该类中维护了 2 个成员变量 , char m_name[32] 数组变量 和 int m_age 变量 , 这两个 成员 都是在 创建时 就会分配内存空间 , 不存在 深拷贝问题 ;

如果 char m_name[32] 数组变量 改为 char* m_name 指针变量 , 就需要考虑 深拷贝问题了 ;

class Student
{
public:Student(){m_age = 10;strcpy(m_name, "NULL");}Student(const char* name, int age) {strcpy(this->m_name, name);this->m_age = age;}void printT() {cout << "name : " << m_name << " , age : " << m_age << endl;}private:char m_name[32];int m_age;
};

3、自定义类可打印 - 左移运算符重载


数组类模板 中 , 实现了 左移运算符 打印日志 , 如果 数组中 存储 自定义类对象 想要通过 cout 打印出来 , 那么 该自定义类 必须 进行 左移运算符重载操作 ;


声明 左移运算符重载函数 :

class Student
{friend ostream& operator<<(ostream& out, const Student& s);
}

实现 左移运算符重载函数 :

// 重载左移运算符实现
ostream& operator<<(ostream& out, const Student& s) {out << "name : " << s.m_name << " , age : " << s.m_age << " ; ";return out;
}




二、代码示例




1、Array.h 头文件


#pragma once#include "iostream"
using namespace std;template <typename T>
class Array
{// 左移 << 操作符重载// 注意 声明时 , 需要在 函数名 和 参数列表之间 注明 泛型类型 <T>//		实现时 , 不能在 函数名 和 参数列表之间 注明 泛型类型 <T>friend ostream& operator<< <T> (ostream& out, const Array& a);public:// 有参构造函数Array(int len = 0);// 拷贝构造函数Array(const Array& array);// 析构函数~Array();public:// 数组下标 [] 操作符重载// 数组元素类型是 T 类型T& operator[](int i);// 等号 = 操作符重载Array& operator=(const Array& a);private:// 数组长度int m_length;// 指向数组数据内存 的指针// 指针类型 是 泛型类型 TT* m_space;
};

2、Array.cpp 代码文件


#include "Array.h"// 左移 << 操作符重载
// 注意 声明时 , 需要在 函数名 和 参数列表之间 注明 泛型类型 <T>
//		实现时 , 不能在 函数名 和 参数列表之间 注明 泛型类型 <T>
template <typename T>
ostream& operator<< (ostream& out, const Array<T>& a)
{for (int i = 0; i < a.m_length; i++){// 在一行内输入数据, 使用空格隔开, 不换行out << a.m_space[i] << " ";}// 换行out << endl;return out;
}// 有参构造函数
template <typename T>
Array<T>::Array(int len)
{// 设置数组长度m_length = len;// 为数组在堆内存中分配内存// 注意 元素类型为 Tm_space = new T[m_length];cout << " 调用有参构造函数 " << endl;
}// 拷贝构造函数
// 这是一个深拷贝 拷贝构造函数
template <typename T>
Array<T>::Array(const Array<T>& array)
{// 设置数组长度m_length = array.m_length;// 创建数组// 注意 元素类型为 Tm_space = new T[m_length];// 为数组赋值for (int i = 0; i < m_length; i++){m_space[i] = array.m_space[i];}cout << " 调用拷贝构造函数 " << endl;
}// 析构函数
template <typename T>
Array<T>::~Array()
{if (m_space != NULL){// 释放 new T[m_length] 分配的内存 delete[] m_space;m_space = NULL;m_length = 0;}cout << " 调用析构函数 " << endl;
}// 数组下标 [] 操作符重载
template <typename T>
T& Array<T>::operator[](int i)
{return m_space[i];
}// 等号 = 操作符重载
template <typename T>
Array<T>& Array<T>::operator=(const Array<T>& a)
{if (this->m_space != NULL){// 释放 new int[m_length] 分配的内存 delete[] this->m_space;this->m_space = NULL;}// 设置数组长度this->m_length = a.m_length;// 创建数组this->m_space = new T[m_length];// 为数组赋值for (int i = 0; i < m_length; i++){this->m_space[i] = a.m_space[i];}cout << " 调用 等号 = 操作符重载 函数" << endl;// 返回是引用类型// 返回引用就是返回本身// 将 this 指针解引用, 即可获取数组本身return *this;
}

3、Test.cpp 主函数代码文件


#define _CRT_SECURE_NO_WARNINGS
#include "iostream"
using namespace std; // 此处注意, 类模板 声明与实现 分开编写
// 由于有 二次编译 导致 导入 .h 头文件 类模板函数声明 无法找到 函数实现
// 必须 导入 cpp 文件
#include "Array.cpp"class Student
{friend ostream& operator<<(ostream& out, const Student& s);
public:Student(){m_age = 10;strcpy(m_name, "NULL");}Student(const char* name, int age) {strcpy(this->m_name, name);this->m_age = age;}void printT() {cout << "name : " << m_name << " , age : " << m_age << endl;}private:char m_name[32];int m_age;
};// 重载左移运算符实现
ostream& operator<<(ostream& out, const Student& s) {out << "name : " << s.m_name << " , age : " << s.m_age << " ; ";return out;
}int main() {// 验证 有参构造函数Array<Student> array(3);Student s0("Tom", 18), s1("Jerry", 12), s2("Jack", 16);array[0] = s0;array[1] = s1;array[2] = s2;// 遍历数组 打印数组元素for (int i = 0; i < 3; i++) {array[i].printT();}cout << array << endl;// 控制台暂停 , 按任意键继续向后执行system("pause");return 0;
}

4、执行结果


执行结果 :

 调用有参构造函数
name : Tom , age : 18
name : Jerry , age : 12
name : Jack , age : 16
name : Tom , age : 18 ;  name : Jerry , age : 12 ;  name : Jack , age : 16 ;Press any key to continue . . .

在这里插入图片描述

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

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

相关文章

百战python02-语言元素

文章目录 指令与程序变量与类型变量命名变量的使用运算符赋值运算符比较运算符和逻辑运算符练习1:华氏温度转换为摄氏温度练习2:输入圆的半径计算计算周长和面积练习3:输入年份判断是不是闰年字符串常用操作注:需要对python有基本了解,可查看本作者python基础专栏,有任何问…

大模型生态新篇章:以AI Agent为引,助企业创新应用落地

文 | 智能相对论 作者 | 沈浪 以聊天机器人、虚拟助手、智能客服等为代表的对话式人工智能 (Conversational AI Agents ) 在具体服务场景中的应用已经十分普遍。今年以来&#xff0c;随着大模型技术的爆发与加持&#xff0c;对话式AI被市场赋予了更高的期望。 “所有行业都值…

Spring 事务失效的7种场景, 事务失效后如何进行处理

文章目录 简单说说spring事务失效的场景Spring 事务失效的7种场景1.1、未启用[spring事务管理](https://so.csdn.net/so/search?qspring事务管理&spm1001.2101.3001.7020)功能1.2、方法不是public类型的1.3、数据源未配置事务管理器1.4、自身调用问题1.5、异常类型错误1.6…

《golang设计模式》第三部分·行为型模式-07-观察者模式(Observer)/发布者—订阅者模式

文章目录 1. 概念1.1 角色1.2 类图 2. 代码示例2.1 代码2.2 类图 1. 概念 观察者&#xff08;Observer&#xff09;指当目标对象状态发生变化后&#xff0c;对状态变化事件进行响应或处理的对象。 1.1 角色 Subject&#xff08;抽象主题&#xff09;&#xff1a; 它可以有多…

微服务实战系列之Feign

前言 不知不觉&#xff0c;“微服务实战系列”已完成了六篇&#xff0c;每篇都聚焦一个主题&#xff0c;目的是便于各位盆友能够快速、全面地接收和消化。 博主从服务注册到服务监控&#xff0c;从服务路由到服务安全&#xff0c;从身份认证到加密技术均有涉猎。凡此均有关微服…

Java核心知识点整理大全10-笔记

往期快速传送门&#xff1a; Java核心知识点整理大全-笔记_希斯奎的博客-CSDN博客文章浏览阅读9w次&#xff0c;点赞7次&#xff0c;收藏7次。Java核心知识点整理大全https://blog.csdn.net/lzy302810/article/details/132202699?spm1001.2014.3001.5501 Java核心知识点整理…

【LeetCode刷题】--67.二进制求和

67.二进制求和 方法&#xff1a;模拟计算 class Solution {public String addBinary(String a, String b) {StringBuilder ans new StringBuilder();int carry 0;for(int ia.length()-1,jb.length()-1;i>0||j>0;i--,j--){int sum carry;sum i >0 ? a.charAt(i) …

web:[WUSTCTF2020]朴实无华

题目 点开页面显示如下 页面显示了一行报错&#xff1a;Cannot modify header information - headers already sent by (output started at /var/www/html/index.php:3) in /var/www/html/index.php on line 4 意思为不能修改报头信息-报头已经发送(输出开始于/var/www/html/i…

vue3 websocket连接 发送数据

先建一个websocket.js放在项目中&#xff0c;内容如下&#xff1a; var websock null; let rec; //断线重连后&#xff0c;延迟5秒重新创建WebSocket连接 rec用来存储延迟请求的代码 let isConnect false; //连接标识 避免重复连接 let checkMsg "heartbeat"; /…

MySQL与Redis如何保证数据的一致性

文章目录 MySQL与Redis如何保证数据的一致性&#xff1f;不好的方案1. 先写 MySQL&#xff0c;再写 Redis2. 先写 Redis&#xff0c;再写 MySQL3. 先删除 Redis&#xff0c;再写 MySQL 好的方案4. 先删除 Redis&#xff0c;再写 MySQL&#xff0c;再删除 Redis5. 先写 MySQL&am…

C# 数据库封装

最近有些地方用到c#&#xff0c;研究了一下&#xff0c;也有数据库方面的操作&#xff0c;那就继续封装&#xff0c;自己用起来好用一点。 数据库连接 using System; using System.Collections.Generic; using System.Data; using System.Linq; using System.Text; using Sys…

Leetcode算法系列| 1. 两数之和(四种解法)

目录 1.题目2.题解解法一&#xff1a;暴力枚举解法二&#xff1a;哈希表解法解法三&#xff1a;双指针(有序状态)解法四&#xff1a;二分查找(有序状态) 1.题目 给定一个整数数组 nums 和一个整数目标值 target&#xff0c;请你在该数组中找出 和为目标值 target 的那 两个 整数…

『RabbitMQ』入门指南(安装,配置,应用)

前言 RabbitMQ 是在 AMQP&#xff08;Advanced Message Queuing Protocol&#xff09; 协议标准基础上完整的&#xff0c;可复用的企业消息系统。它遵循 Mozilla Public License 开源协议&#xff0c;采用 Erlang 实现的工业级的消息队列(MQ)服务器&#xff0c;建立在 Erlang …

产品经理提问常用的ChatGPT通用提示词模板

如何评估产品市场的潜力和可行性&#xff1f; 如何定义和明确产品的目标用户和需求&#xff1f; 如何进行竞品分析和比较&#xff0c;制定产品的差异化策略&#xff1f; 如何设计产品的功能和特性&#xff0c;满足用户需求&#xff1f; 如何制定产品的定价策略和销售计划&a…

qml View3D使用介绍

在Qt Quick 3D中,View3D 是一个用于展示 3D 内容的 QML 类型。View3D 允许你将 3D 场景集成到 Qt Quick 2D 用户界面中,这意味着你可以在传统的 2D UI 元素(如按钮、文本和图像)与 3D 图形之间无缝地进行整合。 View3D 提供了一个视口,用于渲染 3D 场景。它可以包括多个 …

HTTPS攻击怎么防御?

HTTPS 简介 超文本传输安全协议&#xff08; HTTPS &#xff09;是一种通过计算机网络进行安全通信的传输协议。HTTPS 经由 HTTP 进行通信&#xff0c;但利用 SSL/TLS 来加密数据包。 HTTPS 开发的主要目的&#xff0c;是提供对网站服务器的身份认证&#xff0c;保护交换数据的…

批量将本地N个英文Html文档进行中文翻译-操作篇

Unity3D特效百例案例项目实战源码Android-Unity实战问题汇总游戏脚本-辅助自动化Android控件全解手册再战Android系列Scratch编程案例软考全系列Unity3D学习专栏蓝桥系列ChatGPT和AIGC &#x1f449;关于作者 专注于Android/Unity和各种游戏开发技巧&#xff0c;以及各种资源分…

QtCreator9.02不支持JDK11解决

最终效果 使用Android Studio 下载Android SDK Platform 31与Sources for Android 31 下载Android SDK Build Tools 31.0.0 下载NDK 25.1 ,23.1 ,21.3 重要: 下载Android SDK Command-Line Tools ,选择10.0或者9.0其中一个版本 其它版本不支持JDK11 ,本例选择10.0 下载CMak…

如何进行MySQL的主从复制(MySQL5.7)

背景&#xff1a;在一些Web服务器开发中&#xff0c;系统用户在进行数据访问时&#xff0c;基本都是直接操作数据库MySQL进行访问&#xff0c;而这种情况下&#xff0c;若只有一台MySQL服务器&#xff0c;可能会存在如下问题 数据的读和写的所有压力都会由一台数据库独…

浅析jdk8所包含的主要特性

至今Java 8仍然是许多开发者首选的JDK版本&#xff0c;Java 8的生态系统非常成熟&#xff0c;许多库和框架都已经适配了Java 8。迁移到新的Java版本可能需要重新评估和调整现有的依赖关系&#xff0c;这对于一些大型项目可能是一个挑战。那么Java 8有哪些特性让多数开发者钟爱呢…