IT数码 购物 网址 头条 软件 日历 阅读 图书馆
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
图片批量下载器
↓批量下载图片,美女图库↓
图片自动播放器
↓图片自动播放器↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁
 
   -> C++知识库 -> 547-C++11的原子类型 -> 正文阅读

[C++知识库]547-C++11的原子类型

我们在前面的博客里实现的窗口卖票的代码,使用的是下面这种线程安全的操作:

int count=100;
lock_guard<std::mutex> guard(mtx);
count++;

lock_guard<std::mutex> guard(mtx);
count--;

但是互斥锁是比较重的,如果临界区代码做的事情稍稍复杂,比较多,我们用这个互斥锁是应该的。

但是像简单的对变量的++或者–,就不需要用互斥锁了。
我们应该用CAS来保证上面++ --操作的原子特性就足够了, CAS也称作无锁操作(并不是说不加锁,只是这个加锁不在我们的软件层面,CPU和内存通信是通过系统总线进行的,CAS就是通过 exchange/swap指令给总线加锁,当一个线程在做CPU和内存之间数据的交换,也就是说内存有一个数据读到CPU,CPU进行计算再把数据写回内存块的这样一个过程,线程没有做完的话,它是不允许其他线程再去使用总线的,即硬件上实现的加锁操作,对于软件层面可以说是无锁操作)

无锁队列 => 就是由CAS来实现的

C++11的原子类型

#include <atomic>//包含了很多原子类型

代码示例:
因为isReady和mycount是属于数据段上的,一个进程里的线程,不同线程的栈不同,但是数据段是共享的,对共享的变量,多线程是会进行缓存的,这个线程把isReady改成true以后,其他的线程是不是马上能看到isReady变为true呢?并不是的,因为它读的都是自己的缓存,所以,我们要加上volatile关键字。volatile就是防止多线程对共享变量进行缓存, 让大家访问的都是原始内存中的变量的值。

线程缓存可以加快线程运行的效率,因为线程就像人一样,好不容易从内存上跑到CPU上,它还要不断从内存上取东西做运算,它好不容易占1次CPU的时间片,还要在CPU和内存中不断来回跑,很浪费时间,所以操作系统对线程的执行来说,都是会让线程来CPU执行的时候把它们共享的变量在线程的栈上拷贝一份,统一的都带到CPU的缓存里,加个volatile ,就是让所有的线程对共享变量不再进行缓存,保证我们代码的正确性,一个线程对共享变量的改变马上就可以反映到另外一个线程里面了,这就是原子类型。

#include <iostream>
#include <thread>
#include <atomic>//包含了很多原子类型
#include <list>
using namespace std;

volatile std::atomic_bool isReady = false;
volatile std::atomic_int mycount = 0;

void task()
{
	while (!isReady)//还没有准备好 
	{
		std::this_thread::yield();//线程出让当前的CPU时间片,等待下一次调度
		//线程相当于还是就绪状态,等待CPU的下一次调度 
	}

	for (int i = 0; i < 100; ++i)
	{
		mycount++;
	}
}
int main()
{
	list<std::thread> tlist;
	for (int i = 0; i < 10; ++i)
	{
		tlist.push_back(std::thread(task));//启动10个线程 
	}

	std::this_thread::sleep_for(std::chrono::seconds(3));//主线程睡眠3秒 
	isReady = true;//所有子线程都进行mycount++ 

	for (std::thread &t : tlist)
	{
		t.join();
	}
	cout << "mycount:" << mycount << endl;

	return 0;
}

在这里插入图片描述

  C++知识库 最新文章
【C++】友元、嵌套类、异常、RTTI、类型转换
通讯录的思路与实现(C语言)
C++PrimerPlus 第七章 函数-C++的编程模块(
Problem C: 算法9-9~9-12:平衡二叉树的基本
MSVC C++ UTF-8编程
C++进阶 多态原理
简单string类c++实现
我的年度总结
【C语言】以深厚地基筑伟岸高楼-基础篇(六
c语言常见错误合集
上一篇文章      下一篇文章      查看所有文章
加:2021-09-12 12:59:34  更:2021-09-12 13:01:06 
 
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁

360图书馆 购物 三丰科技 阅读网 日历 万年历 2024年5日历 -2024/5/20 3:10:29-

图片自动播放器
↓图片自动播放器↓
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
图片批量下载器
↓批量下载图片,美女图库↓
  网站联系: qq:121756557 email:121756557@qq.com  IT数码