1. 场景

上一节实现了智能指针,其中的拷贝构造函数和赋值运算符是通过增加/减少指针的引用计数来操作的。但是如果是管理一个独占资源呢?我们希望在一个资源使用时被锁定,在使用完毕后被释放。

#include <mutex>

#include <thread>

#include <iostream>

using
namespace std;

mutex mu;

int rc=5;

void
thread1(){

//mu.lock();

rc+=5;

cout<<"thread1:"<<rc<<endl;

//mu.unlock();

}

void
thread2(){

//mu.lock();

rc-=5;

cout<<"thread2:"<<rc<<endl;

//mu.unlock();

}

int
main(){

thread th1(thread1);

thread th2(thread2);

th1.join();

th2.join();

}

在这里,我先把互斥代码去掉,编译运行后的结果是:

C:\Users\SkyFire\Desktop>g++ temp.cpp -std=c++11

C:\Users\SkyFire\Desktop>a

thread1:thread2:510

C:\Users\SkyFire\Desktop>a

thread1:thread2:105

C:\Users\SkyFire\Desktop>a

thread1:thread2:105

C:\Users\SkyFire\Desktop>a

thread1:thread2:510

每次的结果都不确定,因为没加互斥。

那么,把互斥加上:

#include <mutex>

#include <thread>

#include <iostream>

using
namespace std;

mutex mu;

int rc=5;

void
thread1(){

mu.lock();

rc+=5;

cout<<"thread1:"<<rc<<endl;

mu.unlock();

}

void
thread2(){

mu.lock();

rc-=5;

cout<<"thread2:"<<rc<<endl;

mu.unlock();

}

int
main(){

thread th1(thread1);

thread th2(thread2);

th1.join();

th2.join();

}

编译运行的结果是:

C:\Users\SkyFire\Desktop>g++ temp.cpp -std=c++11

C:\Users\SkyFire\Desktop>a

thread1:10

thread2:5

C:\Users\SkyFire\Desktop>a

thread1:10

thread2:5

C:\Users\SkyFire\Desktop>a

thread1:10

thread2:5

但是某些时候,我们可能会将unlock的动作漏写(百密一疏),如下面这种:

#include <mutex>

#include <thread>

#include <iostream>

using
namespace std;

mutex mu;

int rc=5;

void
thread1(){

mu.lock();

rc+=5;

cout<<"thread1:"<<rc<<endl;

//mu.unlock();

}

void
thread2(){

mu.lock();

rc-=5;

cout<<"thread2:"<<rc<<endl;

mu.unlock();

}

int
main(){

thread th1(thread1);

thread th2(thread2);

th1.join();

th2.join();

}

这样的结果就是thread2里面的语句一直得不到执行,程序死锁。

编译运行:

C:\Users\SkyFire\Desktop>g++ temp.cpp -std=c++11

C:\Users\SkyFire\Desktop>a

thread1:10

^C

C:\Users\SkyFire\Desktop>

可以看到,thread2一直没有执行,后面的^C是我使用Ctrl+C中断的结果。

为了避免这种情况,我们使用资源管理类。

  1. 简单的实现

一个简单的实现:

class AutoMutex{

private:

mutex &mu;

public:

AutoMutex(mutex &t):mu(t){

mu.lock();

}

~AutoMutex(){

mu.unlock();

}

};

这个类在构造的时候会将一个互斥量锁定,而在析构时会释放掉这个互斥量。乍一看好像没什么问题。事实上,在"正常的"情况下,这段代码可以工作的很好。

mutex mu;

void
mythread(){

AutoMutex t(mu);

cout<<"hello world"<<endl;

}

int
main(){

for(int i=0;i<10;++i)

thread(mythread).detach();

system("pause");

}

输出:

C:\Users\SkyFire\Desktop>g++ temp.cpp -std=c++11

C:\Users\SkyFire\Desktop>a

hello world

hello world

hello world

hello world

hello world

hello world

hello world

hello world

hello world

hello world

请按任意键继续. . .

  1. 问题

但是,如果出现一些比较调皮的程序员(暂定为小明吧)。

调皮的小明写出了如下的代码:

mutex mu;

mutex mu2;

void
mythread(){

AutoMutex t(mu);

AutoMutex t2(mu2);

t2=t;

cout<<"hello world"<<endl;

}

int
main(){

for(int i=0;i<10;++i)

thread(mythread).detach();

system("pause");

}

