C++基本用法实践之智能指针详解

 更新时间:2023年07月17日 11:28:04   作者:寡人正在Coding  
为了减少手动管理内存带来的困扰,c++提出了智能指针,可以帮助我们进行内存管理,下面小编就来和大家简单聊聊C++中智能指针的基本用法吧

概述

c++效率较高的一个原因是我们可以自己定制策略手动申请和释放内存,当然,也伴随着开发效率降低和内存泄漏的风险。为了减少手动管理内存带来的困扰,c++提出了智能指针,可以帮助我们进行内存管理,有三种:

  • std::unique_ptr 是一种独占所有权的智能指针,它不允许多个指针指向同一个对象。std::unique_ptr 是一种轻量级的智能指针,不需要额外的开销,因此在你不需要共享所有权的情况下,应优先使用 std::unique_ptr
  • std::shared_ptr 是一种可以共享所有权的智能指针。多个 std::shared_ptr 可以指向同一个对象,该对象只有在最后一个指向它的 std::shared_ptr 被销毁时才会被删除。std::shared_ptr 使用引用计数来跟踪有多少个智能指针指向同一个对象。
  • std::weak_ptr 是一种弱引用智能指针,它可以指向 std::shared_ptr 所管理的对象,但它不会增加该对象的引用计数。这对于解决 std::shared_ptr 的循环引用问题很有用。

智能指针的类型决定了何时销毁和释放内存。当智能指针的生命周期结束时,它会自动清理其所有权的内存。这就避免了内存泄漏,使代码更安全、更健壮。
但是智能指针的使用,也要注意一些问题:

  • 智能指针并不能自动解决循环引用的问题,需要手动在合适的场合使用std::weak_ptr弱指针
  • 智能指针的使用是“传染性”的,如果使用智能指针,要考虑好整个逻辑链条都使用智能指针而不是原生的指针
  • 从实际项目使用上看,一些比较上层的模块使用智能指针,更加方便,而在底层模块更多的还是使用原生的指针,我想大概因为智能指针也是会带来一些开销而且可能和一些内存管理策略冲突(如某些优化内存策略可能不使用指针,而使用句柄来指向对象)

智能指针是一个对象,实现是非入侵式的。关于他们的大概的原理:

  • std::unique_ptr 它包含一个原生指针,并在其析构函数中删除这个指针。std::unique_ptr还重载了->*运算符,所以可以像使用原生指针一样使用std::unique_ptr
  • std::shared_ptr 的实现则相对复杂一些。除了存储原生指针,std::shared_ptr还需要存储一个引用计数。每次创建一个新的std::shared_ptr或者调用std::shared_ptr的拷贝构造函数或赋值运算符时,引用计数就会增加。每次销毁一个std::shared_ptr时,引用计数就会减少。只有当引用计数降到0时,才会删除原生指针。类似的还要处理若引用。此外,引用计数还要保证线程安全。

用法举例

参考测试项目代码ModernCppTest/modrenc_smart_pointer.cpp主要包含:

  • 共享指针用法&声明方式
  • 独占指针用法&声明方式
  • 弱指针的用法
#include "ModernCppTestHeader.h"
#include <memory>

#define LOG_UNIQUE_PTR_VALID(ptr) if(ptr) LOG(#ptr << " unique_ptr valid"); else LOG(#ptr << " unique_ptr invalid")
#define LOG_WEAK_PTR_IF_VALID(ptr, exp) if(auto tp = p2.lock()) exp else LOG(#ptr << " weak_ptr invalid")

namespace n_smart_pointer {
	class Obj {
	public:
		Obj(int ID) : ID(ID) { LOG("Obj addr " << this << " ID " << ID << " Create"); }
		~Obj() { LOG("Obj addr " << this << " ID " << ID << " Release"); }

		int ID;
	};

	class Item {};
}

using LocaObj = n_smart_pointer::Obj;

