在现代C++中,智能指针是一个极为重要的工具,尤其std::shared_ptr以其自动内存管理、引用计数和多线程安全性等特性深受开发者喜爱。其中一个不太常用但功能强大的构造方式是别名构造函数
,它允许我们创建一个共享相同底层对象但是指向其内部不同数据成员或子对象的新shared_ptr。本文将探讨这个特性,并通过实例说明其在复杂类型管理和资源控制中的应用场景。
std::shared_ptr是一种智能指针,采用引用计数机制来自动管理所指向的对象生命周期。当没有更多shared_ptr指向同一块动态分配的内存时,该内存会被自动释放。
template<typename Other>
shared_ptr(const shared_ptr<Other>& r, T* ptr);
此构造函数接受两个参数:一个是另一个shared_ptr实例,另一个是指向原始shared_ptr管理对象内的某个成员或子对象的指针。新生成的shared_ptr会共享第一个参数的引用计数器,但其自身指向的是指定的成员地址。
下面通过一个具体的例子来阐述std::shared_ptr的别名构造函数如何用于处理复杂类型:
#include <memory>
#include <iostream>
// 定义一个自定义结构体
struct MyData {
int a;
int b;
};
std::ostream& operator<<(std::ostream& out, const MyData &data) {
out << "a = " << data.a << ", b = " << data.b;
return out;
}
int main()
{
// 创建一个指向MyData结构体对象的shared_ptr
std::shared_ptr<MyData> sptr1(new MyData{10, 20});
// 使用别名构造创建一个新的shared_ptr,它共享sptr1的引用计数,
// 但其内部指针指向sptr1所管理的对象的a成员
int* memberPtr = &sptr1->b;
std::shared_ptr<int> sptr3(sptr1, memberPtr);
std::cout << "sptr1:" << sptr1 << ", use_count:" << sptr1.use_count() << ", get():" << sptr1.get() << ", *sptr1:" << *sptr1 << std::endl;
std::cout << "sptr3:" << sptr3 << ", use_count:" << sptr3.use_count() << ", get():" << sptr3.get() << ", *sptr3:" << *sptr3<< std::endl;
// 此时,改变通过sptr3访问的值会影响到原始对象
*sptr3 = 30;
std::cout << sptr1->b << std::endl; // 输出:30
return 0;
}
sptr1:0x7f9f5fc02aa0, use_count:2, get():0x7f9f5fc02aa0, *sptr1:a = 10, b = 20
sptr3:0x7f9f5fc02aa4, use_count:2, get():0x7f9f5fc02aa4, *sptr3:20
30
在这个场景中,尽管sptr3仅指向MyData结构体的单个整型成员b,但它同样参与了整个MyData对象生命周期的管理。因此,即使我们在操作sptr3时,也间接地影响到了原始MyData对象的状态。