C++没有类似Java、C#等语言的垃圾回收机制,内存管理是最为头痛的工作。

new、delete以及指针的不恰当运用是C++中造成资源获取/释放问题的根源。

智能指针是解决这些问题的一种方案,boost.smart_ptr库提供了六种智能指针,包括:

scoped_ptrscoped_arrayshared_ptrshared_arrayweak_ptrintrusive_ptr

它们都是轻量级对象,速度与原始指针相差无几,都是异常安全的。

要使用smart_ptr组件,需要包含头文件 #include <boost/smart_ptr.hpp>

shared_ptr是一个最像指针的“智能指针”,是boost.smart_ptr库中最有价值、最重要的组成部分。已收入了C++11标准。

shared_ptr是引用计数的智能指针,可以被自由的拷贝和赋值,在任意地方共享它,当没有代码使用(引用计数为0)时删除被包装的动态分配的对象。

shared_ptr可以安全的放到标准容器中,是在STL容器中存储指针的最标准解法。

应用举例:

#include "stdafx.h"
#include <iostream>
#include <cassert>
#include <string>
#include <vector>
#include <boost/smart_ptr.hpp> using std::cout;
using std::endl;
using std::string;
using std::vector; int _tmain(int argc, _TCHAR* argv[])
{
boost::scoped_ptr<string> pstr(new string("scoped_ptr test"));
assert(pstr);
assert(pstr != nullptr);
*pstr = "hello test";
cout << *pstr << ": " << pstr->size(); boost::scoped_ptr<string> another_p_str;
assert(!another_p_str); if (NULL == another_p_str)
{
cout << "\nNull pointer!\n";
} boost::shared_ptr<string> sps(boost::make_shared<string>("shared_ptr_test"));
cout << "\n" << sps->c_str() << endl;
boost::shared_ptr<string> sps2(sps);
*sps2 = "shared_ptr is smart!";
cout << *sps << endl; assert(sps == sps2); //shared_ptr用于容器
typedef vector<boost::shared_ptr<int>> VSI;
VSI v;
v.push_back(boost::make_shared<int>(1));
v.push_back(boost::make_shared<int>(2));
v.push_back(boost::make_shared<int>(3));
v.push_back(boost::make_shared<int>(4));
v.push_back(boost::make_shared<int>(5)); cout << "\nVevtor: ";
boost::shared_ptr<int> pv = v[0];
*pv = 1000;
for (size_t i = 0; i < v.size(); ++i)
{
cout << *v[i] << " ";
}
return 0;
}

在开发实践中,个人觉的最应当使用智能指针的两种场合:

(1) 应用于标准容器

在使用容器管理大量对象指针的时候,必须编写额外的大量代码来保证指针最终被正确释放,通常很麻烦而且容易出错。将shared_ptr作为容器的元素,如vector<shared_ptr<T> >,与存储原始指针的容器所能实现的功能几乎一样,而不用担心资源泄漏。

(2) 应用于工厂模式

在程序中编写自己的工厂类或者工厂函数时,通常需要在堆上使用new动态分配一个对象然后返回对象的指针。这种做法很不安全,应为很容易忘记对指针调用delete。

使用shared_ptr可以解决这个问题,只需要修改工厂方法的接口,不再返回一个原始指针,而是返回一个被shared_ptr包装的智能指针,这样可以很好地保护系统资源,而且会更好地控制对接口的使用。

使用代码来解释shared_ptr应用于工厂模式的用法:

#ifndef OPERATION_H_
#define OPERATION_H_
#include <boost/smart_ptr.hpp> enum OperationType
{
ADD,
SUB,
MUL,
DIV
}; //运算基类
class Operation
{
public:
virtual double GetResult() const = 0; Operation()
: m_Operand1(0.0)
, m_Operand2(0.0)
{ } void SetOperrand1(const double operand1)
{
m_Operand1 = operand1;
}
double GetOperand1() const
{
return m_Operand1;
} void SetOperrand2(const double operand2)
{
m_Operand2 = operand2;
}
double GetOperand2() const
{
return m_Operand2;
} protected:
virtual ~Operation(){} protected:
double m_Operand1;
double m_Operand2;
};
//加法
class Add : public Operation
{
public:
double GetResult() const override;
};
//减法
class Sub : public Operation
{
public:
double GetResult() const override;
};
//乘法
class Mul : public Operation
{
public:
double GetResult() const override;
};
//除法
class Div : public Operation
{
public:
double GetResult() const override;
}; //工厂类
class OperationFactory
{
public:
static boost::shared_ptr<Operation> CreateOperation(OperationType type)
{
boost::shared_ptr<Operation> oper;
switch (type)
{
case ADD:
oper = boost::make_shared<Add>();
break;
case SUB:
oper = boost::make_shared<Sub>();
break;
case MUL:
oper = boost::make_shared<Mul>();
break;
case DIV:
oper = boost::make_shared<Div>();
break;
default: ;
}
return oper;
}
}; #endif

调用代码:

//通过工厂方法创建对象
auto pOper = OperationFactory::CreateOperation(MUL);
if (nullptr != p)
{
pOper->SetOperrand1(100); //可以像普通指针一样使用
pOper->SetOperrand2(123);//不必担心内存泄漏,shared_ptr会自动管理指针
cout << "\n" << pOper->GetResult();
}

C++11标准中std::shared_ptr,功能与boost::shared_ptr基本相同,完全可以定价互换。

包含头文件#incude <memory>

最新文章

  1. JavaMail发送邮件的小程序
  2. CSS Float 以及相关布局模式
  3. 通过js看类似C#中的回掉
  4. poj: 1005
  5. 一目了然c++虚基类!
  6. Android中Cursor(游标)类的概念和用法
  7. 代码优化—From &lt;effective C++&gt;
  8. [Ecmall]ECMALL目录结构设置与数据库表
  9. 保持与 Microsoft Azure Files 的连接
  10. scala-spark练手--dataframe数据可视化初稿
  11. Cocos2dx 3.0 交流篇
  12. 【linux之简介】
  13. 洛谷 [P2024] 食物链
  14. Unity TimeLine 资源结构
  15. 入坑C++之vs 新建C++项目
  16. AJAX跨域问题以及解决思路(更新中)
  17. Redis使用sortedset缓存IP段数据
  18. Meandering Through the Maze of MFC Message and Command Routing MFC消息路由机制分析
  19. oracle:the password has expired
  20. 线上问题定位--CPU100%

热门文章

  1. vue工具 - vue-cli安装使用流程
  2. python之Tkinter控件学习
  3. howdoi 简单分析
  4. oracle dblink 查询 tns:无法解析指定的连接标识符
  5. centos 6.6编译安装nginx
  6. html5 canvas时钟
  7. PHP直接输出一张图片
  8. thinkphp结合layui上传视频
  9. 找到多个与名为“xxx”的控制器匹配的类型。如果为此请求(“{controller}/{action}/{id}”)提供服务的路由没有指定命名空间以搜索与此请求相匹配的控制器,则会发生这种情况。
  10. PAT甲1031 Hello World for U【字符串】