Java练级打怪路--synchronized对比cas

前言

这篇文章的营养非常有限,只是一个夜黑风高的晚上,突发奇想,如果我要统计一个网站的PV,程序应该怎么写呢?

一种挫逼的写法

Java练级打怪路--synchronized对比cas

这是一种无锁的写法,很明显,这个东西是线程不安全的。我们使用12个线程,每个线程执行 10^8次方add的操作,发现最终的结果并没有得到期望的1.2*10^9次方。

Java练级打怪路--synchronized对比cas

上面一个是总数,下面一个是所消耗的时间。

synchronized VS cas

很明显,我们需要一个锁来干这个事情。

synchronized

synchronized关键字,是Java中一个同步锁,主要有一下几种用法:

  • 修饰一个代码块,被修饰的代码块称为同步语句块,其作用的范围是大括号{}括起来的代码

  • 修饰一个方法,被修饰的方法称为同步方法,其作用的范围是整个方法

  • 修饰一个静态的方法,其作用的范围是整个静态方法。

Java练级打怪路--synchronized对比cas

cas

compareAndSet,如果之前了解过C语言或者操作系统,相信对cas不会太陌生,这是一个原子方法。Java中我们可以使用sun.misc.Unsafe#compareAndSwapLong这个方法。

Java练级打怪路--synchronized对比cas

对比

  • 测试环境:

Java练级打怪路--synchronized对比cas

  • 测试条件:

    每个线程执行1千万次add 1操作

  • 测试结果:(4次取平均值)

线程数synchronized耗时(ms)cas耗时(ms)方法三(ms)
129614292
427492611999
8579748451851
1611223101923702
3214949200097779
64314153997413784

发现

  • synchronized好快

    我们发现synchronized关键一开始落后于cas,但是在后期却完成反超。synchronized其实在JDK1.5进行一波更新。速度大大的提升。

    后面我们再继续深入将jdk对synchronized的优化。

    cas在竞争激烈的时候速度反而下降。不难想象反复的失败重试。

  • CPU资源问题

    我们发现了一个事情,synchronized执行的过程中,CPU的资源一直上不去,这个也不难想到原因,因为其他线程一直竞争不到锁,一直处于阻塞的状态。

  • cas的优化

    jdk为我们提供了一个类java.util.concurrent.atomic.AtomicLong,效果可以显著提高。方法三我就是用这个测出来的。虽然方法二的实现跟方法三的一模一样,我最后都直接copy代码出来了,但仍然达不到该效率,估计是有jvm级别的优化。

    当然我们可以模拟jvm对synchronized的优化,简单的说,jvm的moniter会根据竞争的情况而调整synchronize的锁,我们按照这一思路,如果cas交换次数失败到一定的次数,就阻塞这个线程。

  • synchronized是个好东西

    synchronized是一个非常稳定的东西,虽然效率不一定是最佳的。但确实非常好用,下面再来认真研究研究。

Java练级打怪路--synchronized对比cas

  • cas模型的CPU基本打满。

Java练级打怪路--synchronized对比cas


Java练级打怪路--synchronized对比cas

  • 增加了这个条件之后耗时大概减少了40%,CPU的使用降低70%(32线程/64线程条件下),当然这个1000是我胡乱搞出来的一个值,但线程数提升上去后仍然比synchronized慢。

本文为头条号作者原创。未经授权,不得转载。

  • 1
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值