文章目录
前言
今天和人争论C++ 智能指针shared_ptr引用计数问题,然后代码实现验证,这里分享给大家。
一、C++ 智能指针shared_ptr
shared_ptr智能指针是一个类,用于管理new 分配的对象,该智能指针类内部有一个保存引用计数的属性,通过这个属性来判断是否可以释放要管理的对象, 该判断是在智能指针类shared_ptr的析构函数中进行。当引用计数为0时,表示当前程序没有用到该管理对象的地方了,可以释放该管理对象了,即delete 对象。
什么时候引用计数会改变呢?请看下面程序
std::shared_ptr<int> sptr1 = std::make_shared<int>();
std::cout << "sptr1 use count:"<<sptr1.use_count() << std::endl;
{
std::cout << std::endl;
std::shared_ptr<int> sptr2 = sptr1;
std::cout << std::endl << "sptr2=sptr1;" << std::endl;
std::cout << "sptr1 use count:" << sptr1.use_count() << std::endl;
std::cout << "sptr2 use count:" << sptr2.use_count() << std::endl;
std::cout << std::endl;
std::shared_ptr<int> sptr3 = std::make_shared<int>();
std::cout << "sptr3 uwe count:" << sptr3.use_count() << std::endl;
std::cout << std::endl << "sptr2=sptr3;" << std::endl;
sptr2 = sptr3;
std::cout << "sptr1 use count:" << sptr1.use_count() << std::endl;
std::cout << "sptr3 uwe count:" << sptr3.use_count() << std::endl;
std::cout << "sptr2 use count:" << sptr2.use_count() << std::endl;
std::cout << std::endl;
std::shared_ptr<int> sptr5 = sptr1;
std::cout << "sptr5=sptr1;" << std::endl;
std::cout << "sptr5 use count:" << sptr5.use_count() << std::endl;
std::cout << "sptr1 use count:" << sptr1.use_count() << std::endl;
std::cout << "sptr5 end..." << std::endl;
}
std::cout <<std::endl<< "sptr1 use count:" << sptr1.use_count() << std::endl;
std::shared_ptr<int> sptr4 = std::make_shared<int>();
std::cout << std::endl<< "sptr4 use count:" << sptr4.use_count() << std::endl;
sptr1 = sptr4;
std::cout << std::endl<< "sptr1 = sptr4;" << std::endl;
std::cout << "sptr4 use count:" << sptr4.use_count() << std::endl;
std::cout << "sptr1 use count:" << sptr1.use_count() << std::endl;
std::cout << std::endl<<"reset sptr1" << std::endl;
sptr1.reset();// reset 后,sptr1引用计数归0, sptr4引用计数-1
std::cout << "sptr4 use count:" << sptr4.use_count() << std::endl;
std::cout << "sptr1 use count:" << sptr1.use_count() << std::endl;
执行结果:
总结
智能指针引用计数改变:构造函数(sptr1),赋值操作符(sptr2=sptr3),拷贝构造函数(std::shared_ptr sptr2=sptr1)以及调用reset()函数(sptr1.reset()),拷贝的智能指针shared_ptr声明周期结束(sptr5)。