智能指针和所有权

2024-03-27 21:08
文章标签 指针 智能 所有权

本文主要是介绍智能指针和所有权,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

在编程语言中,对堆对象的内存管理是一个麻烦又复杂的问题。一不小心就会带来问题,比如JS里一直引用一个已经不使用的对象导致gc无法回收,或者C++里多个变量指向同一块内存导致重复释放。本文简单探讨一下关于对象所有权的问题。

对象的所有权意味着当我们分配一个对象的时候,谁持有这个对象的所有权,比如下面代码。

Object *obj = new Object();

那么obj就持有了对象的所有权。但是现实往往比较复杂,比如我们看看下面代码。

#include<stdio.h>
using namespace std;
class Demo {public:~Demo(){printf("执行析构函数");}
};
void test() {Demo *d = new Demo();
}
int main()
{test();return 0;
}

执行上面的代码,我们在test函数里分配一个堆对象,执行完test后我们发现Demo对象的析构函数并没有执行,这就造成了内存泄漏。那我们需要怎么做呢?我们需要收到释放对象对应的内存。修改一下test函数的代码。

void test() {Demo *d = new Demo();delete d;
}

这时候我们发现就会输出执行析构函数几个字了,说明析构函数被执行,对象的内存也被释放了。手动管理内存不仅麻烦,而且往往容易出错,比如我们往往会忘了释放,尤其是代码逻辑复杂的时候。这时候,我们可以使用智能指针解决这个问题。

