IT数码 购物 网址 头条 软件 日历 阅读 图书馆
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
图片批量下载器
↓批量下载图片,美女图库↓
图片自动播放器
↓图片自动播放器↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁
 
   -> C++知识库 -> C++智能指针 -> 正文阅读

[C++知识库]C++智能指针

C++智能指针

  • 智能指针本质上是一个模板类,一般使用的这个类的对象,而不是指针
  • 智能指针体现在内存释放问题上,用智能指针管理new的对象,将不在需要手动delete

智能指针的分类

  • shared_ptr
    get()函数:返回数据的指针的引用
    use_count() : 返回管理对象的智能指针个数
    swap() : 交换两个智能指针管理的对象
    reset() : 重置智能指针对象
    针对部分场景需要自己手动添加删除器去释放对象内存

  • weak_ptr
    弱引用指针,不会累计计数
    weak_ptr只能通过shared_ptr或者weak_ptr来构造
    主要应用场景:为了解决shared_ptr循环引用内存导致无法释放问题
    不能使用*访问,可以使用->取值
    通过成员函数lock()获取shared_ptr对象然后在访问数据

  • unique_ptr
    禁止拷贝和赋值 独占型
    任何时候unique_ptr操作管理对象,永远只有一个有效
    可以通过move()函数转交所有权
    reset函数结合release函数移交所有权

以下对于三种类型分别举例:

  • shared_ptr
#include <iostream>
#include <string>
#include <memory>
using namespace std;

class MyPerson {
public:
	MyPerson(int age) :age(age) 
	{ 
		cout << "MyPerson构造函数被调用!" << endl; 
	}
	
	MyPerson()
	{
		cout << "MyPerson无参构造函数被调用" << endl;
	}
	
	~MyPerson() 
	{ 
		cout << "MyPerson析构函数被调用!" << endl; 
	}

	void print()
	{
		cout << "MyPerson的年龄:" << age << endl;
	}
private:
	int age;
};

//这里智能指针类对象作为入参,会发送拷贝构造所以use_count智能指针数会+1
//如果传入引用就不会到导致use_count增加
void printPersonData(shared_ptr<MyPerson> pMyPerson)
{
	pMyPerson->print();
	cout << "管理MyPerson对象的指针数:" << pMyPerson.use_count() << endl;
}

int main()
{
	//针对基本数据类型
	shared_ptr<int> pInt(new int(111));
	//访问数据
	cout << *pInt << endl;
	cout << "使用get()获取数据:" << *pInt.get() << endl;
	//use_count返回对象的指针数
	cout << "管理对象的指针数:" << pInt.use_count() << endl;

	shared_ptr<int> pInt2(pInt); //使用拷贝构造函数
	cout << "管理对象的指针数:" << pInt.use_count() << endl;
	cout << "管理对象的指针数:" << pInt2.use_count() << endl;

	shared_ptr<int> pInt3;
	if (!pInt3)
	{
		cout << "空智能指针对象" << endl;
	}
	pInt3 = pInt2;
	cout << "管理对象的指针数:" << pInt3.use_count() << endl;

	//使用swap交换两个智能指针对象
	shared_ptr<int> aa(new int(11));
	shared_ptr<int> bb(new int(22));
	aa.swap(bb);
	cout << "交换后" << endl;
	cout << "aa=" << *aa << "\tbb=" << *bb << endl;

	bb.reset(new int(123));  //重置
	cout << "*bb=" << *bb << endl;

	//针对自定义数据类型
	//初始化
	shared_ptr<MyPerson> pMyPerson(new MyPerson(10));
    //访问数据
	pMyPerson->print();
	cout << "管理MyPerson对象的指针数:" << pMyPerson.use_count() << endl;
	printPersonData(pMyPerson);
	//shared_ptr智能指针管理的对象会自动释放内存
	system("pause");
	return 0;
}

结果分析:
1、使用shared_ptr智能指针管理自定义对象可以使用->访问对象的成员函数(shared_ptr智能指针类内部实现了重定义operator->)
2、使用shared_ptr智能指针管理的对象会自动释放内存,无需手动释放(针对于new创建内存)
3、使用use_count()返回管理对象的智能指针个数

