c/c++ 深拷贝和浅拷贝

深拷贝与浅拷贝

深拷贝(Deep Copy)和浅拷贝(Shallow Copy)是对象复制的两种不同方式,它们涉及到对象成员数据的复制方式和内存管理。

  • 浅拷贝(Shallow Copy):
    浅拷贝是指将一个对象的成员值简单地复制到另一个对象,包括指针类型的成员变量。浅拷贝只复制指针的值,而不会为指针指向的数据分配新的内存空间。这意味着原始对象和副本对象将共享同一份数据。如果其中一个对象修改了共享的数据,将影响到其他对象。
    在这里插入图片描述

  • 深拷贝(Deep Copy):
    深拷贝是指复制对象的每个成员,包括指针类型的成员变量。深拷贝会为每个指针类型的成员变量分配独立的内存空间,并将原始对象指针指向的数据复制到新分配的内存中。这样,原始对象和副本对象拥有各自独立的数据副本,对其中一个对象的修改不会影响到其他对象。
    在这里插入图片描述

  • 深拷贝和浅拷贝的选择取决于对象的成员数据类型和应用场景。对于简单的数据类型或者不涉及动态内存分配的对象,浅拷贝可能足够满足需求。但对于包含指针类型成员或动态分配内存的对象,通常需要使用深拷贝来确保对象之间的独立性。

  • 在C++中,可以通过自定义拷贝构造函数、赋值运算符重载或使用智能指针等方式来实现深拷贝。C语言中需要手动进行内存分配和复制操作,通过手动分配内存、复制数据并管理内存释放来实现深拷贝。

c语言深拷贝和浅拷贝

在C 语言中,没有像 C++ 中那样提供内置的深拷贝和浅拷贝的机制。在 C 语言中,需要手动编写代码来实现深拷贝和浅拷贝。

  • 浅拷贝(Shallow Copy):
    在 C 语言中,浅拷贝可以通过简单地将一个结构体或数组的成员值赋值给另一个结构体或数组来实现。这将导致两个对象共享相同的内存数据,包括指针类型的成员变量。如果其中一个对象修改了共享的数据,将影响到其他对象。
#include <stdio.h>
#include <stdlib.h>
#include <string.h>typedef struct {char* name;int age;
} Person;int main() {Person person1;person1.name = malloc(sizeof(char) * 10);strcpy(person1.name, "John");person1.age = 25;Person person2 = person1;  // 浅拷贝// 修改 person2 的值person2.name = malloc(sizeof(char) * 10);strcpy(person2.name, "Alice");person2.age = 30;// 输出 person1 和 person2 的值printf("person1: name=%s, age=%d\n", person1.name, person1.age);printf("person2: name=%s, age=%d\n", person2.name, person2.age);// 释放内存free(person1.name);free(person2.name);return 0;
}

在上述示例中,通过将 person1 的成员值赋值给 person2,实现了浅拷贝。由于两个对象共享相同的内存数据,对其中一个对象的修改会影响另一个对象。

  • 深拷贝(Deep Copy):
    在 C 语言中,深拷贝需要手动进行内存分配和数据复制。对于包含指针类型成员变量的结构体或数组,需要为每个指针类型的成员变量分配独立的内存空间,并将原始对象指针指向的数据复制到新分配的内存中。
#include <stdio.h>
#include <stdlib.h>
#include <string.h>typedef struct {char* name;int age;
} Person;// 深拷贝的函数
void deepCopyPerson(Person* dest, const Person* src) {dest->name = malloc(sizeof(char) * (strlen(src->name) + 1));strcpy(dest->name, src->name);dest->age = src->age;
}int main() {Person person1;person1.name = malloc(sizeof(char) * 10);strcpy(person1.name, "John");person1.age = 25;Person person2;deepCopyPerson(&person2, &person1);  // 深拷贝// 修改 person2 的值free(person2.name);person2.name = malloc(sizeof(char) * 10);strcpy(person2.name, "Alice");person2.age = 30;// 输出 person1 和 person2 的值printf("person1: name=%s, age=%d\n", person1.name, person1.age);printf("person2: name=%s, age=%d\n", person2.name, person2.age);// 释放内存free(person1.name);free(person2.name);return 0;
}