这TM就尴尬了……小明将管理了两个不同的mutex的对象相互赋值了。不过还好,这段代码是编译不通过的(小明的奸计未能得逞)。因为mutex类是不允许复制的,他的赋值运算符是删除的。(假设mutex可以复制,会产生什么?)

而且,管理两个mutex的对象的赋值没有任何意义,这个对象就是创建与销毁,并没有其他任何作用,所以,对于这个类,只要简单地把拷贝构造函数和赋值运算符屏蔽就好了:

class AutoMutex{

private:

const AutoMutex&
operator=(const AutoMutex&)=delete;

AutoMutex(const AutoMutex&)=delete;

mutex &mu;

public:

AutoMutex(mutex &t):mu(t){

mu.lock();

}

~AutoMutex(){

mu.unlock();

}

};

为了应对本宝宝的机智,小明又写出下面这段代码:

mutex mu;

void
mythread(){

AutoMutex t(mu);

AutoMutex t2(mu);

cout<<"hello world"<<endl;

}

int
main(){

for(int i=0;i<10;++i)

thread(mythread).detach();

system("pause");

}

不得不说,小明是很奸诈的~~~

一个互斥锁,对于一个线程来说,只有获取和没获取两种状态,而不存在获取两次这种状态。而不存在什么获取多次什么的状态。

我们先看一下,对于mutex,获取多次是个什么结果:

mutex mu;

void
mythread(){

mu.lock();

mu.lock();

cout<<"hello world"<<endl;

mu.unlock();

mu.unlock();

}

int
main(){

for(int i=0;i<10;++i)

thread(mythread).detach();

system("pause");

}

运行结果:

C:\Users\SkyFire\Desktop>g++ temp.cpp -std=c++11

C:\Users\SkyFire\Desktop>a

请按任意键继续. . .

既然mutex本身就是这么设计的,我们还是不改的好~~~

猜想mutex这样设计是为了提供PV锁机制:

下面这段代码,不加任何互斥:

int
main(){

cout<<1<<endl;

thread([](){cout<<3<<endl;}).detach();

cout<<2<<endl;

thread([](){cout<<4<<endl;}).detach();

cout<<5<<endl;

}

输出结果为:

C:\Users\SkyFire\Desktop>g++ temp.cpp -std=c++11

C:\Users\SkyFire\Desktop>a

1

3

2

54

完全没有顺序可言,但是如果加上一些互斥。

mutex mu;

int
main(){

cout<<1<<endl;

thread([](){cout<<3<<endl;mu.unlock();}).detach();

mu.lock();

cout<<2<<endl;

mu.lock();

thread([](){cout<<4<<endl;mu.unlock();}).detach();

mu.lock();

cout<<5<<endl;

mu.unlock();

}

此时的输出结果为:

C:\Users\SkyFire\Desktop>g++ temp.cpp -std=c++11

C:\Users\SkyFire\Desktop>a

1

2

3

4

5

Perfect!!!

这正是mutex为我们提供的特性,既然我们是管理mutex,我们就不该破坏这种特性。

于是~~~上面全是小明的错^_^。

这里实现的只是对mutex对象的管理,采用了禁止拷贝的方式,但是对其他对象的管理就不一定了,要根据对象的特性灵活管理。

常见的拷贝行为有:禁止拷贝(例如本类)、引用计数(例如上节的智能指针),但是要记住,如果实现了拷贝,一定要将所有元素全部拷贝。

