2.1讲一下什么是智能指针、智能指针有哪几种:
相比于裸指针,智能指针是对裸指针的一种封装,初衷是让程序员无需手动释放内存,来避免内存泄漏。常见的智能指针有unique_ptr,shared_ptr,weak_ptr。
2.2 讲一下这几种智能指针的区别和优缺点
- unique对资源是独占的,它不可以复制。它通过在析构函数中释放资源来管理对象的生命周期,来自动管理资源。可以防止多个智能指针指向同一个对象,更方便于管理。相比shared_ptr它的优点是高效,避免了循环引用问题。
- 当一个资源需要多个对象之间共享时,就无法使用unique_ptr了。这就需要使用到shared_ptr,它通过引用计数的方式来对资源进行控制。但是它存在循环引用的问题
- weak_ptr本身不具备内存管理能力,主要是为了解决shared_ptr可能导致的循环引用问题,weak_ptr指向某个资源时,它不会增加这个资源的引用计数。
2.3 具体讲一下shared_ptr自动管理内存的原理/引用计数的具体原理/shared_ptr引用计数什么时候会增加,什么时候会减少?
引用计数的核心原理:
在shared_ptr的内部维护了一个计数器,来跟踪有多少个shared_ptr对象指向了某一个资源。当计数器的值减少到0的时候,shared_ptr就会调用delete(或者用户自定义的方法)来释放资源。
引用计数器何时增加:
1.新建一个shared_ptr并指向一个资源时。
2.复制构造函数创建一个新的shared_ptr时。
3.用复制运算符将一个shared_ptr给另一个shared_ptr对象赋值时。
引用计数器何时减少:
1.当一个shared_ptr对象被销毁时,比如局部变量离开作用域,或者类成员变量析构时。
2. 当一个shared_ptr对象不再指向一个资源时,例如通过reset方法或者赋值运算符指向另一个资源时。
2.4 智能指针是如何自动管理资源的?智能指针的核心原理是什么?
2.5 讲一下循环引用是如何发生的,以及如何解决?
两个对象或多个对象相互引用,可能会导致循环引用,两个对象资源都无法被释放。这时候需要weak_ptr来打破循环引用。因为weak_ptr指向某一个资源时,不会增加这个资源的引用计数。
class B;
class A {
public:
std::shared_ptr<B> bPtr;
}
class B {
public:
std::shared_ptr<A> aPtr;
}
int main () {
std::shared_ptr<A> pa = std::make_shared<A>(); // 在堆区创建类A并使用智能指针管理
std::shared_ptr<B> pb = std::make_shared<B>(); // 在堆区创建类B并使用智能指针管理
pa->bPtr = pb; // b的引用计数为2
pb->aPtr = pa; // a的引用计数为2
// pa生命周期结束 a的引用计数为1 堆区的b的成员aPtr指向a
// pb生命周期结束 b的引用计数为1 堆区的a的成员bPtr指向b
}
class B;
class A {
public:
std::shared_ptr<B> bPtr;
}
class B {
public:
std::weak_ptr<A> aPtr;
}
int main () {
std::shared_ptr<A> pa = std::make_shared<A>(); // 在堆区创建类A并使用智能指针管理
std::shared_ptr<B> pb = std::make_shared<B>(); // 在堆区创建类B并使用智能指针管理
pa->bPtr = pb; // b的引用计数为2
pb->aPtr = pa; // a的引用计数为1
// pa生命周期结束 a的引用计数为0 a被释放
// pb生命周期结束 b的引用计数为1 堆区的a的成员bPtr指向b a被释放后 b的引用计数为0 b被释放
}
2.6 weak_ptr是弱引用,它弱在哪里?
不参与引用计数。
2.7 shared_ptr是线程安全的吗?多线程中使用智能指针要注意什么?
shared_ptr内部的引用计数增加减少是线程安全的。但是shared_ptr本身不是线程安全的,多个线程访问和修改同一个智能指针可能会导致数据竞争。
参考:b站骑猪撞宝马71