音视频学习笔记——C++智能指针

C++智能指针介绍

  • 智能指针主要用于管理在堆上分配的内存,它将普通的指针封装为一个栈对象。当栈对象的生存周期结束后,会在析构函数中释放掉申请的内存,从而防止内存泄漏。
  • C++ 11中最常用的智能指针类型为shared_ptr,它采用引用计数的方法,记录当前内存资源被多少个智能指针引用。该引用计数的内存在==堆上分配。当新增一个时引用计数加1,当过期时引用计数减1。只有引用计数为0时,智能指针才会自动释放引用的内存资源。
  • shared_ptr进行初始化时不能将一个普通指针直接赋值给智能指针,因为一个是指针,一个是类。可以通过make_shared函数或者通过构造函数传入普通指针。并可以通过get函数获得普通指针。

二、为什么要使用智能指针

智能指针的作用是管理一个指针,因为存在以下这种情况:申请的空间在函数结束时忘记释放,造成内存泄漏。

使用智能指针可以很大程度上的避免这个问题,因为智能指针就是一个类,当超出了类的作用域时,类会自动调用析构函数,析构函数会自动释放资源。所以智能指针的作用原理就是在函数结束时自动释放内存空间,不需要手动释放内存空间。

三、三种智能指针

C++11 中引入了智能指针,它利用了一种叫做 RAII(资源获取即初始化)的技术将普通的指针封装为一个栈对象。当栈对象的生存周期结束后,会在析构函数中释放掉申请的内存,从而防止内存泄漏。这使得智能指针实质是一个对象,行为表现的却像一个指针。智能指针主要分为shared_ptr unique_ptr weak_ptr 三种,使用时需要引用头文件<memory>

C++11 中shared_ptr 和weak_ptr 都是参考boost 库实现的。

3.1 unique_ptr

unique_ptr实现独占式拥有或严格拥有概念,保证同一时间内只有一个智能指针可以指向该对象。采用所有权模式。

当程序试图将一个unique_ptr赋值给另一个时,如果源unique_ptr是个临时右值,编译器允许这么做;如果源 unique_ptr 将存在一段时间,编译器将禁止这么做,比如:

cpp 复制代码
	unique_ptr<string> pu1(new string ("hello world"));
	unique_ptr<string> pu2;
	pu2 = pu1; // #1 不允许
	unique_ptr<string> pu3;
	pu3 = unique_ptr<string>(new string ("You")); // #2 允许

3.2 shared_ptr

shared_ptr实现共享式拥有概念。多个智能指针可以指向相同对象,该对象和其相关资源会在"最后一个引用被销毁"时候释放。

使用计数机制表明资源被几个指针共享。可以通过成员函数use_count()来查看资源的所有者个数。

成员函数:

use_count :返回引用计数的个数
unique :返回是否是独占所有权(use_count 为 1)
swap :交换两个hared_ptr 对象(即交换所拥有的对象)
reset :放弃内部对象的所有权或拥有对象的变更, 会引起原有对象的引用计数的减少
get :返回内部对象(指针), 由于已经重载了()方法, 因此和直接使用对象是一样的

例子:

cpp 复制代码
	#include <iostream>
	#include "string"
	using namespace std;
	int main()
	{
		string *s1 = new string("s3");
		shared_ptr<string> ps1(s1);
		shared_ptr<string> ps2;
		ps2 = ps1;
		cout << ps1.use_count() << endl; //2
		cout << ps2.use_count() << endl; //2,ps2和ps1共用
		cout << ps1.unique() << endl; //0
	
		string* s3 = new string("s3");
		shared_ptr<string> ps3(s3);
		cout << (ps1.get()) << endl; //033AEB48
		cout << ps3.get() << endl; //033B2C50
		swap(ps1, ps3); //交换所拥
		cout << (ps1.get()) << endl; //033B2C50
		cout << ps3.get() << endl; //033AEB48
		cout << ps3.use_count() << endl; //2 ps3和ps1交换所拥有的对象
		cout << ps1.use_count() << endl; //1  
		cout << ps2.use_count() << endl; //2
		ps2 = ps1;
		cout << ps1.use_count() << endl; //2
		cout << ps2.use_count() << endl; //2
		ps1.reset(); //放弃ps1的拥有权,引用计数的减少
		cout << ps1.use_count() << endl; //0
		cout << ps2.use_count() << endl; //1
	}