EC笔记:第三部分:14、在资源管理类中小心Copying行为的更多相关文章

  1. Effective C++(14) 在资源管理类中小心copying行为

    问题聚焦:     上一条款所告诉我们的智能指针,只适合与在堆中的资源,而并非所有资源都是在堆中的.     这时候,我们可能需要建立自己的资源管理类,那么建立自己的资源管理类时,需要注意什么呢?. ...

  2. Effective C++ -----条款14: 在资源管理类中小心copying行为

    复制RAII对象必须一并复制它所管理的资源,所以资源的copying行为决定RAII对象的copying行为. 普遍而常见的RAII class copying行为是:抑制copying(使用私有继承 ...

  3. 【14】在资源管理类中小心copying行为

    1.为什么要使用资源管理类? 资源管理类的思路就是,栈上的对象,封装堆上分配的资源,确保一定会释放资源.auto_ptr和shared_ptr就是资源管理类,行为上像指针. 2.auto_ptr和sh ...

  4. 条款14:在资源管理类中小心copying行为

    请牢记: 1.复制RAII对象必须一并复制它所管理的资源,所以资源的copying行为决定RAII对象的copying行为. 2.普遍常见的RAII class copying行为是:抑制copyin ...

  5. [Effective C++ --014]在资源管理类中小心copying行为

    第一节 <背景> 条款13中讲到“资源取得的时机便是初始化时机”并由此引出“以对象管理资源”的概念.通常情况下使用std中的auto_ptr(智能指针)和tr1::shared_ptr(引 ...

  6. effective条款15,在资源管理类中小心copying行为

    class A { private: int *p; void lock(){ cout << p << "is lock" << endl; ...

  7. Effective C++ 条款13/14 以对象管理资源 || 在资源管理类中小心拷贝行为

    三.资源管理       资源就是一旦你使用了它,将来不用的时候必须归还系统.C++中最常用的资源就是动态内存分配.其实,资源还有 文件描述符.互斥器.图形界面中的字形.画刷.数据库连接.socket ...

  8. 读书笔记 effective c++ Item 15 在资源管理类中提供对原生(raw)资源的访问

    1.为什么需要访问资源管理类中的原生资源  资源管理类是很奇妙的.它们是防止资源泄漏的堡垒,没有资源泄漏发生是设计良好的系统的一个基本特征.在一个完美的世界中,你需要依赖这样的类来同资源进行交互,绝不 ...

  9. 读书笔记 effective c++ Item 14 对资源管理类的拷贝行为要谨慎

    1. 自己实现一个资源管理类 Item 13中介绍了 “资源获取之时也是初始化之时(RAII)”的概念,这个概念被当作资源管理类的“脊柱“,也描述了auto_ptr和tr1::shared_ptr是如 ...

随机推荐

  1. Consul-template的简单应用:配置中心,服务发现与健康监测

    简介 Consul-template是Consul的一个方扩展工具,通过监听Consul中的数据可以动态修改一些配置文件,大家比较热衷于应用在Nginx,HAProxy上动态配置健康状态下的客户端反向 ...

  2. Visual Studio 2013 添加一般应用程序(.ashx)文件到SharePoint项目

    默认,在用vs2013开发SharePoint项目时,vs没有提供一般应用程序(.ashx)的项目模板,本文解决此问题. 以管理员身份启动vs2013,创建一个"SharePoint 201 ...

  3. oracle常用函数及示例

    学习oracle也有一段时间了,发现oracle中的函数好多,对于做后台的程序猿来说,大把大把的时间还要学习很多其他的新东西,再把这些函数也都记住是不太现实的,所以总结了一下oracle中的一些常用函 ...

  4. Linux目录结构

  5. 我的MYSQL学习心得(四) 数据类型

    我的MYSQL学习心得(四) 数据类型 我的MYSQL学习心得(一) 简单语法 我的MYSQL学习心得(二) 数据类型宽度 我的MYSQL学习心得(三) 查看字段长度 我的MYSQL学习心得(五) 运 ...

  6. 跨平台的 .NET 运行环境 Mono 3.2 新特性

    Mono 3.2 发布了,对 Mono 3.0 和 2.10 版本的支持不再继续,而且这两个分支也不再提供 bug 修复更新. Mono 3.2 主要新特性: LLVM 更新到 3.2 版本,带来更多 ...

  7. Linux下部署ASP.NET服务连接oracle遇到的问题记录

    一.如何卸载MONO Q:mono是linux系统上跨平台软件,卸载它有两种方式: 1.知道mono安装路径,安装原来的路径直接覆盖安装(最为简单): 2.不知道mono安装路径,首先通过sudo f ...

  8. 舍弃Nunit拥抱Xunit

    前言 今天与同事在讨论.Net下测试框架的时候,说到NUnit等大多数测试框架的SetUp以及TearDown方法并不是显得那么完美,所以在公司内部的项目中采用了Xunit框架.那么究竟是什么样的原因 ...

  9. 一种简单的CQRS架构设计及其实现

    一.为什么要实践领域驱动? 近一年时间我一直在思考一个问题:"如何设计一个松耦合.高伸缩性.易于维护的架构?".之所以有这样的想法是因为我接触的不少项目都是以数据库脚本来实现业务逻 ...

  10. Vue.js——60分钟browserify项目模板快速入门

    概述 在之前的一系列vue.js文章,我们都是用传统模式引用vue.js以及其他的js文件的,这在开发时会产生一些问题. 首先,这限定了我们的开发模式是基于页面的,而不是基于组件的,组件的所有代码都直 ...