C#内置提供的原子操作

  1. Interlocked.Increment:以原子操作的形式递增指定变量的值并存储结果。
  2. Interlocked.Decrement:以原子操作的形式递减指定变量的值并存储结果。
  3. Interlocked.Add:以原子操作的形式,添加两个整数并用两者的和替换第一个整数

问题:如果要进行原子的乘法、除法或者其他操作改怎么办,C#并没有内置提供相应的方法呀?

那我们先来大概理解一下原子操作的流程

以增加变量值为例

  1. 将实例变量添加到CPU寄存器中
  2. 将该变量的值进行增加
  3. 将该变量的增加后的值从CPU寄存器中还原到堆或栈中实例变量的值

可以看到如果是多核CPU在多线程环境下可能会导致在执行完步骤1,步骤2后当前线程失去时间片,其他线程读取改变量的值时就会读取到并未增加增加之前的值,所以就导致了数据的不一致。然而C#提供了上面三种原子操作来保证不出现这样的数据不一致,至于底层原理暂时不深究,我们来看看如何实现除这三种方法之外的操作。

进行原子的获取最大值操作

先放代码如下:

  1. public static int Maximum(ref int target, int value)
  2. {
  3. //注意target前面加了 ref ,这样在方法外改变target的值将会影响到方法内的target值,
  4. //即类似按引用传递
  5. int currentVal = target;//使用currentVal局部变量来存储target值,target值的变更不会影响currentVal
  6. int startVal = 0;
  7. int desiredVal = 0;
  8. do
  9. {
  10. startVal = currentVal; //记录本次迭代的起始值 1
  11. desiredVal = Math.Max(startVal, value);//根据startVal和value计算最大值desiredVal 2
  12. //如果target和startVal值一致则用desiredVal赋值给target,并将target原始值进行返回
  13. //如果target和startVal值不相同则什么都不做,并将target的值进行返回
  14. //target的值因为在别的线程进行操作时可能改变target的值,所以导致target值和startVal不一致
  15. currentVal = Interlocked.CompareExchange(ref target, desiredVal, startVal);// 3
  16. } while (startVal != currentVal); // 4
  17. //因为startVal记录的是开始时target的值,而currentVal记录的则是target最新值
  18. //如果startVale和currentVal不一致则代表其他线程已经更改的target的值,所以需要重新迭代。
  19. //重新迭代时currentVal代表的是target的最新值
  20. //疑问:如果在 3 之后 4之前 target的值被其他线程更改怎么办?请大神帮忙
  21. return desiredVal;
  22. }

大致的逻辑可以参考注释进行理解,最主要的方法是通过currentVal = Interlocked.CompareExchange(ref target, desiredVal, startVal); 可以参考MSDN此方法的解释

其他操作

C# VIA CLR中给出了其他操作的模板,可以参考如下:


  1. delegate int Morpher<TResult, TArgument>(int startValue, TArgument argument, out TResult morphResult);
  2. static TResult Morph<TResult,TArgument>(ref int target,TArgument argument,Morpher<TResult,TArgument> morpher)
  3. {
  4. TResult morphResult;
  5. int currentVal = target;
  6. int startVal = 0;
  7. int desiredVal = 0;
  8. do
  9. {
  10. startVal = currentVal;
  11. desiredVal = morpher(startVal, argument, out morphResult);
  12. currentVal = Interlocked.CompareExchange(ref target, desiredVal, startVal);
  13. } while (startVal != currentVal);
  14. return morphResult;
  15. }

可以通过委托定义自己想要的操作。乘法、除法等等。

大神写出的书就是牛,膜拜Jeffrey Richter