在上述示例中,通过自定义的函数 deepCopyPerson,实现了深拷贝。该函数手动为 person2 的成员变量分配独立的内存,并将原始对象的数据复制到新分配的内存中。这样,原始对象和副本对象拥有各自独立的数据副本。在使用完后,还需要手动释放内存。

c++深拷贝和浅拷贝

在 C++ 中,可以通过拷贝构造函数赋值运算符来实现深拷贝和浅拷贝。

  • 浅拷贝(Shallow Copy):
    (1)默认情况下,C++ 提供的拷贝构造函数和赋值运算符执行的是浅拷贝操作,即简单地复制成员变量的值。
    (2)对于指针类型的成员变量,直接复制指针的值,这样两个对象将共享相同的内存数据。
class ShallowCopy {
public:ShallowCopy(int* data, int size) {this->data = data;this->size = size;}// 默认拷贝构造函数执行浅拷贝ShallowCopy(const ShallowCopy& other) {data = other.data;  // 复制指针的值size = other.size;}// 默认赋值运算符执行浅拷贝ShallowCopy& operator=(const ShallowCopy& other) {if (this != &other) {data = other.data;  // 复制指针的值size = other.size;}return *this;}private:int* data;int size;
};

在上述示例中,ShallowCopy 类具有一个指针类型的成员变量 data 和一个整数 size。默认的拷贝构造函数和赋值运算符执行浅拷贝,直接复制指针的值,这样两个对象将共享相同的数据。

  • 深拷贝(Deep Copy):
    (1)在自定义拷贝构造函数和赋值运算符时,先为新对象分配独立的内存空间。
    (2)将原始对象的数据复制到新的内存空间中,确保每个对象拥有独立的数据副本。
class DeepCopy {
public:DeepCopy(int* data, int size) {this->data = new int[size];for (int i = 0; i < size; ++i) {this->data[i] = data[i];}this->size = size;}// 自定义拷贝构造函数执行深拷贝DeepCopy(const DeepCopy& other) {data = new int[other.size];for (int i = 0; i < other.size; ++i) {data[i] = other.data[i];}size = other.size;}// 自定义赋值运算符执行深拷贝DeepCopy& operator=(const DeepCopy& other) {if (this != &other) {delete[] data;data = new int[other.size];for (int i = 0; i < other.size; ++i) {data[i] = other.data[i];}size = other.size;}return *this;}~DeepCopy() {delete[] data;}private:int* data;int size;
};

在上述示例中,DeepCopy 类具有一个指针类型的成员变量 data 和一个整数 size。自定义的拷贝构造函数和赋值运算符执行深拷贝,为新对象分配独立的内存空间,并将原始对象的数据复制到新的内存空间中,确保每个对象拥有独立的数据副本。

需要注意的是,在自定义拷贝构造函数和赋值运算符时,还需要注意内存管理,确保在对象销毁时释放已分配的内存,以避免内存泄漏。在示例中,通过在析构函数中释放 data 数组所占用的内存。

参考

  • AI。

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

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

相关文章

C++:继承:面向对象编程的重要特性