运行结果:

share_ptr智能指针也是会发生内存泄露

当两个对象相互使用一个shared_ptr成员变量指向对方,会造成循环引用,使引用计数失效,从而导致内存泄漏。这种情况需要通过智能指针weak_ptr解决。

3.3 weak_ptr

weak_ptr 是一种不控制对象生命周期的智能指针, 它指向一个shared_ptr管理的对象。进行该对象的内存管理的是那个强引用的shared_ptrweak_ptr只是提供了对管理对象的一个访问手段。
设计目的 :为配合 shared_ptr 而引入的一种智能指针来协助 shared_ptr 工作, 只能从一个 shared_ptr 或另一个weak_ptr对象构造, 其构造和析构不会引起引用记数的增加或减少。
weak_ptr是用来解决shared_ptr相互引用时的死锁问题。是对对象的一种弱引用,不会增加对象的引用计数,和shared_ptr之间可以相互转化,shared_ptr可以直接赋值给它,它可以通过调用lock函数来获得shared_ptr

weak_ptr 没有重载*->但可以使用lock 获得一个可用的 shared_ptr 对象
expired 用于检测所管理的对象是否已经释放, 如果已经释放, 返回 true; 否则返回false
lock 用于获取所管理的对象的强引用(shared_ptr). 如果 expiredtrue, 返回一个空的shared_ptr; 否则返回一个 shared_ptr, 其内部对象指向与 weak_ptr 相同.
weak_ptr常用操作

cpp 复制代码
	weak_ptr<T> w; // 空weak_ptr可以指向类型为T的对象
	weak_ptr<T> w(shared_ptr p); // 与p指向相同对象的weak_ptr, T必须能转换为sp指向
	w = p; // p可以是shared_ptr或者weak_ptr,赋值后w和p共享对象
	w.reset(); // weak_ptr置为空
	w.use_count(); // 与w共享对象的shared_ptr的计数
	w.expired(); // w.use_count()为0则返回true,否则返回false
	w.lock(); // w.expired()为true,返回空的shared_ptr;否则返回指向w的shared_ptr
相关推荐
HackKong1 分钟前
高校网络安全_网络安全之道
java·网络·c++·python·学习·web安全·黑客技术
annesede13 分钟前
计算机操作系统与安全复习笔记
笔记
BUG制造机.16 分钟前
修炼之道 ---其四
linux·c++
滴_咕噜咕噜18 分钟前
学习笔记 --C#基础其他知识点(持续更新)
笔记·学习·c#
cdprinter44 分钟前
涉密行业跨网数据摆渡,光盘审计刻录输出,生产音视频刻录,电子档案长期保存应用
安全·自动化·音视频
素**颜1 小时前
uniapp 基于xgplayer(西瓜视频) + renderjs开发,实现APP视频播放
javascript·uni-app·音视频
畅联云平台1 小时前
美畅物联丨如何在视频汇聚平台上添加RTMP主动推流设备?
网络协议·音视频
随风LL1 小时前
AE/PR智能视频变速补帧插帧慢动作插件 Aescripts SpeedX v1.2.0.1 Win/Mac
音视频
Michael_36121 小时前
win系统B站播放8k视频启用HEVC编码
音视频
cuijiecheng20181 小时前
音视频入门基础:MPEG2-TS专题(22)——FFmpeg源码中,获取TS流的音频信息的实现
ffmpeg·音视频