#include <iostream>
#include<stdio.h>
using namespace std;
class Demo {public:~Demo(){printf("执行析构函数");}
};template<class T>
class SmartPoint
{T* point;
public:SmartPoint(T *ptr = nullptr) :point(ptr) {}~SmartPoint() {if (point) {// 会调用point指向对象的的析构函数delete point;}}// 使用智能指针就像使用内部包裹的的对象一样T& operator*() { return *point; }T* operator->() { return point; }
};void test() {SmartPoint<Demo> p(new Demo());
}int main()
{test();return 0;
}

智能指针的原理比较简单,因为智能指针对象是在栈上面分配的,离开作用域的时候会被自动释放,然后在智能指针的析构函数里释放包裹的内部对象。看起来是很完美的解决方案。但是智能指针也带来了一些问题,那就是在复制或赋值的时候。我们看看代码。

int main()
{SmartPoint<Demo> p(new Demo());SmartPoint<Demo> p2 = p;return 0;
}

执行下面代码会导致core dump,为什么呢?我们来看看这个过程。当执行p2=p的时候会导致p2和p的内部指针point都指向了Demo对象的地址,最后代码执行完毕后,两个智能指针都执行了释放内存的操作,重复释放内存导致了core dump。那如何解决这个问题呢?一种方式是复制一份point指向的内存,但是我们可能不知道这个内存多大,无法复制,另一种方式就是所有权转移。我们继续看代码。

#include <iostream>
#include<stdio.h>
using namespace std;
class Demo {public:~Demo(){printf("执行析构函数");}
};template<class T>
class SmartPoint
{T* point;
public:SmartPoint(T *ptr = nullptr) :point(ptr) {}// 实现复制构造函数SmartPoint(SmartPoint & p) { // 指向p.point对应的内存point = p.point;// p.point置nullp.point = nullptr;}~SmartPoint() {if (point) {// 会调用point指向对象的的析构函数delete point;}}// 使用智能指针就像使用内部包裹的的对象一样T& operator*() { return *point; }T* operator->() { return point; }
};int main()
{SmartPoint<Demo> p(new Demo());SmartPoint<Demo> p2 = p;return 0;
}

我们实现了一个复制构造函数,在main里执行p2=p时会被执行,在复制构造函数中,我们实现了所有权转移,这时候p2时Demo对象的持有者,而p指向null,这时候不能再对p进行操作。这时候我们可以在SmartPoint中实现一个isNull函数用于判断智能指针的有效性。

bool isNull() {return point == nullptr; 
}

然后在使用的地方加一下判断。

if (p.isNull()) {// 
}

这显然很麻烦。我们看看Rust怎么做。

struct Demo(u32);fn main() {let _box1 = Box::new(Demo(1));// 所有权转移let _box2 = _box1;// 报错println!("{}", _box1.0);
}

编译上面代码会报错,是编译而不是运行,这就是Rust,在编译期就解决了这个问题。Box是智能指针,以上代码和刚才C++中的代码类似,当执行_box2=box1的时候,堆对象的所有权就转移到了_box2,box1相当于包裹了一个空指针,而Rust不允许你再访问_box1管理里的内存。

这篇关于智能指针和所有权的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/853391

相关文章

基于Python打造一个智能单词管理神器

《基于Python打造一个智能单词管理神器》这篇文章主要为大家详细介绍了如何使用Python打造一个智能单词管理神器,从查询到导出的一站式解决,感兴趣的小伙伴可以跟随小编一起学习一下... 目录1. 项目概述:为什么需要这个工具2. 环境搭建与快速入门2.1 环境要求2.2 首次运行配置3. 核心功能使用指

Python实现word文档内容智能提取以及合成

《Python实现word文档内容智能提取以及合成》这篇文章主要为大家详细介绍了如何使用Python实现从10个左右的docx文档中抽取内容,再调整语言风格后生成新的文档,感兴趣的小伙伴可以了解一下... 目录核心思路技术路径实现步骤阶段一:准备工作阶段二:内容提取 (python 脚本)阶段三:语言风格调

go 指针接收者和值接收者的区别小结

《go指针接收者和值接收者的区别小结》在Go语言中,值接收者和指针接收者是方法定义中的两种接收者类型,本文主要介绍了go指针接收者和值接收者的区别小结,文中通过示例代码介绍的非常详细,需要的朋友们下... 目录go 指针接收者和值接收者的区别易错点辨析go 指针接收者和值接收者的区别指针接收者和值接收者的

Java Optional避免空指针异常的实现

《JavaOptional避免空指针异常的实现》空指针异常一直是困扰开发者的常见问题之一,本文主要介绍了JavaOptional避免空指针异常的实现,帮助开发者编写更健壮、可读性更高的代码,减少因... 目录一、Optional 概述二、Optional 的创建三、Optional 的常用方法四、Optio

使用Python实现表格字段智能去重

《使用Python实现表格字段智能去重》在数据分析和处理过程中,数据清洗是一个至关重要的步骤,其中字段去重是一个常见且关键的任务,下面我们看看如何使用Python进行表格字段智能去重吧... 目录一、引言二、数据重复问题的常见场景与影响三、python在数据清洗中的优势四、基于Python的表格字段智能去重

Spring AI集成DeepSeek三步搞定Java智能应用的详细过程

《SpringAI集成DeepSeek三步搞定Java智能应用的详细过程》本文介绍了如何使用SpringAI集成DeepSeek,一个国内顶尖的多模态大模型,SpringAI提供了一套统一的接口,简... 目录DeepSeek 介绍Spring AI 是什么?Spring AI 的主要功能包括1、环境准备2

Spring AI与DeepSeek实战一之快速打造智能对话应用

《SpringAI与DeepSeek实战一之快速打造智能对话应用》本文详细介绍了如何通过SpringAI框架集成DeepSeek大模型,实现普通对话和流式对话功能,步骤包括申请API-KEY、项目搭... 目录一、概述二、申请DeepSeek的API-KEY三、项目搭建3.1. 开发环境要求3.2. mav

解决java.lang.NullPointerException问题(空指针异常)

《解决java.lang.NullPointerException问题(空指针异常)》本文详细介绍了Java中的NullPointerException异常及其常见原因,包括对象引用为null、数组元... 目录Java.lang.NullPointerException(空指针异常)NullPointer

Python3脚本实现Excel与TXT的智能转换

《Python3脚本实现Excel与TXT的智能转换》在数据处理的日常工作中,我们经常需要将Excel中的结构化数据转换为其他格式,本文将使用Python3实现Excel与TXT的智能转换,需要的可以... 目录场景应用:为什么需要这种转换技术解析:代码实现详解核心代码展示改进点说明实战演练:从Excel到

嵌入式QT开发:构建高效智能的嵌入式系统

摘要: 本文深入探讨了嵌入式 QT 相关的各个方面。从 QT 框架的基础架构和核心概念出发,详细阐述了其在嵌入式环境中的优势与特点。文中分析了嵌入式 QT 的开发环境搭建过程,包括交叉编译工具链的配置等关键步骤。进一步探讨了嵌入式 QT 的界面设计与开发,涵盖了从基本控件的使用到复杂界面布局的构建。同时也深入研究了信号与槽机制在嵌入式系统中的应用,以及嵌入式 QT 与硬件设备的交互,包括输入输出设