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>

boost--smart_ptr库的更多相关文章

  1. 初探boost之smart_ptr库学习笔记

    概述 Boost.smart_ptr库提供了六种智能指针,除了shared_ptr 和 weak_ptr 以外还包含 scoped_ptr .scoped_array . shared_array . ...

  2. [boost] : test库

    最小化的测试套件minimal_test test库提供一个最小化的测试套件minimal_test, 类似lightweight_test适合入门级测试. 需要包含文件文#include <b ...

  3. [boost] : lightweight_test库

    lightweight_test轻量级单元测试框架, 只支持最基本的单元测试, 不支持测试用例, 测试套件的概念, 简单小巧, 适合要求不高或者快速测试的工作. 基本用法 需要包含头文件#includ ...

  4. 如何在WINDOWS下编译BOOST C++库 .

    如何在WINDOWS下编译BOOST C++库 cheungmine 2008-6-25   写出来,怕自己以后忘记了,也为初学者参考.使用VC8.0和boost1.35.0.   1)下载boost ...

  5. Windows下如何使用BOOST C++库 .

    Windows下如何使用BOOST C++库 我采用的是VC8.0和boost_1_35_0.自己重新编译boost当然可以,但是我使用了 http://www.boostpro.com/produc ...

  6. Boost线程库学习笔记

    一.创建一个线程 创建线程 boost::thread myThread(threadFun); 需要注意的是:参数可以是函数对象或者函数指针.并且这个函数无参数,并返回void类型. 当一个thre ...

  7. Boost正则表达式库regex常用search和match示例 - 编程语言 - 开发者第2241727个问答

    Boost正则表达式库regex常用search和match示例 - 编程语言 - 开发者第2241727个问答 Boost正则表达式库regex常用search和match示例 发表回复   Boo ...

  8. Boost::thread库的使用

    阅读对象 本文假设读者有几下Skills [1]在C++中至少使用过一种多线程开发库,有Mutex和Lock的概念. [2]熟悉C++开发,在开发工具中,能够编译.设置boost::thread库. ...

  9. 一起学习Boost标准库--Boost.StringAlgorithms库

    概述 在未使用Boost库时,使用STL的std::string处理一些字符串时,总是不顺手,特别是当用了C#/Python等语言后trim/split总要封装一个方法来处理.如果没有形成自己的com ...

  10. 一起学习Boost标准库--Boost.texical_cast&format库

    今天接续介绍有关字符串表示相关的两个boost库: lexical_cast 将数值转换成字符串 format 字符串输出格式化 首先,介绍下lexical_cast ,闻其名,知其意.类似C中的at ...

随机推荐

  1. 原生js--表单

    阅读了<javascript权威指南>P396-P409. 一.表单和表单元素的选取 1.选取表单(包含name=“address”属性的form表单) document.querySel ...

  2. Android Studio 第一次启动配置

    第一次启动AS前,为了避免重新下载新版本的SDK 操作如下: AS启动前,请先将bin目录的idea.properties文件中增加一行:disable.android.first.run=true ...

  3. LeetCode 30 Substring with Concatenation of All Words(确定包含所有子串的起始下标)

    题目链接: https://leetcode.com/problems/substring-with-concatenation-of-all-words/?tab=Description   在字符 ...

  4. Visual Studio 2013附加进程调试IE加载的ActiveX Control无效解决方法

    默认Attach to选择了Automatically determine the type of code to debug,显示Native Code.但附加进程到iexplore.exe断点无法 ...

  5. 一辈子只有1次成为BAT的机会,你如何把握?

    本文转自:http://www.fmi.com.cn/index.php?m=content&c=index&a=show&catid=9&id=614308 感谢作者 ...

  6. [通信] C# TCP实现多个客户端与服务端 数据 与 文件的传输

    说明: http://download.csdn.net/detail/chwei_cson/4423874 源码: http://download.csdn.net/download/meicanj ...

  7. C++ 输入/输出

    std:: 是什么?有什么作用? 输入和输出的iostream 库.iostream 库的基础是两种命名为 istream 和 ostream 的类型,分别表示输入流和输出流. 标准库定义了 4 个 ...

  8. iOS - 处理计算精度要求很高的数据,floatValue,doubleValue等计算不精确问题

    .问题描述:服务器返回的double类型9...94的数字时 .之前处理方式是 :(从内存.cpu计算来说double都是比较合适的,一般情况下都用double) goodsPrice.floatVa ...

  9. RabbitMQ服务端配置详解(转自:http://www.cnblogs.com/zhen-rh/p/6884297.html)

    RabbitMQ支持三种配置方式: 1) 读取环境变量中配置, 这包括shell中环境变量和rabbitmq-env.conf/rabbitmq-env-conf.bat文件中配置的环境变量 可配置如 ...

  10. C#控制台窗口居中显示(转)

    private struct RECT { public int left, top, right, bottom; } [DllImport("kernel32.dll", Se ...