C++ Effective Modern Pointer (智能指针模块)

简介: C++ Effective Modern Pointer (智能指针模块)

在C++中,有三种主要的智能指针类型:

  1. std::shared_ptr<T>在这里T是类模板)共享指针是一种引用计数型的智能指针。多个shared_ptr对象可以共享同一个堆上的对象,并且当最后一个引用被销毁时,会自动释放相关资源。
  2. std::unique_ptr<T>唯一指针是一种独占所有权型的智能指针。每个unique_ptr对象拥有对其所指向对象的独立所有权,不能进行复制或共享。当该对象被销毁时,相关资源也会被自动释放。
  3. std::weak_ptr<T>弱引用指针是一种不增加引用计数的智能指针。它用于解决循环引用导致的内存泄漏问题,并通常与shared_ptr一起使用。
  4. 这些智能指针类型都定义在 <memory> 头文件中,并通过RAII(资源获取即初始化)原则来确保在离开作用域时正确地释放资源。使用智能指针可以减少手动管理内存的复杂性和风险,提高代码的安全性和可维护性。

1.std::shared_ptr<T> 核心算法 :引用计数算法(每个对象都需要额外的空间来储存计数器 自动跟踪对象被引用的次数 简单来说就是 std::share_ptr所指向的个数 核心要点:多个指针指向一块地址)

//这些智能指针类型都定义在 <memory> 头文件中,并通过RAII(资源获取即初始化)原则来确保在离开作用域时正确地释放资源。
//使用智能指针可以减少手动管理内存的复杂性和风险,提高代码的安全性和可维护性。
#include<bits/stdc++.h>
using namespace std;
//std::shared_ptr
class MyClass 
{
public:
    MyClass() 
    {
        std::cout << "MyClass constructor" << std::endl;
    }
    ~MyClass() 
    {
        std::cout << "MyClass destructor" << std::endl;
    }
    void sayHello()
    {
        std::cout << "Hello, World!" << std::endl;
    }
};
int main() {
    // 创建第一个shared_ptr对象
    std::shared_ptr<MyClass> ptr1(new MyClass());
    // 使用箭头操作符访问成员函数
    ptr1->sayHello();
    // 创建第二个shared_ptr对象,指向同一个对象
    shared_ptr<MyClass> ptr2 = ptr1;
    
    //在没改变开始进行引用计数 有两个hared_ptr<MyClass>指针指向当前对象
    cout << "Current reference count: " << ptr2.use_count() << std::endl;//2
    // 创建第三个shared_ptr对象,并使用reset方法指向新的对象 可以减少引用
    ptr2.reset(new MyClass());
    // 还可以通过*操作符解引用指针来访问对象的成员函数
    (*ptr2).sayHello();
    // 打印当前引用计数
    cout << "Current reference count: " << ptr2.use_count() << std::endl;//1
    // 最后一个shared_ptr被销毁时自动释放内存资源
    //智能指针被最后释放
    return 0;
}

std::unique_ptr<T>:(唯一指针 独占指针 通过std::move(ptr))来操作:

代码实例如下:

#include<bits/stdc++.h>
using namespace std;
class MyClass {
public:
    MyClass() {
        std::cout << "MyClass constructor" << std::endl;
    }
    ~MyClass() {
        std::cout << "MyClass destructor" << std::endl;
    }
    void sayHello() {
        std::cout << "Hello from MyClass!" << std::endl;
    }
};
int main() {
    // 创建一个管理 MyClass 对象的 unique_ptr
    std::unique_ptr<MyClass> ptr(new MyClass());
    ptr->sayHello();  // 使用箭头运算符访问成员函数
    // 转移所有权给另一个 unique_ptr
    std::unique_ptr<MyClass> otherPtr(std::move(ptr));
    if (ptr == nullptr) {
        std::cout << "ptr is empty!" << std::endl;  // 输出指针为空
    }
    otherPtr->sayHello();  // 在新的 unique_ptr 中访问被管理对象
    return 0;
}

std::weak_ptr<T>(弱指针 为什么说比较弱呢 因为他不像shared_ptr 那样 通过引用计数)

主要解决shared_ptr那样的循环引用问题

shared_ptr循环引用问题是指

循环引用问题是指在使用 std::shared_ptr 管理对象时,如果存在相互引用的情况,可能导致对象无法正常销毁,从而造成内存泄漏。

循环引用通常发生在两个或多个对象之间互相持有对方的 shared_ptr。当这些对象的引用计数都不为零时,它们将永远无法被析构,从而导致资源泄漏。

循环引用(强引用代码实例):

#include <memory>
#include<iostream>
class A;
class B;
class A {
public:
    std::shared_ptr<B> bPtr;
};
class B {
public:
    std::shared_ptr<A> aPtr;
};
int main() {
    std::shared_ptr<A> aPtr = std::make_shared<A>();
    std::shared_ptr<B> bPtr = std::make_shared<B>();
    // 创建循环引用
    aPtr->bPtr = bPtr;
    bPtr->aPtr = aPtr;
    return 0;
}

智能指针弱引用解决问题:

#include <iostream>
#include <memory>
class MyClass;
// 定义一个使用 weak_ptr 的类
class MyOtherClass {
public:
    std::weak_ptr<MyClass> myPtr;  // 使用 weak_ptr
    void doSomething() {
        if (auto ptr = myPtr.lock()) {  // 使用 lock() 获取 shared_ptr
            std::cout << "Doing something with MyClass object!" << std::endl;
        }
        else {
            std::cout << "MyClass object has been destroyed!" << std::endl;
        }
    }
};
class MyClass {
public:
    std::shared_ptr<MyOtherClass> otherPtr;  // 使用 shared_ptr
    ~MyClass() {
        std::cout << "MyClass destructor" << std::endl;
    }
};
int main() {
    auto obj1 = std::make_shared<MyClass>();
    auto obj2 = std::make_shared<MyOtherClass>();
    obj1->otherPtr = obj2;         // 将 shared_ptr 赋值给成员变量
    obj2->myPtr = obj1;            // 将 weak_ptr 赋值给成员变量
    obj2->doSomething();           // 此时 MyClass 对象还存在
    obj1.reset();                  // 释放对 MyClass 对象的所有权
    obj2->doSomething();           // MyClass 对象已被销毁
    return 0;
}

此代码避免了强指针占有权的问题 能够安全的释放内存和相对应的指针

C++ 11新增的主要的就这几个指针   C++20出了新的智能指针 这几个智能指针已经够用在开发和创作企业级项目当中了

总结:我们在使用智能指针的时候 要注意强引用问题std::weak_ptr和std::shared_ptr通常都是成对的出现和使用   我们需要考虑所有权的时候 使用std::unique_ptr指针 如果是要节约空间并且不考虑所有权的问题 推荐std::weak_ptr和std::shared_ptr配套使用

在这里我想给大家推荐一个高质量课程 小编学了以后收获满满 性价比高 链接地址如下:

https://xxetb.xetslk.com/s/2PjJ3T

技术总是一点一点积累的 这期的文章就到这里了 小编在此跟大家说拜拜

相关文章
|
1月前
指针(Pointer)的深度理解(2)
指针(Pointer)的深度理解(2)
34 1
|
1月前
|
C++
析构造函数就是为了释放内存,就是在局部指针消失前释放内存,拷贝构造函数就是以构造函数为模块,在堆里面新开一块,同一个变量在堆里面的地址
本文讨论了C++中构造函数和析构函数的作用,特别是它们在管理动态内存分配和释放中的重要性,以及如何正确地实现拷贝构造函数以避免内存泄漏。
37 2
|
1月前
|
NoSQL API Redis
如何使用 C++ 开发 Redis 模块
如何使用 C++ 开发 Redis 模块
|
2月前
|
C++
C++(十八)Smart Pointer 智能指针简介
智能指针是C++中用于管理动态分配内存的一种机制,通过自动释放不再使用的内存来防止内存泄漏。`auto_ptr`是早期的一种实现,但已被`shared_ptr`和`weak_ptr`取代。这些智能指针基于RAII(Resource Acquisition Is Initialization)原则,即资源获取即初始化。RAII确保对象在其生命周期结束时自动释放资源。通过重载`*`和`-&gt;`运算符,可以方便地访问和操作智能指针所指向的对象。
|
3月前
|
安全 NoSQL Redis
C++新特性-智能指针
C++新特性-智能指针
|
4月前
|
NoSQL API Redis
c++开发redis module问题之为什么在使用RedisModule_GetApi之前要通过((void**)ctx)[0]这种方式获取其地址
c++开发redis module问题之为什么在使用RedisModule_GetApi之前要通过((void**)ctx)[0]这种方式获取其地址
|
5月前
|
C++ 容器
【编程技巧】 C++11智能指针
C++11引入了智能指针以自动管理内存,防止内存泄漏和悬挂指针: - `shared_ptr`:引用计数,多所有权,适用于多个对象共享资源。 - `unique_ptr`:独占所有权,更轻量级,适用于单一对象所有者。 - `weak_ptr`:弱引用,不增加引用计数,解决`shared_ptr`循环引用问题。 ## shared_ptr - 支持引用计数,所有者共同负责资源释放。 - 创建方式:空指针、new操作、拷贝构造/移动构造,以及自定义删除器。 - 提供`operator*`和`operator-&gt;`,以及`reset`、`swap`等方法。 ## unique_ptr
255 10
|
4月前
|
NoSQL 编译器 Redis
c++开发redis module问题之如果Redis加载了多个C++编写的模块,并且它们都重载了operator new,会有什么影响
c++开发redis module问题之如果Redis加载了多个C++编写的模块,并且它们都重载了operator new,会有什么影响
|
4月前
|
NoSQL Redis C++
c++开发redis module问题之在复杂的Redis模块中,特别是使用第三方库或C++开发时,接管内存统计有哪些困难
c++开发redis module问题之在复杂的Redis模块中,特别是使用第三方库或C++开发时,接管内存统计有哪些困难
|
3月前
|
安全 编译器 容器
C++STL容器和智能指针
C++STL容器和智能指针