智能指针shared_ptr的用法

本文介绍了C++11中智能指针的概念及其三种类型:std::shared_ptr, std::unique_ptr 和 std::weak_ptr。重点讲解了shared_ptr的工作原理及使用注意事项,包括初始化方法、获取原始指针、指定删除器等,并列举了一些使用时常见的问题。

源文章链接:http://www.cnblogs.com/jiayayao/archive/2016/12/03/6128877.html
为了解决C++内存泄漏的问题,C++11引入了智能指针(Smart Pointer)。

  智能指针的原理是,接受一个申请好的内存地址,构造一个保存在栈上的智能指针对象,当程序退出栈的作用域范围后,由于栈上的变量自动被销毁,智能指针内部保存的内存也就被释放掉了(除非将智能指针保存起来)。

  C++11提供了三种智能指针:std::shared_ptr, std::unique_ptr, std::weak_ptr,使用时需添加头文件。

  shared_ptr使用引用计数,每一个shared_ptr的拷贝都指向相同的内存。每使用他一次,内部的引用计数加1,每析构一次,内部的引用计数减1,减为0时,删除所指向的堆内存。shared_ptr内部的引用计数是安全的,但是对象的读取需要加锁。

shared_ptr的基本用法
初始化
  可以通过构造函数、std::make_shared辅助函数和reset方法来初始化shared_ptr:

#include "stdafx.h"
#include <iostream>
#include <future>
#include <thread>

using namespace std;
class Person
{
public:
    Person(int v) {
        value = v;
        std::cout << "Cons" <<value<< std::endl;
    }
    ~Person() {
        std::cout << "Des" <<value<< std::endl;
    }

    int value;

};

int main()
{
    std::shared_ptr<Person> p1(new Person(1));// Person(1)的引用计数为1

    std::shared_ptr<Person> p2 = std::make_shared<Person>(2);

    p1.reset(new Person(3));// 首先生成新对象,然后引用计数减1,引用计数为0,故析构Person(1)
                            // 最后将新对象的指针交给智能指针

    std::shared_ptr<Person> p3 = p1;//现在p1和p3同时指向Person(3),Person(3)的引用计数为2

    p1.reset();//Person(3)的引用计数为1
    p3.reset();//Person(3)的引用计数为0,析构Person(3)
    return 0;
}

  注意,不能将一个原始指针直接赋值给一个智能指针,如下所示,原因是一个是类,一个是指针。

    std::shared_ptr<int> p4 = new int(1);// error

  reset()有两种操作。当智能指针中有值的时候,调用reset()会使引用计数减1.当调用reset(new xxx())重新赋值时,智能指针首先是生成新对象,然后将就对象的引用计数减1(当然,如果发现引用计数为0时,则析构旧对象),然后将新对象的指针交给智能指针保管。

获取原始指针

    std::shared_ptr<int> p4(new int(5));
    int *pInt = p4.get();

指定删除器
  智能指针可以指定删除器,当智能指针的引用计数为0时,自动调用指定的删除器来释放内存。std::shared_ptr可以指定删除器的一个原因是其默认删除器不支持数组对象,这一点需要注意。

  2. 使用shared_ptr需要注意的问题

  但凡一些高级的用法,使用时都有不少陷阱。

不要用一个原始指针初始化多个shared_ptr,原因在于,会造成二次销毁,如下所示:

    int *p5 = new int;
    std::shared_ptr<int> p6(p5);
    std::shared_ptr<int> p7(p5);// logic error

不要在函数实参中创建shared_ptr。因为C++的函数参数的计算顺序在不同的编译器下是不同的。正确的做法是先创建好,然后再传入。
function(shared_ptr(new int), g());
禁止通过shared_from_this()返回this指针,这样做可能也会造成二次析构。
避免循环引用。智能指针最大的一个陷阱是循环引用,循环引用会导致内存泄漏。解决方法是AStruct或BStruct改为weak_ptr。

struct AStruct;
struct BStruct;

struct AStruct {
    std::shared_ptr<BStruct> bPtr;
    ~AStruct() { cout << "AStruct is deleted!"<<endl; }
};

struct BStruct {
    std::shared_ptr<AStruct> APtr;
    ~BStruct() { cout << "BStruct is deleted!" << endl; }
};

