深入理解和应用C++ std::shared_ptr别名构造函数

深入理解和应用C++ std::shared_ptr别名构造函数

引言

在现代C++中,智能指针是一个极为重要的工具,尤其std::shared_ptr以其自动内存管理、引用计数和多线程安全性等特性深受开发者喜爱。其中一个不太常用但功能强大的构造方式是别名构造函数,它允许我们创建一个共享相同底层对象但是指向其内部不同数据成员或子对象的新shared_ptr。本文将探讨这个特性,并通过实例说明其在复杂类型管理和资源控制中的应用场景。

一、std::shared_ptr及其别名构造函数简介

std::shared_ptr是一种智能指针,采用引用计数机制来自动管理所指向的对象生命周期。当没有更多shared_ptr指向同一块动态分配的内存时,该内存会被自动释放。

cpp 复制代码
template<typename Other>
shared_ptr(const shared_ptr<Other>& r, T* ptr);

此构造函数接受两个参数:一个是另一个shared_ptr实例,另一个是指向原始shared_ptr管理对象内的某个成员或子对象的指针。新生成的shared_ptr会共享第一个参数的引用计数器,但其自身指向的是指定的成员地址。

二、别名构造函数的实际应用场景

下面通过一个具体的例子来阐述std::shared_ptr的别名构造函数如何用于处理复杂类型:

cpp 复制代码
#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;
}
cpp 复制代码
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对象的状态。

相关推荐
张太行_8 分钟前
C++中的析构器(Destructor)(也称为析构函数)
开发语言·c++
SteveKenny2 小时前
Python 梯度下降法(六):Nadam Optimize
开发语言·python
Hello.Reader3 小时前
深入浅出 Rust 的强大 match 表达式
开发语言·后端·rust
涛ing5 小时前
32. C 语言 安全函数( _s 尾缀)
linux·c语言·c++·vscode·算法·安全·vim
xrgs_shz5 小时前
MATLAB的数据类型和各类数据类型转化示例
开发语言·数据结构·matlab
独正己身5 小时前
代码随想录day4
数据结构·c++·算法
我不是代码教父8 小时前
[原创](Modern C++)现代C++的关键性概念: 流格式化
c++·字符串格式化·流格式化·cout格式化
利刃大大8 小时前
【回溯+剪枝】找出所有子集的异或总和再求和 && 全排列Ⅱ
c++·算法·深度优先·剪枝
子燕若水8 小时前
mac 手工安装OpenSSL 3.4.0
c++