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

多线程 绕过mutex的保护

mutex,能够解决线程安全的问题,但它不是万能的。下面的例子虽然使用了mutex,但是恶意注入了一个外部函数,导致把被mutex保护的双向链表,让一个外部的指针指向了,结果就可以通过这个外部的指针操作被保护的双向链表,也就绕过了metex的保护。

例子:

  1. #include <list>
  2. #include <iostream>
  3. #include <mutex>
  4. #include <algorithm>
  5. #include <thread>
  6. #include <unistd.h>
  7. using namespace std;
  8. //是线程安全的
  9. class data_protect{
  10. public:
  11. list<int> alist{1,2};
  12. mutex m;
  13. public:
  14. void add_list(int val){
  15. //操作双向链表时,加锁了
  16. lock_guard<mutex> g(m);
  17. alist.push_back(val);
  18. }
  19. bool contains(int val){
  20. //操作双向链表时,加锁了
  21. lock_guard<mutex> g(m);
  22. return find(alist.begin(), alist.end(), val) != alist.end();
  23. }
  24. //参数接受一个外来的函数,并把双向链表传递给了外来的函数
  25. template<typename Function>
  26. void do_data(Function func){
  27. lock_guard<mutex> g(m);
  28. func(alist);
  29. }
  30. };
  31. list<int>* unprotect_list;
  32. //外来的函数接收到了双向链表的引用,并把这个双向链表暴露给外面的非保护的双向链表的指针
  33. void bad_func(list<int>& alist){
  34. unprotect_list = &alist;
  35. }
  36. void gl(data_protect& dp){
  37. dp.do_data(bad_func);
  38. //这个插入操作没有加锁,所以不是线程安全的
  39. unprotect_list->push_back(99);
  40. }
  41. int main(){
  42. data_protect dp;
  43. gl(dp);
  44. //alist被改变了
  45. for(auto& s : dp.alist){
  46. cout << s << endl;
  47. }
  48. }

为了避免上面的情况,最好遵守下面的准则:

不要将受保护的数据的指针或者引用传递到锁的范围之外,无论是通过函数返回它们,还是作为参数传递给用户提供的函数。

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号