有必要同步全局变量的使用

It's necessary to synchronize the use of the global variable

本文关键字:全局变量 同步      更新时间:2023-10-16

我有这个代码,这里需要同步使用全局变量c?流将同时开始工作并且一个线程将覆盖另一个线程的结果并最终获得 2 或 7 是否可行?

#include <iostream>
#include <stdio.h> 
#include <pthread.h> 
#include <stdlib.h>
int c = 0;
void* write(void*)
{
    c += 2;
}
void* read(void*)
{
    c += 7; 
}
int main()
{
    pthread_t t1;
    pthread_t t2;
    std::cout << "first C = " << c << std::endl;
    int r1 = pthread_create(&t1, 0, &write, 0);
    int r2 = pthread_create(&t2, 0, &read, 0);
    pthread_join(t1, 0);
    pthread_join(t2, 0);
    std::cout << " C = " << c << std::endl;
    return 0;
}

是的,您需要保护对全局变量的访问。您要么需要使用互斥锁,要么需要使用原子类型(包括原子操作来修改它们)。

如果不这样做,问题是正常的+=需要读取操作,添加一些值,然后将结果写回内存。如果两个线程同时执行此操作,它们可能会重叠,例如,两个线程都将读取示例代码中的0,然后两个线程都写入其结果,这意味着最后一个写入操作获胜,因此另一个添加操作丢失。你最终会得到27,而不是你已经怀疑的9

如果您有多个线程可能同时访问一个对象,并且至少其中一个线程更改了该对象,则除非同步对该对象的访问,否则您将存在数据争用。如果您有数据争用,则程序的行为是未定义的。

也就是说,是的,您需要使用适当的同步来访问c

在 SMP 机器(具有单独的数据缓存内存)中,还有另一件微妙的事情需要考虑,那就是线程间内存同步。如果线程在不同的内核/CPU 中运行,即使两个线程更新c运行时间不完全相同,它们也可能无法立即看到来自另一个线程的内存更新。例如,使用 pthread_mutex_t 进行同步将解决此问题。

请注意,pthread_join() 将内存与其他线程同步,因此在加入线程后不需要其他措施来保证c看到其更新的值。

以下是保证提供与其他线程的内存同步的 POSIX 线程函数列表:http://pubs.opengroup.org/onlinepubs/9699919799/basedefs/V1_chap04.html#tag_04_11。