当前位置: 首页 > news >正文

C++ —— 线程安全

C++ —— 线程安全

  • 线程安全的基本概念
    • 定义
    • 共享资源和临界区
  • 三个概念的介绍
  • 如何保证线程安全
    • volatile 关键字

线程安全的基本概念

定义

个线程同时访问同一个资源(例如变量、数据结构或对象)时,如果程序通过适当的同步机制保证这些访问不会引起数据损坏不一致性,那么这段代码就是线程安全的。也就是说,无论线程如何调度执行,其结果都与单线程环境下的行为一致

共享资源和临界区

多个线程同时操作的共享资源(如全局变量、静态变量、共享内存等)如果没有保护,就会产生竞争条件(race condition)。通常,访问共享资源代码段称为临界区,需要特殊的机制来保证同一时间只有一个线程能进入。

示例代码:

#include <iostream>
#include <thread>
using namespace std;

// 线程任务函数
// 普通函数
void func (int n, const string& s) {
    for (int i = 1; i <= 5; i++) {
        cout << "No." << i << ", n = " << n << ", s = " << s << endl;
        this_thread::sleep_for(chrono::seconds(1));
    }
}

// lambda函数
auto f = [] (int n, const string& s) {
    for (int i = 1; i <= 5; i++) {
        cout << "No." << i << ", n = " << n << ", s = " << s << endl;
        this_thread::sleep_for(chrono::seconds(1));
    }
};

// 仿函数
class mythread_1 {
public:
    void operator() (int n,const string& s) {
        for (int i = 1; i <= 5; i++) {
            cout << "No." << i << ", n = " << n << ", s = " << s << endl;
            this_thread::sleep_for(chrono::seconds(1));
        }
    }
};

// 类的静态成员函数
class mythread_2 {
public:
    static void func (int n, const string& s) {
        for (int i = 1; i <= 5; i++) {
            cout << "No." << i << ", n = " << n << ", s = " << s << endl;
            this_thread::sleep_for(chrono::seconds(1));
        }
    }
};

// 类的非静态成员函数
class mythread_3 {
public:
    void func (int n, const string& s) {
        for (int i = 1; i <= 5; i++) {
            cout << "No." << i << ", n = " << n << ", s = " << s << endl;
            this_thread::sleep_for(chrono::seconds(1));
        }
    }
};

int main () {
    thread t1(func, 1, "普通函数 t1");
    thread t2(func, 2, "普通函数 t2");
    thread t3(f, 3, "lambda函数 t3");
    thread t4(mythread_1(), 4, "仿函数 t4");
    thread t5(mythread_2::func, 5, "静态成员函数 t5");
    mythread_3 mt;
    thread t6(&mythread_3::func, &mt, 6, "非静态成员函数 t6");

    cout << "=== start task ===" << endl;
    for (int i = 1; i <= 5; i++) {
        cout << "" << endl;
        this_thread::sleep_for(chrono::seconds(1));
    }
    cout << "=== task completed ===" << endl;

    t1.join();
    t2.join();
    t3.join();
    t4.join();
    t5.join();
    t6.join();

    return 0;
}

编译运行这段代码,你会发现打印出来的信息有些奇怪。这是因为,cout全局对象。在目前这个程序中,所有线程共享cout对象,每个线程都会利用cout向屏幕发送数据。如果同一时间点多个线程使用cout,打印出来的内容肯定会有些奇怪。
再看一段代码:

#include <iostream>
#include <thread>
using namespace std;

int res = 0; // 定义全局变量 res

void func_add () {
    for (int i = 1; i <= 1000000; i++) {
        res++;
    }
}

int main () {
    thread t1(func_add);
    thread t2(func_add);

    t1.join();
    t2.join();

    cout << "now res = " << res << endl;

    return 0;
}

我的PC某次运行的结果如下:

now res = 1398091

这段代码使用了两个线程t1t2并行执行func_add 数,因此会并行对全局变量res进行操作。由于res是一个全局变量,两个线程在对其进行修改时可能会发生竞争条件,即两个线程可能会同时读取和修改res,从而导致res的值不准确。不管运行多少次,结果都不会是2000000

三个概念的介绍

  1. 顺序性
    程序按照代码的先后顺序执行
    CPU为了提高程序整体的执行效率,可能会对代码进行优化,按照更高的顺序执行。
  • 顺序性指的是:CPU虽然不能保证完全按照代码的顺序执行,但它会保证最终的结果和按照代码的顺序执行的结果一致
  1. 可见性
    线程操作共享变量时,会将该变量从内存加载到CPU缓存中,修改该变量后,CPU会立即更新缓存,但不一定会立即将它写入内存。若此时有其他线程访问该变量,从内存中读到的是旧数据,而非第一个线程操作后的数据。
  • 可见性指的是:当多线程并发访问共享变量时,一个线程对共享变量的修改,其它线程能够立即看到。
  1. 原子性
    CPU执行指令的顺序:读取指令、从内存读取数据、执行指令、把数据写回内存。
  • 原子性指的是:一个操作(有可能包含有多个步骤)要么全部执行(生效),要么全部都不执行(都不生效)。

如何保证线程安全

  • volatile关键字
  • 原子操作(原子类型)
  • 线程同步(

volatile 关键字

volatile关键字包含两方面

  1. 保证内存变量的可见性
  2. 禁止代码优化(重排序)

volatile关键字修饰res变量(其他代码不变):

volatile int res = 0;

修改后编译运行程序,并不能解决问题。因为,volatile关键字仅解决了可见性问题,并不能完全解决线程安全问题。

其他两个保证线程安全的方法未完待续

感谢浏览

相关文章:

  • 神经网络的数据集处理
  • vxe-table中vxe-grid中的合并单元格(合并行、列)
  • 音视频入门基础:RTP专题(17)——音频的SDP媒体描述
  • 主流向量数据库对比
  • ubuntu 24 安装 python3.x 教程
  • 深度学习篇---Opencv中Haar级联分类器的自定义
  • 怎样进行相关论文的调研——How to conduct research on relevant papers?
  • WebSocket 使用教程
  • C# net deepseek RAG AI开发 全流程 介绍
  • 大数据学习(66)- CDH管理平台
  • 爬虫一些基础知识的备忘录(需要自取)
  • 大语言模型微调和大语言模型应用的区别?
  • Python中将Markdown文件转换为Word
  • hadoop案例实践:气象大数据离线分析
  • 悬挂指针与野指针:如何避免常见内存问题
  • MySql学习_基础Sql语句
  • WSL 中的高级设置配置
  • kkFileView文件预览组件部署说明
  • 工控hmi医疗终端机的界面如何来设计?本文为你解答
  • Spring Boot 参数校验异常与错误编码映射方案
  • 龚正会见巴基斯坦卡拉奇市市长穆尔塔扎·瓦哈卜、巴西圣保罗市市长里卡多·努内斯
  • 马上评丨一些影视剧的片名,越来越让人看不懂
  • 云南洱源县4.8级地震:房屋受损442户,无人员伤亡报告
  • 叶迪奇任陆金所控股董事长,赵容奭继续担任CEO
  • “下一个高增长市场,还是中国”,龚正市长会见参加上海车展的国际企业高管
  • 全国双拥模范城(县)名单