C# 原子操作理解的更多相关文章

  1. 十、future其他成员函数、shared_future、atomic(原子操作)

    一. int mythread(){ cout<<"thread"<<endl; std::chrono::milliseconds dura();//5秒 ...

  2. 深入理解Atomic原子操作和volatile非原子性

    原子操作可以理解为: 一个数,很多线程去同时修改它,不加sync同步锁,就可以保证修改结果是正确的 Atomic正是采用了CAS算法,所以可以在多线程环境下安全地操作对象. volatile是Java ...

  3. 深入理解java:2.3.1. 并发编程concurrent包 之Atomic原子操作(循环CAS)

    java中,可能有一些场景,操作非常简单,但是容易存在并发问题,比如i++, 此时,如果依赖锁机制,可能带来性能损耗等问题, 于是,如何更加简单的实现原子性操作,就成为java中需要面对的一个问题. ...

  4. 关于P,V操作理解的突破,关于并发设计与并行

    今天又找了一篇博客研究P,V操作.. 发现..它有一个变量没有声明.. 我就换了篇博客..http://c.biancheng.net/cpp/html/2600.html 然后就看懂了.. 关键突破 ...

  5. sun.misc.Unsafe的理解

    以下sun.misc.Unsafe源码和demo基于jdk1.7: 最近在看J.U.C里的源码,很多都用到了sun.misc.Unsafe这个类,一知半解,看起来总感觉有点不尽兴,所以打算对Unsaf ...

  6. pthread_cond_wait()函数的理解(摘录)

    /************pthread_cond_wait()的使用方法**********/pthread_mutex_lock(&qlock);    /*lock*/pthread_c ...

  7. SOA架构介绍和理解

    SOA架构介绍和理解 SOA的正确方法论及目标模型,其实SOA在实现架构落地上,需要考虑到对服务的组合,不断的重用现有的服务,让企业应用可以逐步集成,快速实现业务的迭代. 通过SOA架构分层将服务按照 ...

  8. 【Todo】【转载】深入理解Java内存模型

    提纲挈领地说一下Java内存模型: 什么是Java内存模型 Java内存模型定义了一种多线程访问Java内存的规范.Java内存模型要完整讲不是这里几句话能说清楚的,我简单总结一下Java内存模型的几 ...

  9. oracle rac理解和用途扩展

    Oracle RAC的优势在于利用多个节点(数据库实例)组成一个数据库,这样在保证了数据库高可用性的情况下更充分的利用了多个主机的性能,而且可以通过增加节点进行性能的扩展.实现Oracle RAC需要 ...

随机推荐

  1. python zip压缩文件并设置密码

    zip     -P "123" -r  app.zip         app压缩文件 密码 压缩后名称 压缩对象名称 def zipDir(dirpath, outFullNa ...

  2. 错误InnoDB:Attemptedtoopenapreviouslyopenedtablespace.

    2013-08-04 13:48:22 760 [ERROR] InnoDB: Attempted to open a previously opened tablespace. Previous t ...

  3. mysql 优化之 is null ,is not null 索引使用测试

    关于mysql优化部分,有很多网友说尽量避免使用is null, is not null,select * 等,会导致索引失效,性能降低?那是否一定收到影响呢?真的就不会使用索引了吗? 本文的测试数据 ...

  4. jquery 全选样例

    代码: $(function(){ $("#checkAllOld").click(function() { $("input[id^='box_old_']" ...

  5. 死磕 java同步系列之Semaphore源码解析

    问题 (1)Semaphore是什么? (2)Semaphore具有哪些特性? (3)Semaphore通常使用在什么场景中? (4)Semaphore的许可次数是否可以动态增减? (5)Semaph ...

  6. Asp.Net Mvc自定义控件之树形结构数据生成表格 - WPF特工队内部资料

    最近项目中有一个需求,将树形结构的数据,以表格的形式展示在页面中,下图是最终呈现效果: 源码: @{ Layout = null; } <!DOCTYPE html> <html&g ...

  7. 动软生成Model(dapper.common)

    <#@ template language="c#" HostSpecific="True" #><#@ output extension= ...

  8. Java内存模型以及happens-before规则

    本人免费整理了Java高级资料,涵盖了Java.Redis.MongoDB.MySQL.Zookeeper.Spring Cloud.Dubbo高并发分布式等教程,一共30G,需要自己领取.传送门:h ...

  9. Python爬取十四万条书籍信息告诉你哪本网络小说更好看

    前言 本文的文字及图片来源于网络,仅供学习.交流使用,不具有任何商业用途,版权归原作者所有,如有问题请及时联系我们以作处理. 作者: TM0831 PS:如有需要Python学习资料的小伙伴可以加点击 ...

  10. centos 7 搭建Samba

    一.Samba简介 Samba是一个能让Linux系统应用Microsoft网络通讯协议的软件,由客户端和服务端构成. SMB(Server Message Block的缩写,即服务器消息块)主要是作 ...