锁用来在多线程访问同一个资源时防止数据竞险,保证数据的一致性访问。

多线程本来就是为了提高效率和响应速度,但锁的使用又限制了多线程的并行执行,这会降低效率,但为了保证数据正确,不得不使用锁,它们就是这样纠缠。

作为效率优先的c++开发人员,很多人谈锁色变。

虽然有很多的无锁技术应用到项目中来,但是还是很有必要对锁的技术有一个基础的理解。本文主要讨论c++11中的两种锁:lock_guard 和 unique_lock。

结合锁进行线程间同步的条件变量使用,请参考条件变量condition variable 。

lock_guard


lock_guard是一个互斥量包装程序,它提供了一种方便的RAII(Resource acquisition is initialization )风格的机制来在作用域块的持续时间内拥有一个互斥量。

创建lock_guard对象时,它将尝试获取提供给它的互斥锁的所有权。当控制流离开lock_guard对象的作用域时,lock_guard析构并释放互斥量。

它的特点如下:

  • 创建即加锁,作用域结束自动析构并解锁,无需手工解锁
  • 不能中途解锁,必须等作用域结束才解锁
  • 不能复制

示例代码如下:

#include <thread>
#include <mutex>
#include <iostream>int g_i = 0;
std::mutex g_i_mutex;  // protects g_ivoid safe_increment()
{const std::lock_guard<std::mutex> lock(g_i_mutex);++g_i;std::cout << std::this_thread::get_id() << ": " << g_i << '\n';// g_i_mutex is automatically released when lock// goes out of scope
}int main()
{std::cout << "main: " << g_i << '\n';std::thread t1(safe_increment);std::thread t2(safe_increment);t1.join();t2.join();std::cout << "main: " << g_i << '\n';
}

输出:

main: 0
140641306900224: 1
140641298507520: 2
main: 2

unique_lock


unique_lock是一个通用的互斥量锁定包装器,它允许延迟锁定,限时深度锁定,递归锁定,锁定所有权的转移以及与条件变量一起使用。

简单地讲,unique_lock 是 lock_guard 的升级加强版,它具有 lock_guard 的所有功能,同时又具有其他很多方法,使用起来更强灵活方便,能够应对更复杂的锁定需要。

特点如下:

  • 创建时可以不锁定(通过指定第二个参数为std::defer_lock),而在需要时再锁定
  • 可以随时加锁解锁
  • 作用域规则同 lock_grard,析构时自动释放锁
  • 不可复制,可移动
  • 条件变量需要该类型的锁作为参数(此时必须使用unique_lock)

示例代码:

#include <mutex>
#include <thread>
#include <chrono>struct Box {explicit Box(int num) : num_things{num} {}int num_things;std::mutex m;
};void transfer(Box &from, Box &to, int num)
{// don't actually take the locks yetstd::unique_lock<std::mutex> lock1(from.m, std::defer_lock);std::unique_lock<std::mutex> lock2(to.m, std::defer_lock);// lock both unique_locks without deadlockstd::lock(lock1, lock2);from.num_things -= num;to.num_things += num;// 'from.m' and 'to.m' mutexes unlocked in 'unique_lock' dtors
}int main()
{Box acc1(100);Box acc2(50);std::thread t1(transfer, std::ref(acc1), std::ref(acc2), 10);std::thread t2(transfer, std::ref(acc2), std::ref(acc1), 5);t1.join();t2.join();
}

总结


所有 lock_guard 能够做到的事情,都可以使用 unique_lock 做到,反之则不然。

那么何时使用lock_guard呢?很简单,

  • 需要使用锁的时候,首先考虑使用 lock_guard

它简单、明了、易读。如果用它完全ok,就不要考虑其他了。

如果现实不允许,就让实力派 unique_lock 出马吧!

参考资料

std::lock_guard
std::unique_lock
Difference between std::lock_guard and std::unique_lock

