int linux 原子操作_linux c++编程之多线程:原子操作如何解决线程冲突

在多线程中操作全局变量一般都会引起线程冲突,为了解决线程冲突,引入原子操作。

1.线程冲突

#include #include #include #include int g_count = 0;void count(void *p){Sleep(100);    //do some work//每个线程把g_count加1共10次for (int i = 0; i < 10; i++){g_count++;}Sleep(100);   //do some work}int main(void){printf("******多线程访问全局变量演示***by David***");//共创建10个线程HANDLE handles[10];for (int i = 0; i < 10; i++){for (int j = 0; j < 10; j++){handles[j] = _beginthread(count, 0, NULL);}WaitForMultipleObjects(10, handles, 1, INFINITE);printf("%d time g_count = %d", i, g_count);//重置g_count = 0;}getchar();return 0;}

运行

94593949c57324e5c88f24df79136133.png

理论上,g_count的最后结果应该是100,可事实却并非如此,不但结果不一定是100,而且每次的结果还很可能不一样。原因是,多个线程对同一个全局变量进行访问时,特别是在进行修改操作,会引起冲突。详细解释:

设置断点,查看反汇编

be4ebbfdea4fb85f1b8e640a47d81f98.png

g_count++;被分为三步操作:

1.把g_count的内容从内存中移动到寄存器eax

2.把寄存器eax加1

3.把寄存器eax中的内容移动到内存g_count的地址

三步以后,g_count的值被顺利加1。

cpu执行的时间片内包含多条指令,每条指令执行期间不会被打断,但如果一个操作包含多条指令,则很有可能该操作会被打断。g_count++;就是这样的操作。

g_count被顺利加到100的前提:每次加1,都建立在上一次加1顺利完成的基础上。也就是说,如果上一次加1被打断,这一次的加1就得不到上一次加1的累积效果。自然,最后的结果,多半会小于100。

2.原子操作

所谓原子操作,是指不会被线程调度机制打断的操作,操作一旦开始,就得执行到结束为止。原子操作可以是一个步骤,也可以是多个操作步骤,但是其顺序是不可以被打乱,或者切割掉只执行部分。原子操作一般靠底层汇编实现。

在头文件winnt.h中提供了很多的原子操作函数,它们使用自加锁的方式,保证操作的原子性,如自增操作

InterlockedIncrement,

函数原型

LONG CDECL_NON_WVMPURE InterlockedIncrement(
_Inout_ _Interlocked_operand_ LONG volatile *Addend
);

其它相关操作,请自行查看头文件。

使用该函数,我们修改线程函数count。修改很简单,只需把g_count++改为InterlockedIncrement((LONG)&g_count);即可。

运行如下

1bc928dbefea93e4aef3c9ae32d62f1d.png

显然,在原子操作下,我们肯定是可以得到正确结果的。

需要C/C++ Linux服务器架构师学习资料私信“资料”(资料包括C/C++,Linux,golang技术,Nginx,ZeroMQ,MySQL,Redis,fastdfs,MongoDB,ZK,流媒体,CDN,P2P,K8S,Docker,TCP/IP,协程,DPDK,ffmpeg等),免费分享

fb2bebf1f6972de30d2f775b4f8bd782.png

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/459291.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

透明明兼容

opacity:0.5; filter:alph(opacity50);转载于:https://www.cnblogs.com/yjhua/p/4580864.html

ORA-00119错误的解决。

今天在启动服务器上的ORACLE时遇到如下错误&#xff1a;SQL> startup;ORA-00119: invalid specification for system parameter LOCAL_LISTENERORA-00132: syntax error or unresolved network name ’LISTENER_ORCL’然后&#xff0c;在网上找了一些资料&#xff0c;解决了…

python查询斐波那契数列通项公式_分享一个神奇的操作系统——斐波那契+MACD,每一波都有20%以上的收益!...

斐波那契数列&#xff0c;又称兔子数列&#xff0c;或者黄金分割数列。指的是这样一个数列&#xff1a;0、1、1、2、3、5、8、13、21……从第三项起&#xff0c;它的每一项都等于前两项的和。为什么是兔子数列?我们假设兔子在出生两个月后&#xff0c;就有繁殖能力&#xff0c…

FPGA 状态机设计

数字系统有两大类有限状态机&#xff08;Finite State Machine&#xff0c;FSM&#xff09;&#xff1a;Moore状态机和Mealy状态机。 Moore状态机 其最大特点是输出只由当前状态确定&#xff0c;与输入无关。Moore状态机的状态图中的每一个状态都包含一个输出信号。这是一个典型…

Cisco堆叠配置步骤+链路聚合实例

步骤:配置堆叠堆叠主: switch 1 priority 15copy run startup-configreload slot 1堆叠备:switch 1 renumber2switch 1priority 15copy run startup-configreload slot 1连接堆叠线缆配置端口聚合interface Port-channel2descriptionxxxxswitchport mode accessswitchport acce…

c++面试常考的知识点汇总

1. 链表和顺序表的优缺点&#xff0c;在什么场合下适合顺序表&#xff0c;什么场合下适合用链表。 1&#xff09;顺序表存储原理&#xff1a;顺序表存储是将数据元素放到一块连续的内存存储空间&#xff0c;存取效率高&#xff0c;速度快。但是不可以动态增加长度优点&#xf…

