C++11原子操作详解

 更新时间:2021年11月19日 09:56:21   作者:ufgnix0802  
这篇文章主要为大家介绍了C++的原子操作,具有一定的参考价值,感兴趣的小伙伴们可以参考一下,希望能够给你带来帮助

C++11原子操作

原子操作的概念

所谓原子操作,其意义就是“原子是最小的,不可分割的最小个体”。**表示当多个线程访问同一个全局资源的时候,能够确保所有其它的线程都不在同一时间访问相同的资源。**也就是它确保在同一时刻只有唯一的线程对这个资源进行访问。类似于共享资源的访问保护。但是原子操作更加接近底层,即效率更高。

在以往C++中没有原子操作的规定,更多使用的都是汇编语言或者借助第三方库,如Intel的pthread来实现。但在C++11中的特性引入原子操作的相关概念,并通过新的头文件提供了多种原子操作数据类型。如atomic_bool,atomic_int等。如果需要多个线程对这些类型的共享资源进行操作,编译器将保证这些操作都是具有原子性的。通俗地说,就是确保在任意时刻只有一个线程对这个资源进行访问,编译器将保证多个线程访问这个资源的正确性,从而避免锁的使用,提高效率。

示例

#include <iostream>
#include <thread>//C++11线程库且跨平台
#include <windows.h>//Sleep函数需要使用的库文件
using std::cout;
using std::endl;
using std::cin;
int g_total = 0;
void click()
{
	for (int i = 0; i < 1000000; i++)
	{
		g_total++;
	}
}

int main()
{
	for (int i = 0; i < 4; i++)
	{
		std::thread t(click);
		t.detach();
	}
	Sleep(1000);
	cout << "result:" << g_total << endl;
	return 0;
}

我们很正常的认为这样做是可以提高CPU的利用效率的,但是实际上执行结果并不正确。

效果图

紧接着,我们肯定想到使用互斥锁对共享资源进行保护。

#include <iostream>
#include <thread>//C++11线程库是跨平台的
#include <mutex>//C++11互斥锁
#include <windows.h>//Sleep函数需要使用的库文件
using std::cout;
using std::endl;
using std::cin;

int g_total = 0;
std::mutex g_mutex;
void click()
{
	for (int i = 0; i < 1000000; i++)
	{
		g_mutex.lock();//访问之前锁定互斥对象
		g_total++;
		g_mutex.unlock();//访问之后释放互斥对象
	}
}

int main()
{
	for (int i = 0; i < 4; i++)
	{
		std::thread t(click);
		t.detach();
	}
	Sleep(1000);
	cout << "result:" << g_total << endl;
	return 0;
}

初始对象的使用,保证同一时刻只有唯一一个线程对这个共享对象进行访问。

效果图

在C++11之前,互斥锁的概念已经足够了,但是在C++11提出之后,进一步利用CPU性能。在C++11中实现了原子操作的数据类型(如atomic_bool,atomic_int等)。使用原子操作的数据类型线程对其进行访问的时候无需借助mutex等锁机制,也能实现对共享资源的正确访问。

#include <iostream>
#include <thread>//C++11线程库是跨平台的
#include <atomic>//C++11原子操作库
#include <windows.h>//Sleep函数需要使用的库文件
using std::cout;
using std::endl;
using std::cin;
std::atomic_int g_total = 0;//使用原子操作的数据类型
void click()
{
	for (int i = 0; i < 1000000; i++)
	{
		//mutex.lock();
		g_total++;
		//mutex.unlock();
	}
}

int main()
{
	for (int i = 0; i < 4; i++)
	{
		std::thread t(click);
		t.detach();
	}
	Sleep(1000);
	cout << "result:" << g_total << endl;
	return 0;
}

效果图

总结

本篇文章就到这里了,希望能够给你带来帮助,也希望您能够多多关注脚本之家的更多内容!

相关文章

  • 使用Qt实现获取本机IP和定位

    使用Qt实现获取本机IP和定位

    这篇文章主要为大家详细介绍了如何使用Qt实现获取本机IP和定位,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下
    2024-11-11
  • C++排序算法之插入排序

    C++排序算法之插入排序

    这篇文章主要为大家详细介绍了C++排序算法之插入排序,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2019-08-08
  • C++课程设计之图书馆管理系统

    C++课程设计之图书馆管理系统

    这篇文章主要为大家详细介绍了C++课程设计之图书馆管理系统,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2022-03-03
  • C语言堆结构处理TopK问题详解

    C语言堆结构处理TopK问题详解

    TopK问题即在N个数中找出最大的前K个,这篇文章将详细讲解如何利用小根堆的方法解决TopK问题,文中代码具有一定参考价值,快跟随小编一起学习一下吧
    2022-06-06
  • C++中std::transform的使用小结

    C++中std::transform的使用小结

    std::transform 是 C++ 标准库中的一个算法,本文主要介绍了C++中std::transform的使用,具有一定的参考价值,感兴趣的可以了解一下
    2024-05-05
  • C语言实现两个递减数列中寻找某一个数

    C语言实现两个递减数列中寻找某一个数

    这篇文章主要介绍了C语言实现两个递减数列中寻找某一个数,是一类经典的数组操作算法,需要的朋友可以参考下
    2014-09-09
  • 剖析C++的面向对象编程思想

    剖析C++的面向对象编程思想

    这篇文章主要介绍了剖析C++的面向对象编程思想,C++的面向对象特性是其对C语言的重要拓展之处,需要的朋友可以参考下
    2015-09-09
  • 详解C++编程中用数组名作函数参数的方法

    详解C++编程中用数组名作函数参数的方法

    这篇文章主要介绍了详解C++编程中用数组名作函数参数的方法,是C++入门学习中的基础知识,需要的朋友可以参考下
    2015-09-09
  • OJ中G++和C++的区别

    OJ中G++和C++的区别

    今天小编就为大家分享一篇关于OJ中G++和C++的区别,小编觉得内容挺不错的,现在分享给大家,具有很好的参考价值,需要的朋友一起跟随小编来看看吧
    2018-10-10
  • C++实现将内容写入文件的方法总结

    C++实现将内容写入文件的方法总结

    本文主要总结了一下C/C++将内容写入文件的方法,C的方法有些单调,毕竟没有库函数。C++则丰富些,下面我把搜集到的整理一下,供大家参考
    2023-04-04

最新评论