void TestLoopReference()
{
    std::shared_ptr<AStruct> ap(new AStruct);
    std::shared_ptr<BStruct> bp(new BStruct);
    ap->bPtr = bp;
    bp->APtr = ap;
}
<think>好的,我现在需要回答用户关于C++shared_ptr的使用方法及示例的问题。首先,我应该回忆一下自己对shared_ptr的了解,并结合提供的引用资料来确保信息的准确性。 首先,shared_ptrC++智能指针的一种,用于自动管理动态分配的内存,防止内存泄漏。根据引用1,shared_ptr采用引用计数机制,多个shared_ptr可以共享同一个对象,当最后一个shared_ptr被销毁时,对象才会被释放。引用2提到shared_ptr允许拷贝和赋值,这说明它们支持共享所有权。引用3则补充了shared_ptr销毁对象的条件,即最后一个拥有对象的shared_ptr被销毁或重置时。 接下来,我需要组织回答的结构。用户需要使用方法及示例,所以应该分步骤讲解,并附上代码示例。可能的要点包括:创建shared_ptr、共享所有权、自定义删除器、注意事项等。 创建shared_ptr部分,需要说明如何用make_shared或直接构造。共享所有权要展示多个指针指向同一对象的情况。自定义删除器可以举文件关闭的例子。注意事项要提到循环引用的问题,并引入weak_ptr作为解决方案。 在写代码示例时,要确保语法正确,并且每段代码有注释解释。同时,要引用提供的资料中的信息,比如在说明引用计数时引用[^1],在提到拷贝赋值时引用[^2],在销毁条件时引用[^3]。 另外,用户的问题可能涉及相关技术问题,比如循环引用,所以需要解释问题原因和解决方法。这里需要提到weak_ptr,并给出示例,如引用3中提到的weak_ptr用于打破循环。 最后,生成相关问题时要围绕shared_ptr的扩展内容,比如与unique_ptr的区别、循环引用的解决、性能分析等。 需要检查是否符合用户的要求:使用中文、正确格式的数学公式(如果有的话)、引用标识、代码块格式正确。但在这个问题中可能没有数学公式,但如果有涉及计算的地方,需要按格式处理。 总结回答的结构: 1. shared_ptr基本概念和引用计数机制 2. 创建方法(make_shared) 3. 共享所有权示例 4. 自定义删除器 5. 注意事项(循环引用和weak_ptr) 6. 示例代码 7. 相关问题 确保每个部分都简洁明了,引用正确的资料,示例代码正确无误。</think>### C++ shared_ptr 使用方法及示例 #### 1. shared_ptr 基本概念 `shared_ptr` 是 C++ 标准库提供的智能指针,通过**引用计数机制**管理动态分配的内存。多个 `shared_ptr` 可以指向同一对象,当最后一个 `shared_ptr` 被销毁或重置时,对象内存会被自动释放[^1]。 --- #### 2. 使用方法 ##### (1) 创建 shared_ptr 推荐使用 `std::make_shared` 创建,既安全又高效: ```cpp #include <memory> // 创建指向 int 的 shared_ptr auto sp1 = std::make_shared<int>(42); // 直接构造(不推荐,可能引发异常安全问题) std::shared_ptr<int> sp2(new int(100)); ``` ##### (2) 共享所有权 多个 `shared_ptr` 可以共享同一对象,引用计数自动增加/减少: ```cpp auto sp3 = sp1; // 引用计数变为2 sp3.reset(); // 引用计数恢复为1 ``` ##### (3) 自定义删除器 默认删除器调用 `delete`,但可自定义(如用于文件句柄): ```cpp std::shared_ptr<FILE> file_ptr( fopen("test.txt", "r"), [](FILE* f) { if (f) fclose(f); } ); ``` --- #### 3. 注意事项 ##### 循环引用问题 若两个 `shared_ptr` 互相引用,会导致内存泄漏。需用 `weak_ptr` 打破循环: ```cpp class B; // 前向声明 class A { public: std::shared_ptr<B> b_ptr; }; class B { public: std::weak_ptr<A> a_weak_ptr; // 使用 weak_ptr 避免循环引用 }; ``` --- #### 4. 完整示例 ```cpp #include <iostream> #include <memory> struct Data { Data() { std::cout << "Data created\n"; } ~Data() { std::cout << "Data destroyed\n"; } }; int main() { // 创建 shared_ptr auto d1 = std::make_shared<Data>(); { auto d2 = d1; // 引用计数变为2 std::cout << "d2 使用中,引用计数=" << d2.use_count() << "\n"; } // d2 销毁,引用计数恢复为1 std::cout << "d1 引用计数=" << d1.use_count() << "\n"; return 0; } ``` **输出**: ``` Data created d2 使用中,引用计数=2 d1 引用计数=1 Data destroyed ``` ---
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值