c++11中的lock_guard和unique_lock使用浅析相关推荐

  1. C++11中std::lock_guard的使用

    互斥类的最重要成员函数是lock()和unlock().在进入临界区时,执行lock()加锁操作,如果这时已经被其它线程锁住,则当前线程在此排队等待.退出临界区时,执行unlock()解锁操作.更好的 ...

  2. Multi-thread--C++11中std::lock_guard的使用

    互斥类的最重要成员函数是lock()和unlock().在进入临界区时,执行lock()加锁操作,如果这时已经被其它线程锁住,则当前线程在此排队等待.退出临界区时,执行unlock()解锁操作.更好的 ...

  3. C++11中std::condition_variable的使用

    <condition_variable>是C++标准程序库中的一个头文件,定义了C++11标准中的一些用于并发编程时表示条件变量的类与方法等. 条件变量是并发程序设计中的一种控制结构.多个 ...

  4. C++11中头文件thread的使用

    C++11中加入了<thread>头文件,此头文件主要声明了std::thread线程类.C++11的标准类std::thread对线程进行了封装.std::thread代表了一个线程对象 ...

  5. Multi-thread--C++11中std::condition_variable的使用

    <condition_variable>是C++标准程序库中的一个头文件,定义了C++11标准中的一些用于并发编程时表示条件变量的类与方法等. 条件变量是并发程序设计中的一种控制结构.多个 ...

  6. Multi-thread--C++11中thread的使用

    C++11中加入了<thread>头文件,此头文件主要声明了std::thread线程类.C++11的标准类std::thread对线程进行了封装.std::thread代表了一个线程对象 ...

  7. C++之lock_guard和unique_lock自动加解锁区别

    一.std::unique_lock也可以提供自动加锁.解锁功能,比std::lock_guard更加灵活.1.std::lock_guard std::lock_guard是RAII模板类的简单实现 ...

  8. Visual C++ 11 中新的并发功能

    最新的 C++ 迭代(称为 C++11,在去年通过了国际标准化组织 (ISO) 的审批)形式化了一组新库和一些保留字以处理并发. 许多开发者以前都在 C++ 中使用过并发功能,但都是通过第三方的库,即 ...

  9. lock_guard和unique_lock

    锁 锁用来在多线程访问同一个资源时防止数据竞险,保证数据的一致性访问.多线程本来就是为了提高效率和响应速度,但锁的使用又限制了多线程的并行执行,这会降低效率,但为了保证数据正确,不得不使用锁,它们就是 ...

最新文章

  1. 优雅的理解 call 和 apply 的使用方法
  2. Android Studio 模拟器 不要皮肤,效果更好
  3. 中电福富外包offer要等多久_校招|双非本科,非科班,自学JAVA 1年时间终于斩获offer!...
  4. 利用Inception-V3训练的权重微调,实现猫狗分类(基于keras)
  5. 7-5 列出连通集 (6 分)
  6. 散粉在哪个步骤用_底妆的正确步骤是什么?花三分钟带你破译底妆密码
  7. python经济统计学论文_统计学论文
  8. 坐标转换-换带计算(附软件下载)
  9. python二维游戏编程 最强大脑游戏_看完《最强大脑》,我决定用Python做这个游戏...
  10. linux如何磁盘管理工具下载,Linux_Linux系统的磁盘空间管理工具, 1.fdisk -lDis - phpStudy...
  11. modelsim 波形设置显示时间单位
  12. 每天,每周,每月每隔cron表达式
  13. M1 Mac 开发Android遇到的坑点与我的解决方法
  14. 橘子与萝卜同食容易导致甲状腺肿(图)
  15. Hibernate 关联关系解除
  16. android高分段进阶攻略(9)——ViewPager补间动画实现京东广告Banner
  17. C#如何获取本机IP地址,两种方法
  18. 基于C/C++获取电脑网卡的IP地址信息
  19. 法拉帝港股上市:市值76亿港元 9个月交付179艘游艇
  20. hdu 6656 Final Exam

热门文章

  1. 人工智能发展简史——未来是属于AI人工智能的。
  2. 第七届蓝桥杯——生日蜡烛
  3. qchart画完以后删除_国画山水精典学堂:泼墨山水画
  4. mac vscode vim 中文输入法搜狗问题处理
  5. crossover安装oracle,linux通过CrossOver安装最新版本TIM
  6. 3.3 SQL---单表查询
  7. 华为ENSP中AP与AC的配置
  8. html+css网页开发实战——1、头部背景和文字的制作
  9. zzu数学 实验十一最速降线
  10. 每公里配速9分18秒,双足机器人完成5公里慢跑