c++ 互斥量的常见问题(未保护共享数据)

namespace mutex_invalid {
    void func_asign(void(*func)(int&), int& data) {
        std::cout << "in func: " << data << std::endl;
        func(data);   // data 传入的本意可能只是在线程中使用, 而不是修改
    }

    void func_asign_test(int& num) {
        num++;   // data 被恶意回调函数修改
    }

    void test_main() {
        std::mutex lock;
        int n = 1;
        std::lock_guard<std::mutex> l(lock);
        std::thread t(func_asign, func_asign_test, std::ref(n));
        t.join();
        //func_asign(func_asign_test, num);

        std::cout << "in test_main: " << n << std::endl;
    }
}

运行结果
在这里插入图片描述

在该段代码中, 看似给可访问的数据标记互斥, 但实际根本没有保护数据, 把数据的可访问性暴露在回调函数 func 中, 用户如果传入恶意回调函数就会导致共享数据被修改, 不仅仅是回调, 在类成员函数等其他方式也会出现诸如此类的问题

规避该现象的方法是, 避免把保护数据的指针或引用传递到互斥作用域之外

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
C++中,进程通信是操作系统级别的一种机制,允许不同进程之间的数据交换和协调执行。信号(Semaphore)是一种同步原语,用于控制多个进程对共享资源的访问。它是一种计数型对象,可以用来进行互斥访问、同步操作或者作为线程池中的任务数限制。 信号主要有以下几个方面: 1. **基本概念**:信号由两个部分组成,一个值(通常是整数)和一个计数器。当一个进程请求使用资源时,会检查信号的值。如果大于0,值减一并继续;如果为0,则阻塞直到其他进程释放。当一个进程完成任务并释放资源时,信号值加一。 2. **互斥**:信号常用于实现进程间的互斥,确保在同一时间只有一个进程可以访问某个临界区(共享资源)。 3. **二进制信号**:最常见的是二进制信号,其值只有两种状态:0(等待)或1(可访问),这使得信号成为简单粗暴的同步工具。 4. **二进制信号示例**: ```cpp #include <semaphore.h> sem_t semaphore; sem_init(&semaphore, 0, 1); // 初始化,初始值为1,表示资源可用 sem_wait(&semaphore); // 请求资源,若不可用则阻塞 // 在这里执行临界区代码 sem_post(&semaphore); // 释放资源,使其他等待进程可以继续 ``` 5. **相关问题--:** 1. 除了二进制信号,还有哪些类型的信号? 2. 信号如何处理多个进程的并发请求? 3. 信号在高并发场景下有何优势和局限性? 如果你对信号或其他特定类型或高级用法感兴趣,请继续提问。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值