// // Created by light on 19-12-12. // #include "../RAII/shape.h" class shared_count { public: shared_count() : count_(1) {} // 增加计数 void add_count() { ++count_; } // 减少计数 long reduce_count() { return --count_; } // 获取当前计数 long get_count() const { return count_; } private: long count_; }; template class shared_ptr { public: explicit shared_ptr(T *ptr = nullptr) noexcept : ptr_(ptr) { if (ptr) { shared_count_ = new shared_count(); } } // 实现强制类型转换需要的构造函数 template shared_ptr(const shared_ptr &other, T *ptr) noexcept { ptr_ = ptr; if (ptr_) { other.shared_count_->add_count(); shared_count_ = other.shared_count_; } } ~shared_ptr() noexcept { // 最后一个shared_ptr再去删除对象与共享计数 // ptr_不为空且此时共享计数减为0的时候,再去删除 if (ptr_ && !shared_count_->reduce_count()) { delete ptr_; delete shared_count_; } } T &operator*() const noexcept { return *ptr_; } T *operator->() const noexcept { return ptr_; } operator bool() const noexcept { return ptr_; } T *get() const noexcept { return ptr_; } // 带模板的拷贝与移动构造函数 模板的各个实例间并不天然就有 friend // 关系,因而不能互访私有成员 ptr_ 和 shared_count_。 需要下面显示声明 template friend class shared_ptr; template shared_ptr(const shared_ptr &other) noexcept { // cout << "调用了带模板的拷贝构造!" << endl; ptr_ = other.ptr_; if (ptr_) { other.shared_count_->add_count(); shared_count_ = other.shared_count_; } } template shared_ptr(shared_ptr &&other) noexcept { // cout << "调用了带模板的移动构造!" << endl; ptr_ = other.ptr_; if (ptr_) { shared_count_ = other.shared_count_; other.ptr_ = nullptr; other.shared_count_ = nullptr; } } // copy and swap 始终只有一个对象有管理这块空间的权限 shared_ptr &operator=(shared_ptr rhs) noexcept { rhs.swap(*this); return *this; } void swap(shared_ptr &rhs) noexcept { using std::swap; swap(ptr_, rhs.ptr_); swap(shared_count_, rhs.shared_count_); } long use_count() const noexcept { if (ptr_) { return shared_count_->get_count(); } else { return 0; } } private: T *ptr_; shared_count *shared_count_; }; template void swap(shared_ptr &lhs, shared_ptr &rhs) noexcept { lhs.swap(rhs); } template shared_ptr dynamic_pointer_cast(const shared_ptr &other) noexcept { T *ptr = dynamic_cast(other.get()); return shared_ptr(other, ptr); } template shared_ptr static_pointer_cast(const shared_ptr &other) noexcept { T *ptr = static_cast(other.get()); return shared_ptr(other, ptr); } template shared_ptr const_pointer_cast(const shared_ptr &other) noexcept { T *ptr = const_cast(other.get()); return shared_ptr(other, ptr); } template shared_ptr reinterpret_pointer_cast(const shared_ptr &other) noexcept { T *ptr = reinterpret_cast(other.get()); return shared_ptr(other, ptr); } int main() { shared_ptr ptr1(new circle()); cout << "use count of ptr1 is " << ptr1.use_count() << endl; shared_ptr ptr2, ptr3; cout << "use count of ptr2 was " << ptr2.use_count() << endl; ptr2 = ptr1; // shared_ptr隐式转换shared_ptr 调用带模板的拷贝构造 // cout<<"======="< circle* // 使用dynamic_cast转换后,指针为空.此时资源还是被dptr2拥有,dptr1为0 shared_ptr dptr2(new shape); shared_ptr dptr1 = dynamic_pointer_cast(dptr2); // 基类转子类 cout << "use count of dptr1 is now " << dptr1.use_count() << endl; // 0 cout << "use count of dptr2 is now " << dptr2.use_count() << endl; // 1 // circle* -> circle* // 使用dynamic_cast转换后,指针不为空,此时资源被两者共同使用,引用计数为2 shared_ptr dptr3(new circle); // shared_ptr dptr3(new circle); // // 上面或者当前行,后面输出一样! shared_ptr dptr1_1 = dynamic_pointer_cast(dptr3); // 基类转子类 cout << "use count of dptr1_1 is now " << dptr1_1.use_count() << endl; // 2 cout << "use count of dptr3 is now " << dptr3.use_count() << endl; // 2 // circle* -> circle* // 使用dynamic_cast转换后,指针不为空,此时资源被两者共同使用,引用计数为2 shared_ptr dptr3_1(new circle); shared_ptr dptr2_1 = dynamic_pointer_cast(dptr3_1); // 子类转基类 上行转换,安全! cout << "use count of dptr2_1 is now " << dptr2_1.use_count() << endl; // 2 cout << "use count of dptr3_1 is now " << dptr3_1.use_count() << endl; // 2 // shape* -> circle* 使用static_cast转换后,指针为空 与dynamic_cast相比,不安全 shared_ptr sptr2(new shape); shared_ptr sptr1 = static_pointer_cast(sptr2); // 基类转子类 cout << "use count of sptr1 is now " << dptr1.use_count() << endl; // 0 cout << "use count of sptr2 is now " << dptr2.use_count() << endl; // 1 // circle* -> circle* // 使用dynamic_cast转换后,指针不为空,此时资源被两者共同使用,引用计数为2 shared_ptr sptr3(new circle); // shared_ptr sptr3(new circle); // // 上面或者当前行,后面输出一样! shared_ptr sptr1_1 = static_pointer_cast(sptr3); // 基类转子类 cout << "use count of sptr1_1 is now " << sptr1_1.use_count() << endl; // 2 cout << "use count of sptr3 is now " << sptr3.use_count() << endl; // 2 // circle* -> circle* // 使用static_cast转换后,指针不为空,此时资源被两者共同使用,引用计数为2 // 等价于dynamic_cast shared_ptr sptr3_1(new circle); shared_ptr sptr2_1 = static_pointer_cast(sptr3_1); // 子类转基类 上行转换,安全! cout << "use count of sptr2_1 is now " << sptr2_1.use_count() << endl; // 2 cout << "use count of sptr3_1 is now " << sptr3_1.use_count() << endl; // 2 shared_ptr constV(new int); shared_ptr s = const_pointer_cast(constV); *s = 10; int a = reinterpret_pointer_cast(s); cout << a << endl; }