首 页 ┆ 源码下载 ┆ IT学院 ┆ 字体下载 ┆ 模板下载 ┆ 源码发布 ┆ 广告合作 ┆ 网站地图
► 设为首页
► 加入收藏
► 联系我们
源码下载 >> ASP源码 | PHP源码 | ASP.net源码 | JSP源码 | CGI源码 | VC/C++源码 | VB源码 | Delphi源码 | Flash源码
文章学院 >> 网络编程 | 网页设计 | 图形图象 | 数据库 | 服务器 | 网络媒体 | 网络安全 | 操作系统 | 办公软件 | 软件开发 | 黑客知识
字体下载 >> 精制字体 | 非英字体 | 艺术字体 | 著名字体 | 哥特式 | 简单字体 | 手写体 | 节假日 | 图案字体 | 精度像素 | 中文字体
模板下载 >> 企业门户 | 数码网络 | 休闲娱乐 | 影视音乐 | 旅游名胜 | 文化艺术 | 电子商务 | 个性展示 | 登陆导航 | Flash模板
►►您当前的位置:源码园 → IT学院 → 软件开发 → VC编程 → 文章内容

用 auto_ptr 类模板帮助动态内存管理

作者:佚名  来源:网上收集  发布时间:2007-3-27 10:04:28

用auto_ptr类模板帮助动态内存管理

赵湘宁

    动态内存使用最多的是在C++应用程序的代码中。有过编程经验的程序员虽然都知道new操作符的使用一定要与delete匹配,在某些场合仍然可能有内存溢出。当异常被掷出时,程序的正常控制流程被改变,因此导致潜在的内存溢出。例如,
void g() //可能掷出  {    if (some_condition == false)      throw X();  }  void func()   {    string * pstr = new string;    g(); //如果 g 掷出一个异常,内存溢出    delete pstr; //如果 g 掷出一个异常,则此行为不能达到的代码行。  }  int main()  {    try    {      func();    }    catch(...)    {}  }
    当 g 掷出一个异常,异常处理机制展开堆栈:g()退出,同时控制被转移到 main() 的 catch(...)代码块。这时,无论怎样,func()中的delete语句都不会被执行,由此导致pstr的内存溢出。要是使用局部自动串变量,而不是使用动态分配-内存溢出就不会出现了:

string str; //局部自动对象
g(); //没有内存溢出

    许多数据重要的结构以及应用,象链表,STL容器,串,数据库系统以及交互式应用必须使用动态内存分配,因此仍然冒着万一发生异常导致内存溢出的风险。C++标准化委员会意识到了这个漏洞并在标准库中添加了一个特殊的类模板,它就是std::auto_ptr,其目的是促使动态内存和异常之前进行平滑的交互。Auto_ptr保证当异常掷出时分配的对象(即:new操作符分配的对象)能被自动销毁,内存能被自动释放。下面我们就来讨论使用动态内存时,如何正确和有效地使用auto_ptr来避免资源溢出。这个技术适用于文件,线程,锁定以及与此类似的资源。

    Auto_ptr的定义可以在<memory.h>中找到。与标准库中其它的成员一样,它被声明在命名空间std::中。当你实例化auto_ptr对象时,对它进行初始化的方法是用一个指针指向动态分配的对象,下面是实例化和初始化auto_ptr对象的例子:

include <memory>  #include <string>  using namespace std;  void func()  {    auto_ptr<string> pstr (new string); /* 创建并初始化auto_ptr */  }
    auto_ptr 后面的尖括弧里指定 auto_ptr 指针的类型,在这个例子中是 string。然后  auto_ptr 句柄的名字,在这个例子中是pstr。最后是用动态分配的对象指针初始化这个实例。注意你只能使用auto_ptr构造器的拷贝,也就是说,下面的代码是非法的:
auto_ptr<string> pstr  = new string; //编译出错
    Auto_ptr是一个模板,因此它是完全通用的。它可以指向任何类型的对象,包括基本的数据类型:
auto_ptr<int> pi  (new int);
    一旦你实例化一个auto_ptr,并用动态分配的对象地址对它进行了初始化,就可以将它当作普通的对象指针使用,例如:
*pstr = "hello world"; //赋值  pstr->size(); //调用成员函数
    之所以能这样做是因为auto_ptr重载了操作符&,*和->。不要被语法误导,记住pstr是一个对象,不是一个指针。

    auto_ptr是如何解决前面提到的内存溢出问题呢?auto_ptr的析构函数自动摧毁它绑定的动态分配对象。换句话说,当pstr的析构函数执行时,它删除构造pstr期间创建的串指针。你绝不能删除auto_ptr,因为它是一个本地对象,它的析构函数是被自动调用的。让我们看一下函数func()的修订版本,这次使用了auto_ptr:
void func()   {    auto_ptr<string>  pstr (new string);    g();  //如果g()掷出异常,pstr 被自动摧毁   }
   
C++保证在堆栈展开过程中,自动存储类型的对象被自动摧毁。因此,如果g()掷出异常,pstr的析构函数将会在控制被转移到catch(...)块之前执行。因为pstr的析构函数删除其绑定的串指针,所以不会有内存溢出发生。这样我们在使用动态分配对象时,利用auto_ptr就实现了自动和安全的本地对象。

如何避免使用auto_ptr的缺陷

    auto_ptr并不是完美无缺的,它的确很方便,但也有缺陷,在使用时要注意避免。首先,不要将auto_ptr对象作为STL容器的元素。
C++标准明确禁止这样做,否则可能会碰到不可预见的结果(在另文中讨论)。

    auto_ptr的另一个缺陷是将数组作为auto_ptr的参数:
auto_ptr<char>  pstr (new char[12] ); //数组;为定义
    记住不管什么时候使用数组的 new 操作时,必须要用 delete[] 来摧毁数组。因为 auto_ptr 的析构函数只对非数组类型起作用。所以数组是不能被正确摧毁的话,程序的行为是不明确的。总之,auto_ptr 控制一个由 new 分配的单对象指针,仅此而已。

[] [返回上一页] [打 印]
  • 上一篇文章:在预先定义的内存位置构造一个对象
  • 下一篇文章:初始化类对象

  • 相关文章:
  • CORBA对象生命周期之实现和内存管理
  • [图文]代码最优化.NET中的内存管理
  • 用 auto_ptr 类模板帮助动态内存管理
  • 读核日记(八) --linux的内存管理机制(2)
  • 读核日记(七) --linux的内存管理机制(1)
  • Linux 核心--4.内存管理
  • Linux管理员手册(4)--内存管理
  • [图文]Linux编程白皮书 第二章 内存管理
  • [图文]Linux编程白皮书 第二章 内存管理 2.1.1 请求...
  • 改善SQL Server内存管理
  • 深入SQL SERVER 2000的内存管理机制(二)
  • 深入SQL SERVER 2000的内存管理机制(三)
关于本站 - 网站帮助 - 广告合作 - 下载声明 - 友情连接 - 网站地图 - 源码发布
Copyright © 2003-2009 Ymyasp.Com. All Rights Reserved .
备案序号:粤ICP备07029071号