(❁◡❁)(●◡●)╰(*▽*)╯(*/ω&#xff3c;*)(^///^)(❁◡❁)(❁◡❁)(●◡●)╰(*▽*)╯(*/ω&#xff3c;*)(❁◡❁)(●’◡’●)╰(▽)╯(/ω&#xff3c;)(///) C&#xff1a;继承&#xff1a;面向对象编程的重要特性 前言**继承**1.继承的概念及定义1.1继承的概念1.2继…

算法笔记p251队列循环队列

目录 队列循环队列循环队列的定义初始化判空判满入队出队获取队列内元素的个数取队首元素取队尾元素 队列 队列是一种先进先出的数据结构&#xff0c;总是从队尾加入元素&#xff0c;从队首移除元素&#xff0c;满足先进先出的原则。队列的常用操作包括获取队列内元素的个数&a…

【Godot4.2】颜色完全使用手册

概述 本篇简单汇总Godot中的颜色的构造和使用&#xff0c;内容包括了&#xff1a; RGB、RGBA&#xff0c;HSV以及HTML16进制颜色值、颜色常量等形式构造颜色颜色的运算以及取反、插值用类型化数组、紧缩数组或PNG图片形式存储多个颜色 构造颜色 因为颜色是一种视觉元素&…

Jmeter接口测试步骤

一、使用工具测试 1、使用Jmeter对接口测试 首先我们说一下为什么用Posman测试后我们还要用Jmeter做接口测试&#xff0c;在用posman测试时候会发现的是一个接口一个接口的测试&#xff0c;我们每次测试成功后的数据&#xff0c;在工具中是无法保存的&#xff0c;再次测试的时…

【蓝桥杯入门记录】继电器、蜂鸣器及原理图分析

一、继电器、继电器概述 &#xff08;1&#xff09;蜂鸣器原理 蜂鸣器的发声原理由振动装置和谐振装置组成&#xff0c;而蜂鸣器又分为无源他激型与有源自激型&#xff0c;蜂鸣器的发声原理为: 1、无源他激型蜂鸣器的工作发声原理是&#xff1a;方波信号输入谐振装置转换为声…

day11【网络编程】

day11【网络编程】 1.4 网络编程三要素 协议 **协议&#xff1a;**计算机网络通信必须遵守的规则&#xff0c;已经介绍过了&#xff0c;不再赘述。 IP地址 IP地址&#xff1a;指互联网协议地址&#xff08;Internet Protocol Address&#xff09;&#xff0c;俗称IP。IP地…

Vue.js开发基础

单文件组件 使用Vite创建Vue项目后&#xff0c;目录结构中包含一些扩展名为.vue的文件&#xff0c;每个.vue文件都可用来定义一个单文件组件。Vue中的单文件组件是Vue组件的文件格式。每个单文件组件由模板、样式和逻辑3个部分构成。 运行结果&#xff1a; 1.模板 模板用于搭建…

每日OJ题_字符串②_力扣5. 最长回文子串(中心拓展)

目录 力扣5. 最长回文子串 解析代码&#xff08;中心拓展&#xff09; 力扣5. 最长回文子串 5. 最长回文子串 难度 中等 给你一个字符串 s&#xff0c;找到 s 中最长的回文子串。 如果字符串的反序与原始字符串相同&#xff0c;则该字符串称为回文字符串。 示例 1&#…

latex变量上下加自适应长度箭头

latex变量上下加自适应长度箭头 在变量上加箭头的代码如下 % 在上 \overrightarrow{A B} \overleftarrow{A B} % 在下 \underrightarrow{A B} \underleftarrow{A B}得出的结果依次如下 A B → , A B ← , A B → , A B ← \overrightarrow{A B},\overleftarrow{A B}, \under…

探索并发编程:深入理解线程池

文章目录 前言一、线程池是什么&#xff1f;二、如何创建线程池1.使用Executors类2.使用ThreadPoolExecutor类手动配置线程池 总结 前言 随着计算机系统的不断发展和进步&#xff0c;我们需要处理更多的并发任务和复杂的操作。而线程池作为一种高效的线程管理机制&#xff0c;…

PyTorch 深度学习(GPT 重译)(一)

第一部分&#xff1a;PyTorch 核心 欢迎来到本书的第一部分。在这里&#xff0c;我们将与 PyTorch 迈出第一步&#xff0c;获得理解其结构和解决 PyTorch 项目机制所需的基本技能。 在第一章中&#xff0c;我们将首次接触 PyTorch&#xff0c;了解它是什么&#xff0c;解决了…

爬虫基础:HTTP基本原理

爬虫基础&#xff1a;HTTP基本原理 前言HTTP基本原理URI 和 URLHTTP 和 HTTPSHTTP 请求过程请求与响应HTTP请求HTTP响应请求与响应的交互过程 HTTP 2.0二进制传输多路复用Header压缩服务器端提前响应内容安全 前言 了解 HTTP的基本原理&#xff0c;了解从往测览器中输人 URL到获…

设计模式总结(四)

上一篇总结了设计模式的结构型&#xff0c; 接下来总结一下设计模式的几种行为型模式。 模板方法模式&#xff1a; 模板方法模式定义了一个算法的骨架&#xff0c;将一些步骤延迟到子类中实现。这样做的目的是使得子类可以在不改变算法结构的情况下重新定义算法的某些步骤&…

「连载」边缘计算(三十二)03-14:边缘部分源码(源码分析篇)

&#xff08;接上篇&#xff09; EdgeCore之edgemesh 前面对EdgeCore组件的edged、devicetwin、edgehub、eventbus、metamanager功能模块进行了分析&#xff0c;本节对EdgeCore组件的另一个功能模块edgemesh进行剖析。目前&#xff0c;KubeEdge官网没有edgemesh相关介绍&…

Visio 2021专业版密钥下载安装

Visio 2021 专业版是微软公司开发的一款功能强大的流程图制作软件&#xff0c;适用于创建各种类型的图表和流程图&#xff0c;包括&#xff1a; 流程图&#xff1a;用于可视化业务流程、工作流程和系统流程。组织结构图&#xff1a;用于展示组织结构和人员关系。网络图&#x…

算法沉淀——贪心算法四(leetcode真题剖析)

算法沉淀——贪心算法四 01.最长回文串02.增减字符串匹配03.分发饼干04.最优除法 01.最长回文串 题目链接&#xff1a;https://leetcode.cn/problems/longest-palindrome/ 给定一个包含大写字母和小写字母的字符串 s &#xff0c;返回 通过这些字母构造成的 最长的回文串 。 …

YOLOV5 改进:修改网络结构--C2f 模块

1、前言 YOLOV5 采用C3模块,类似于残差结构的思想 class C3(nn.Module):# CSP Bottleneck with 3 convolutionsdef __init__(self, c1, c2, n=1, shortcut=True, g=1, e=0.5): # ch_in, ch_out, number, shortcut, groups, expansionsuper().__init__()c_ = int(c2 * e) #…

一篇文章搞懂vue基础(上)

01._初识vue 初识vue: 1.想让Vue工作&#xff0c;就必须创建一个vue实例&#xff0c;且要传入一个配置对象&#xff1b; 2.root容器里的代码依然符合html规范&#xff0c;只不过混入了一些特殊的vue语法 3.root容器里的代码被称为【Vue模版】 4.Vue实例和容器是一一对应的 5.真…

《1w实盘and大盘基金预测 day8》

昨日预测还行&#xff0c;大盘最低到3054缺口附近&#xff0c;并没有补缺口 &#xff0c;顶住了&#xff0c;震荡向上&#xff0c;评分B。 今天行情是指数涨却赚不到钱&#xff0c;像极了拉指数掩护蓝筹股出货 公众号&#xff1a;JavaHelmet 昨天预测&#xff1a; 3042-305…

大数据开发(Hadoop面试真题)

大数据开发&#xff08;Hadoop面试真题&#xff09; 一、HDFSⅠ、HDFS基础概念和架构1、HDFS的作用2、介绍下HDFS&#xff0c;说下HDFS优缺点&#xff0c;以及使用场景&#xff1f;3、请解释一下Hadoop中NameNode和DataNode的作用。4、请解释一下HDFS架构中NameNode和DataNode之…