经验首页 前端设计 程序设计 Java相关 移动开发 数据库/运维 软件/图像 大数据/云计算 其他经验
当前位置:技术经验 » 程序设计 » C » 查看文章
c/c++ 多线程 unique_lock的使用
来源:cnblogs  作者:小石王  时间:2018/11/6 10:17:44  对本文有异议

多线程 unique_lock的使用

unique_lock的特点:

1,灵活。可以在创建unique_lock的实例时,不锁,然后手动调用lock_a.lock()函数,或者std::lock(lock_a, …),来上锁。当unique_lock的实例被析构时,会自动调用unlock函数,释放锁。

  1. unique_lock<mutex> lock_a(d1.m, std::defer_lock);

2,unique_lock的实例可以调用unlock函数。这个意味着,在unique_lock的实例销毁前,你可以有选择的在程序的分支释放锁。持有锁的时间比所需时间更长,可能会导致性能下降,因为其他等待该锁的线程,被阻止运行的时间超过了所需的时间。

注意:有个弊端,当不拥有锁的时候,调用了unlock成员方法,程序崩溃。崩溃信息如下:

  1. terminate called after throwing an instance of 'std::system_error'
  2. what(): Operation not permitted
  3. Aborted (core dumped)

3,可以在作用域之间转移锁的所有权。右值的话,会自动被转移;左值的话,必须手动调用std::move()函数,来进行锁的所有权的转移。

通常使用这种模式,是在待锁定的互斥元依赖于程序的当前状态,或者依赖于传递给返回std::unique_lock对象的函数的地方。

例子:关于上述第一点和第二点的

  1. #include <list>
  2. #include <iostream>
  3. #include <mutex>
  4. #include <algorithm>
  5. #include <thread>
  6. #include <unistd.h>
  7. using namespace std;
  8. class data_protect;
  9. void swap(data_protect& , data_protect& );
  10. //是线程安全的
  11. class data_protect{
  12. friend void swap(data_protect& , data_protect& );
  13. private:
  14. list<int> alist{1,2};
  15. mutex m;
  16. public:
  17. void add_list(int val){
  18. //操作双向链表时,加锁了
  19. lock_guard<mutex> g(m);
  20. alist.push_back(val);
  21. }
  22. bool contains(int val){
  23. //操作双向链表时,加锁了
  24. lock_guard<mutex> g(m);
  25. return find(alist.begin(), alist.end(), val) != alist.end();
  26. }
  27. };
  28. void swap(data_protect& d1, data_protect& d2){
  29. //if(d1 == d2) return;
  30. //造成死锁
  31. //d1.add_list(11);
  32. unique_lock<mutex> lock_a(d1.m, std::defer_lock);
  33. unique_lock<mutex> lock_b(d2.m, std::defer_lock);
  34. std::lock(lock_a, lock_b);
  35. swap(d1.alist, d2.alist);
  36. //有unlock成员函数,并可以手动调用unlock函数
  37. //如果没有持有锁,就调用unlock成员函数就会导致程序崩溃。所以要检查是否拥有锁。
  38. if(lock_a.owns_lock() && lock_b.owns_lock()){
  39. lock_a.unlock();
  40. lock_b.unlock();
  41. }
  42. }
  43. int main(){
  44. data_protect d1, d2;
  45. swap(d1, d2);
  46. d2.add_list(11);
  47. }

github源代码

例子:关于上述的第三点

  1. #include <mutex>
  2. std::unique_lock<std::mutex> getlock(){
  3. std::mutex sm;
  4. std::unique_lock<std::mutex> lk(sm);
  5. //prepare_data();
  6. return lk;//因为lk是右值,所以自动调用了std::move函数,把锁的所有权转移了出去。
  7. }
  8. void process_data(){
  9. std::unique_lock<std::mutex> lk(getlock());
  10. //do_something();
  11. }
  12. int main(){
  13. process_data();
  14. }

github源代码

c/c++ 学习互助QQ群:877684253

本人微信:xiaoshitou5854

 友情链接:直通硅谷  点职佳  北美留学生论坛

本站QQ群:前端 618073944 | Java 606181507 | Python 626812652 | C/C++ 612253063 | 微信 634508462 | 苹果 692586424 | C#/.net 182808419 | PHP 305140648 | 运维 608723728

W3xue 的所有内容仅供测试,对任何法律问题及风险不承担任何责任。通过使用本站内容随之而来的风险与本站无关。
关于我们  |  意见建议  |  捐助我们  |  报错有奖  |  广告合作、友情链接(目前9元/月)请联系QQ:27243702 沸活量
皖ICP备17017327号-2 皖公网安备34020702000426号