111
使用get()获取数据:111
管理对象的指针数:1
管理对象的指针数:2
管理对象的指针数:2
空智能指针对象
管理对象的指针数:3
交换后
aa=22   bb=11
*bb=123
MyPerson构造函数被调用!
MyPerson的年龄:10
管理MyPerson对象的指针数:1
MyPerson的年龄:10
管理MyPerson对象的指针数:2

针对如下场景

不支持创建连续内存
shared_ptr<MyPerson> pMyPerson2(new MyPerson(10)[4]);

如果想要实现上面常见则自己需要在创建智能指针对象类时添加删除器(自己手动写释放内存过程)`

//带删除器写法; 自己写释放内存过程
	shared_ptr<MyPerson> pMyPerson2(new MyPerson[4], [](MyPerson* array) {delete[] array; });

  • weak_ptr
#include <iostream>
#include <string>
#include <memory>
using namespace std;

class MyPerson {
public:
	MyPerson(int age) :age(age) 
	{ 
		cout << "MyPerson构造函数被调用!" << endl; 
	}
	
	MyPerson()
	{
		cout << "MyPerson无参构造函数被调用" << endl;
	}
	
	~MyPerson() 
	{ 
		cout << "MyPerson析构函数被调用!" << endl; 
	}

	void print()
	{
		cout << "MyPerson的年龄:" << age << endl;
	}
private:
	int age;
};
 
int main()
{
	//针对自定义数据类型
	//初始化
	shared_ptr<MyPerson> pMyPerson(new MyPerson(10));
    //访问数据
	pMyPerson->print();
	cout << "管理MyPerson对象的指针数:" << pMyPerson.use_count() << endl;

	//weak_ptr初始化只能通过shared_ptr或者weak_ptr
	weak_ptr< MyPerson> pWeakPtr(pMyPerson);
	//通过weak_ptr不会导致引用计数增加
	cout << "管理MyPerson对象的指针数:" << pMyPerson.use_count() << endl;
	//*pWeakPtr.print(); weak_ptr不存在这种访问方式

	//通过lock()获取shared_ptr对象
	pWeakPtr.lock().get()->print();

	system("pause");
	return 0;
}
  • unique_ptr
#include <iostream>
#include <string>
#include <memory>
using namespace std;

class MyPerson {
public:
	MyPerson(int age) :age(age) 
	{ 
		cout << "MyPerson构造函数被调用!" << endl; 
	}
	
	MyPerson()
	{
		cout << "MyPerson无参构造函数被调用" << endl;
	}
	
	~MyPerson() 
	{ 
		cout << "MyPerson析构函数被调用!" << endl; 
	}

	void print()
	{
		cout << "MyPerson的年龄:" << age << endl;
	}
private:
	int age;
};

int main()
{
	//针对自定义数据类型
	//初始化
	unique_ptr<MyPerson> pMyPerson(new MyPerson(111));
	
	//unique_ptr不能拷贝和赋值 如下:
	//unique_ptr<MyPerson> pMyPerson2(pMyPerson);
	unique_ptr<MyPerson> pMyPerson3;
	//pMyPerson3 = pMyPerson;

	//move移交所有权后原来的则不能继续使用
	pMyPerson3 = move(pMyPerson);
	pMyPerson3->print();


	//reset重置对象
	pMyPerson.reset(new MyPerson(20));

	unique_ptr<MyPerson> pMyPerson4;
	pMyPerson4.reset(pMyPerson3.release());
	system("pause");
	return 0;
}
  C++知识库 最新文章
【C++】友元、嵌套类、异常、RTTI、类型转换
通讯录的思路与实现(C语言)
C++PrimerPlus 第七章 函数-C++的编程模块(
Problem C: 算法9-9~9-12:平衡二叉树的基本
MSVC C++ UTF-8编程
C++进阶 多态原理
简单string类c++实现
我的年度总结
【C语言】以深厚地基筑伟岸高楼-基础篇(六
c语言常见错误合集
上一篇文章      下一篇文章      查看所有文章
加:2022-05-08 07:51:17  更:2022-05-08 07:52:52 
 
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁

360图书馆 购物 三丰科技 阅读网 日历 万年历 2025年1日历 -2025/1/11 3:53:54-

图片自动播放器
↓图片自动播放器↓
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
图片批量下载器
↓批量下载图片,美女图库↓
  网站联系: qq:121756557 email:121756557@qq.com  IT数码