C++资源管理操作方法详解

 更新时间:2022年09月24日 11:53:04   作者:RolleX  
系统中的资源,诸如动态申请的内存,文件描述符,数据库连接,网络socket等,在不用的时候,应该及时归还给系统,否则就会造成内存泄露

以对象管理资源

class A{...};
//工厂函数createA来提供特定的A对象
A* createA();
//坏情况
void f(){
    A* p=createA();
    ...
    delete p;//如何在delete之前程序先return了,则无法delete
}
//为了确保资源释放 将资源放进对象内 利用对象的析构函数来释放
void f(){
    std::shared_ptr<A> p(createA());
    ...
    //程序结束后会经shared_ptr的析构函数释放
}

auto_ptr已经被废除,主要原因是其拷贝会造成所以权转移,所以使用shared_ptr更好

以对象管理资源的关键想法:

1、获得资源后立刻放进管理对象内

2、管理对象运用析构函数释放资源

注意是:shared_ptr和auto_ptr两者都在析构函数上使用delete而不是delete[],所以在动态分配的数组上使用它们不好。当然我觉得还是少用动态数组,用vector,string啥的就能代替咯。

在资源管理类中小心copy行为

复制RAII对象时有两种选择:

1、禁止复制--------将copying操作声明为private

class Lock:private Uncopyable{
public:
    ....
};

2、对底层资源祭出"引用计数法"--------即shared_ptr

class Lock{
public:
    explicit Lock(Mutex* pm):mutexPtr(pm,unlock)//unlock函数为删除器
    {
        lock(mutexPtr.get());
    }
private:
    shared_ptr<Mutex>mutexPtr;
}

复制底部资源:需要资源管理类对象的唯一理由是不需要某个复件资源时确保被释放,在此情况下复制资源管理类对象,应该也复制其所包括的资源,即深拷贝------当一个对象被复制时,不论指针或其所指内存都会被制作出一个复件,即深拷贝。

转移底部资源所有权:auto_ptr,你只希望有一个RAII对象指向一个未加工资源,即使被复制也是如此。

在资源管理类中提供对原始资源的访问

前提:智能指针其实是一个类

shared_ptr<A>p(createA());
//假如有以下函数
int func(const A* pi);
//如下调用错误 因为p是一个智能指针不是一个指向A的指针
int f=func(p);
//调用get函数返回原始资源
int f=func(p.get());

智能指针重载了指针取值操作符(->,*) 允许隐式转换到原始指针

//例如A有一个函数,p是一个指向A的智能指针
int A::getNum();
//如下调用合格,其实是发生了一个智能指针到原始指针的隐式转换
int num=p->getNum();

一般而言显示转换比较安全,隐式转换客户使用方便。

成对使用new和delete时要采用相同形式

delete的最大问题在于:即将被删除的内存之内究竟有多少个对象

即:被删除的那个指针是指的单一对象还是成对数组?

string* p1=new string;
string* p2=new string[100];
delete p1;
delete[] p2;

规则很简单:new中用了[],delete就要用[]

以独立语句将new对象置入智能指针

假设有一个函数来揭示处理程序的优先权,另一个函数用来在动态分配所得的Widget上进行某些带有优先权的处理

int priority();
void processWidget(shared_ptr<Widget>pw,int priority);
//错误
processWideget(new Widget,priority());
//因为shared_ptr的构造函数需要一个原始指针,该构造函数是explicit,无法隐式转换
processWideget(shared_ptr<Widget>(new Widget),priority());//可以

但是编译器执行顺序不确定,调用该函数前,编译器需要:

调用priority,执行new Widget,shared_ptr构造函数

但调用priority的顺序可以是第一第二或第三(new 和智能指针的先后顺序不能变)

如何是先new,后priority,再shared_ptr,万一priority调用失常,则new出来的指针遗失,尚未置入shaerd_ptr的构造函数,则会资源泄漏。

因此分开写最好:

shared_ptr<Widget>pw(new Wideget);
processWidget(pw,priority());

到此这篇关于C++资源管理操作方法详解的文章就介绍到这了,更多相关C++资源管理内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • C语言之循环语句详细介绍

    C语言之循环语句详细介绍

    大家好,本篇文章主要讲的是C语言之循环语句详细介绍,感兴趣的同学赶快来看一看吧,对你有帮助的话记得收藏一下,方便下次浏览
    2021-12-12
  • C++控制台版扫雷游戏

    C++控制台版扫雷游戏

    这篇文章主要为大家详细介绍了C++控制台版扫雷游戏,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2022-05-05
  • C语言中 int main(int argc,char *argv[])的两个参数详解

    C语言中 int main(int argc,char *argv[])的两个参数详解

    这篇文章主要介绍了C语言中 int main(int argc,char *argv[])的两个参数详解的相关资料,需要的朋友可以参考下
    2017-03-03
  • C语言实现无规律数据加密、解密功能

    C语言实现无规律数据加密、解密功能

    这篇文章主要为大家详细介绍了C语言实现无规律数据加密、解密功能,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2019-03-03
  • C++设计模式之装饰模式

    C++设计模式之装饰模式

    这篇文章主要介绍了C++设计模式之装饰模式,装饰模式能够实现动态的为对象添加功能,是从一个对象外部来给对象添加功能,需要的朋友可以参考下
    2014-10-10
  • 如何运用Capstone实现64位进程钩子扫描

    如何运用Capstone实现64位进程钩子扫描

    本章将通过Capstone引擎实现64位进程钩子的扫描,读者可使用此段代码检测目标进程内是否被挂了钩子,感兴趣的朋友跟随小编一起看看吧
    2024-08-08
  • 关于背包问题的一些理解和应用

    关于背包问题的一些理解和应用

    这篇文章主要介绍了关于背包问题的一些理解和应用,本文可以说是背包问题九讲的补充、读后感,需要的朋友可以参考下
    2014-08-08
  • c++冒泡排序详解

    c++冒泡排序详解

    冒泡排序(Bubble Sort),是一种计算机科学领域的较简单的排序算法。它重复地走访过要排序的数列,一次比较两个元素,如果他们的顺序错误就把他们交换过来。走访数列的工作是重复地进行直到没有再需要交换,也就是说该数列已经排序完成。
    2017-05-05
  • C++深入探究友元使用

    C++深入探究友元使用

    采用类的机制后实现了数据的隐藏与封装,类的数据成员一般定义为私有成员,成员函数一般定义为公有的,依此提供类与外界间的通信接口。但是,有时需要定义一些函数,这些函数不是类的一部分,但又需要频繁地访问类的数据成员,这时可以将这些函数定义为该类的友元函数
    2022-07-07
  • 全面了解结构体、联合体和枚举类型

    全面了解结构体、联合体和枚举类型

    下面小编就为大家带来一篇全面了解结构体、联合体和枚举类型。小编觉得挺不错的,现在就分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2016-07-07

最新评论