void smart_pointer_test()
{
	LOG_FUNC();


	LOG_TAG("std::shared_ptr 共享指针");
	{
		{
			std::shared_ptr<LocaObj> p1 = std::make_shared<LocaObj>(1);
			{
				std::shared_ptr<LocaObj> p2 = p1;
				LOG_VAR_DESC(p1->ID, " p1->ID");
				LOG_VAR_DESC(p2->ID, " p2->ID");
				LOG("p2 离开作用域");
			}
			LOG("p1 离开作用域");
		}
	}


	LOG_TAG("std::shared_ptr 共享指针的声明方式");
	{
		std::shared_ptr<LocaObj> p1(new LocaObj(1));
		std::shared_ptr<LocaObj> p2 = std::make_shared<LocaObj>(2);
		std::shared_ptr<LocaObj> p3(p2);
	}


	LOG_TAG("std::unique_ptr 唯一指针");
	{
		{
			std::unique_ptr<LocaObj> p1;
			{
				std::unique_ptr<LocaObj> p2 = std::make_unique<LocaObj>(1);
				LOG_VAR_DESC(p2->ID, " p2->ID");

				LOG("唯一指针转移控制权使用=运算符报错,须使用std::move()");
				// std::unique_ptr<LocaObj> p1 = p2;
				p1 = std::move(p2);
				LOG_VAR_DESC(p1->ID, " p1->ID");

				LOG("测试被std::move的p2是否还有效,可以看到 无效");
				LOG_UNIQUE_PTR_VALID(p2);
				LOG("p2 离开作用域, ID为1的Obj没有析构");
			}

			LOG("p1 离开作用域, ID为1的Obj析构");
		}
	}


	LOG_TAG("std::unique_ptr 唯一指针的声明方式");
	{
		std::unique_ptr<LocaObj> p1(new LocaObj(1));
		std::unique_ptr<LocaObj> p2 = std::make_unique<LocaObj>(2);
		std::unique_ptr<LocaObj> p3(std::move(p2));
	}


	LOG_TAG("std::weak_ptr 弱指针");
	{
		std::shared_ptr<LocaObj> p1 = std::make_shared<LocaObj>(1);
		std::weak_ptr<LocaObj> p2 = p1;

		LOG_VAR_DESC(p1->ID, " p1->ID");
		LOG_WEAK_PTR_IF_VALID(p2, LOG_VAR_DESC(tp->ID, " p2->ID"););

		LOG("p1 reset Obj对象被释放");
		p1.reset();

		LOG("p2 弱指针失效");
		LOG_WEAK_PTR_IF_VALID(p2, LOG_VAR_DESC(tp->ID, " p2->ID"););
	}
}

到此这篇关于C++基本用法实践之智能指针详解的文章就介绍到这了,更多相关C++智能指针内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • c语言中比较特殊的输入函数举例详解

    c语言中比较特殊的输入函数举例详解

    C语言提供了丰富的标准库函数,用于处理各种输入输出操作,下面这篇文章主要介绍了c语言中比较特殊的输入函数的相关资料,文中通过代码介绍的非常详细,需要的朋友可以参考下
    2024-09-09
  • C++中remove与erase区别小结

    C++中remove与erase区别小结

    remove函数和 erase函数都可以实现元素的删除,本文主要介绍了C++中remove与erase区别小结,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2024-08-08
  • 详解C语言实现推箱子的基本功能

    详解C语言实现推箱子的基本功能

    这篇文章主要为大家详细介绍了C语言实现推箱子的基本功能的方法,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下,希望能够给你带来帮助
    2022-02-02
  • C++ Primer 标准库vector示例详解

    C++ Primer 标准库vector示例详解

    该文章主要介绍了C++标准库中的vector类型,包括其定义、初始化、成员函数以及常见操作,文章详细解释了如何使用vector来存储和操作对象集合,并提供了代码示例来说明vector的使用方法,感兴趣的朋友一起看看吧
    2025-03-03
  • C语言数据结构顺序表的进阶讲解

    C语言数据结构顺序表的进阶讲解

    程序中经常需要将一组数据元素作为整体管理和使用,需要创建这种元素组,用变量记录它们,传进传出函数等。一组数据中包含的元素个数可能发生变化,顺序表则是将元素顺序地存放在一块连续的存储区里,元素间的顺序关系由它们的存储顺序自然表示
    2022-04-04
  • Qt实现高精度定时器

    Qt实现高精度定时器

    这篇文章主要为大家详细介绍了Qt实现高精度定时器,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2022-08-08
  • C++的QT项目打包成独立可执行和发布的exe文件(项目构建过程)

    C++的QT项目打包成独立可执行和发布的exe文件(项目构建过程)

    这篇文章主要介绍了C++的QT项目打包成独立可执行和发布的exe文件(项目构建过程),本文通过实例图文相结合给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2021-11-11
  • Qt使用SQLite数据库实现数据增删改查

    Qt使用SQLite数据库实现数据增删改查

    这篇文章主要为大家详细介绍了Qt如何使用SQLite数据库实现数据增删改查功能,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起了解一下
    2023-06-06
  • C++实现LeetCode(209.最短子数组之和)

    C++实现LeetCode(209.最短子数组之和)

    这篇文章主要介绍了C++实现LeetCode(209.最短子数组之和),本篇文章通过简要的案例,讲解了该项技术的了解与使用,以下就是详细内容,需要的朋友可以参考下
    2021-08-08
  • C语言实现猜数字小项目

    C语言实现猜数字小项目

    这篇文章主要为大家详细介绍了C语实现猜数字小项目,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2022-01-01

最新评论