深入理解和应用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对象的状态。

相关推荐
nbsaas-boot2 小时前
Java 正则表达式白皮书:语法详解、工程实践与常用表达式库
开发语言·python·mysql
岁忧2 小时前
(LeetCode 面试经典 150 题 ) 11. 盛最多水的容器 (贪心+双指针)
java·c++·算法·leetcode·面试·go
chao_7892 小时前
二分查找篇——搜索旋转排序数组【LeetCode】两次二分查找
开发语言·数据结构·python·算法·leetcode
风无雨2 小时前
GO 启动 简单服务
开发语言·后端·golang
斯普信专业组2 小时前
Go语言包管理完全指南:从基础到最佳实践
开发语言·后端·golang
我是苏苏4 小时前
C#基础:Winform桌面开发中窗体之间的数据传递
开发语言·c#
斐波娜娜4 小时前
Maven详解
java·开发语言·maven
小码氓5 小时前
Java填充Word模板
java·开发语言·spring·word
暮鹤筠5 小时前
[C语言初阶]操作符
c语言·开发语言
蜉蝣之翼❉7 小时前
CRT 不同会导致 fopen 地址不同
c++·mfc