亲宝软件园·资讯

展开

C/C++中智能指针的用法详解

余识- 人气:0

前言

本章主要介绍一些C/C++中智能指针的实现原理以及如何使用

一、什么是智能指针

C/C++中,指针是一个非常重要的概念,其强大但也麻烦

麻烦之处就在于一旦你申请了内存,那就必须要手动去释放内容,否则就会造成内存泄漏

当然了,在代码量少的情况下你可能会不以为意,因为这点内存即使泄露了也根本看不出来,而且一旦程序执行结束,所有内存都会被系统释放

但如果一旦写比较大点的项目,内存管理就显得很重要了,比如QQ,微信等等,一般都是一直挂着的

如果挂几个小时就把电脑内存耗干净了,瞬间电脑变卡,谁还用啊

所以智能指针的作用就是防止我们麻痹大意忘记释放内存,帮助我们管理内存的

当然也有多次释放一个指针,导致程序崩溃的问题也能就此解决

二、使用方法

虽然智能指针听着很高级,但使用起来并不算复杂,熟悉之后,其实和普通指针差别不大。但会更加好用

自C++11之后,智能指针共有三个:shared_ptr、unique_ptr、weak_ptr

1.shared_ptr

看名字就知道,它是可以分享的指针,其使用方法很简单:

比如这里有一个类:

class User {
public:
	User() {
		cout << "这是构造函数" << endl;
	}
	~User()
	{
		cout << "这是析构函数" << endl;
	}

	void TestFun() {
		cout << "这是一个测试函数" << endl;
	}

};

然后使用共享智能指针:

#include<iostream>
using namespace std;
//上面的那个类可以放在这里
int main() {
	shared_ptr<User> p(new User());
	shared_ptr<User> p1 = p;
	shared_ptr<User> p2 = p;

	p->TestFun(); //调用函数的方式和指针一样
	cout << p.use_count() << endl; //输出共享个数
}

即:通过模板参数,传入要构造的指针类型,然后在初始化的时候,就可以直接new一个对象即可

因为是共享的,所以它还能互相赋值,并可以用函数use_count返回当前共享的个数

其使用方法,如调用类的函数和属性之类的,就和普通的指针一样,用->进行调用即可,但是却不需要我们去亲自清理内存了!

看,现在我们并没有清理内存,但这个类的析构函数却被调用了!这就说明内存已经被正常释放了

这就是智能指针的好处!

但智能指针写着有点麻烦,每次声明其类型都有一长串,所以一般我们会对指针进行重定义,达到简化的目的:

typedef shared_ptr<User> SPUser;
int main() {
    SPUser p(new User());
    SPUser p1 = p;
    SPUser p2 = p;
    p->TestFun(); //调用函数的方式和指针一样
    cout << p.use_count() << endl; //输出共享个数
}

2.unique_ptr

上面的共享指针的使用方法和普通指针区别并不大

但有时候,我们想要某个对象同时只能存在一份,即不允许像共享指针那样,可以到处随意赋值给别人

这时候就可以用unique_ptr,其使用方法如下:

typedef unique_ptr<User> UPUser; //重新定义一个名称,便于使用
int main() {
	UPUser p(new User);
	//UPUser p1 = p; //错误,不能进行赋值
	UPUser p2;
	p2.swap(p); //但可以交换,即p2现在保存有变量,但p变为了空指针
	if (p == nullptr) {
		cout << "p为空指针" << endl;
	}
	p2->TestFun(); //正常调用

	UPUser p3 = move(p2); //也可以用move函数移动
	if (p2 == nullptr) { //此时p2就是空指针
		cout << "p2为空指针" << endl;
	}
	p3->TestFun(); //p3则保存对象指针
}

可以看到,它的使用方法其实和共享指针是差不多的,唯一不同之处就是,它内部的指针值,同一时刻只能存在一份

即,你不能对它进行任何形式的复制,但是可以移动

3.weak_ptr

这个智能指针用的不太多,因为它本身并没有太多实际的用途,而是主要作为shared_ptr的一个辅助类存在

比如有多少指向相同的 shared_ptr 指针、shared_ptr 指针指向的堆内存是否已经被释放等等。

其使用方法如下:

typedef shared_ptr<User> SPUser;
typedef weak_ptr<User> WPUser;
int main() {
	
	SPUser p(new User());
	SPUser p1 = p;
	SPUser p2 = p;

	WPUser wp(p);
	cout << wp.use_count() << endl; //查看这个共享指针使用次数
	cout << wp.expired() << endl; //判断这个指针是否为空,或者内存已经被释放

}

加载全部内容

相关教程
猜你喜欢
用户评论