1.操作系统如何实现原子性
在单处理器,单核,运行多线程的情况下,我们不使用线程同步工具,
我们会出现,线程之间会互相抢夺,临界区的资源,造成数据不符合我们预期的结果,
后面再说解决办法,那么我们怎么帮助实现原子性
1 屏蔽中断,不让线程之间切换,让它完成再切换
2 底层硬件自旋锁,也是不让切换的思路
在多处理器,多核,运行多线程的情况下,除了上面两点,我们该如何保持原子性呢?
让我们先了解一下 多核 cpu 之间是如何通信的:
如图,通过CPU总线 进行数据交换
问题 1
因为我们每个核心都在执行任务,当我们执行任务时,都在对同一个数据进行修改
那么到底用 哪一个数据呢?
问题 2
当 我们 使用缓存时,数据在缓存 与 磁盘上不一致时,CPU 该如何保持呢?
一致性协议,就是解决这种问题,
所以 当我们 想使用 具有 原子性 的变量时,使用 修改(modified)和独占(exclusive) 的数据
2.在c++中如何实现线程同步
1.使用 条件变量和锁
写一个例子:
三个线程轮流 输出 1 2 3 到 100
cpp
std::condition_variable cond;
std::mutex mtu;
int nums = 1;
void fun1()
{
while (nums <= 100)
{
{
std::unique_lock<std::mutex> lock(mtu);
if (nums % 3 == 1)
{
cout << "nums 1:" << nums << endl;
nums++;
cond.notify_all();
}
else
{
cond.wait(lock);
}
}
}
cond.notify_all();
}
void fun2()
{
while (nums <= 100)
{
{
std::unique_lock<std::mutex> lock(mtu);
if (nums % 3 == 2)
{
cout << "nums 2:" << nums << endl;
nums++;
cond.notify_all();
}
else
{
cond.wait(lock);
}
}
}
cond.notify_all();
}
void fun3()
{
while (nums <= 100)
{
{
std::unique_lock<std::mutex> lock(mtu);
if (nums % 3 == 0)
{
cout << "nums 3:" << nums << endl;
nums++;
cond.notify_all();
}
else
{
cond.wait(lock);
}
}
}
cond.notify_all();
}
int main()
{
std::thread t1(fun1);
std::thread t2(fun2);
std::thread t3(fun3);
t1.join();
t2.join();
t3.join();
cout << "打印完毕" << endl;
return 0;
}
2.使用atomic
cpp
std::atomic<int> nums(1);
void print_nums(int remainder)
{
while (nums <= 100)
{
int current_nums;
do
{
current_nums = nums.load();
} while (current_nums % 3 != remainder && !nums.compare_exchange_weak(current_nums, current_nums + 1));
if (current_nums <= 100)
{
std::cout << "nums " << remainder + 1 << ":" << current_nums << std::endl;
}
}
}
int main()
{
std::thread t1(print_nums, 1);
std::thread t2(print_nums, 2);
std::thread t3(print_nums, 0);
t1.join();
t2.join();
t3.join();
std::cout << "打印完毕" << std::endl;
return 0;
}