aimesh node重启_ASSIMP的数据结构

数据结构是对文件的描述&#xff0c;也是我们想要获取的最终产物&#xff0c;一般来说我们会用这个结果映射到我们自己的模型里面去(这样做是为了降低对这个项目的依赖&#xff0c;如果你的模型都用别人的&#xff0c;还搞毛啊)1.aiScene &#xff1a;根节点struct aiScene{uns…

WCF 初识(一)

WCF的前世今生 在.NETFramework 2.0以及前版本中&#xff0c;微软发展了Web Service&#xff08;SOAP with HTTP communication&#xff09;&#xff0c;.NET Remoting&#xff08;TCP/HTTP/Pipeline communication&#xff09;以及基础的Winsock等通信支持。 由于各个通信方法…

iOS 点转成字符串,再字符串转换成点

CGPointFromString(<#NSString *string#>) NSStringFromCGPoint(<#CGPoint point#>)转载于:https://www.cnblogs.com/56ik/p/5111848.html

C/C++关键字解析

2、C/C分别有多少个关键字&#xff1f; 假如别人问某一个关键字是否属于C/C&#xff0c;要能正确的答出来。 1&#xff09;由ANSI标准定义的C语言关键字共32个 auto double int struct break else long switch case enum register typedef char extern return union const flo…

webpack打开项目命令_webpack打包好的页面在项目中怎么运行?

使用webpack打包好了页面和相关的js文件&#xff0c;然后命令输入webpack-dev-server&#xff0c;再浏览器中输入&#xff1a;http://localhost:9090/WebRoot...打开页面能正常的加载样式和数据&#xff0c;无报错。但是本项目是使用Jfnal框架(不知道有没有关系),然后在eclipse…

常见HTTP状态值

状态代码有三位数字组成&#xff0c;第一个数字定义了响应的类别&#xff0c;且有五种可能取值&#xff1a;1xx&#xff1a;指示信息--表示请求已接收&#xff0c;继续处理2xx&#xff1a;成功--表示请求已被成功接收、理解、接受3xx&#xff1a;重定向--要完成请求必须进行更进…

Linux系统:软链接与硬链接的原理分析

1、相关概念 1、索引节点inode(index node)&#xff1a;inode就是索引节点&#xff0c;它用来存放档案及目录的基本信息&#xff0c;包含时间、档名、使用者及群组等。 inode 是 UNIX/Linux 操作系统中的一种数据结构&#xff0c;其本质是结构体它包含了与文件系统中各个文件相…

python编写命令行框架_python的pytest框架之命令行参数详解(上)

前言pytest是一款强大的python自动化测试工具&#xff0c;可以胜任各种类型或者级别的软件测试工作。pytest提供了丰富的功能&#xff0c;包括assert重写&#xff0c;第三方插件&#xff0c;以及其他测试工具无法比拟的fixture模型。pytest是一个软件测试框架&#xff0c;是一款…

Android学习笔记——Handler(二)

对比请看http://blog.sina.com.cn/s/blog_78c913e30100uqmf.html 以下代码是MainActivity.java中的代码 package com.example.handlertest;import android.app.Activity; import android.os.Bundle; import android.os.Handler;public class MainActivity extends Activity {pr…

纯后台生成highcharts图片有哪些方法?

比如说&#xff0c;领导抛给你一个需求&#xff0c;把一些数据做成图表&#xff0c;每天通过邮件发送&#xff0c;让领导能在邮件中就看到图片&#xff0c;你会有什么思路呢&#xff1f;本人使用的是phantomjs这个神器&#xff0c;它的内核是WebKit引擎&#xff0c;不提供图形界…

操作系统:虚拟页式存储管理(缺页中断、页面置换算法)

1、基本工作原理 1、基本工作原理 在进程开始运行之前&#xff0c;不是全部装入页面&#xff0c;而是装入一个或者零个页面&#xff0c;之后根据进程运行的需要&#xff0c;动态装入其他页面&#xff1b;当内存已满&#xff0c;而又需要装入 新的页面时&#xff0c;则根据某种…

小程序 长按api_高质量的微信小程序样式模板应该长什么样?

现在不懂技术的小白若想快速制作自己的小程序&#xff0c;一般是通过小程序模板来实现。通过在模板上添加自己的图片、文字、商品等等&#xff0c;可以很简单地生成一个小程序。不过要想把小程序做得好看&#xff0c;你得找高质量的小程序样式模板才行。那么高质量的微信小程序…

(笔记)Mysql命令drop database:删除数据库

drop命令用于删除数据库。drop命令格式&#xff1a;drop database <数据库名>;例如&#xff0c;删除名为 xhkdb的数据库&#xff1a;mysql> drop database xhkdb;[例子1] 删除一个已经确定存在的数据库&#xff1a; mysql> drop database drop_database; Query…

web安全测试-AppScan使用分享

这里主要分享如何使用AppScan对一大项目的部分功能进行安全扫描。 ------------------------------------------------------------------------ 其实&#xff0c;对于安全方面的测试知道的甚少。因为那公司每个月要求对产品进行安全扫描。掌握了一人点使用技巧&#xff0c;所…