CAS和Synchronized的区别

1.什么是cas?
cas是乐观锁的一种实现方式,是一种轻量级锁,在java1.5的时候开始引入
2.cas的实现原理?
cas有3个操作数,内存值V,旧的预期值A,要更新的新值C,当且仅当预期值A和内存值V相同时,将内存值V修改为B,否则什么也不做.当多个线程同时尝试使用CAS更新一个变量时,任何时候只有一个线程可以更新成功,若失败线程会重新进入循环再次进行尝试.
cas的核心实现是unsafe,unsafe类是使java有了像c一样的操作内存空间的能力,一旦能够直接操作内存就意味着不受jvm管理,也就无法被GC,需要我们手动GC,所以很容易造成内存泄漏.具体详情不在这里展开.
3.cas的ABA问题和解决办法?*
ABA问题就是俩个线程1和2,线程1从内存X中取出A,线程2也从内存中取出A,并且将A变成了B,然后线程2又将B变成了A,这时候线程1执行cas操作成功,虽然线程1执行成功了但是执行成功的A已经不是原来获取的A了,整个执行过程是有问题的,比如链表的头在进行了俩次变化后恢复为原来的值并不代表链表就没有发生变化.
解决办法:在对象中额外再增加一个字段,用来标识有没有发生过变化,比如version,每次变化进行加1或者用当前时间戳.
4.CAS的不足?
1.如果CAS操作不成功的话,会一直自旋直到成功,cpu消耗会很大.
2.不能保证多个操作的原子性.
3.cas存在ABA问题
3.Synchronized的实现原理**
Synchronized的主要作用有三个:
1.原子性:确保线程互斥的访问同步的代码
2.可见性: 保证共享变量的修改能够及时的可见,
3.有序性:有效解决重排的问题
synchronized的用法
1.当Synchronized用在实例方法上时,监视锁(monitor)便是对象实例
2.当用在静态方法上时,监视的便是对象的class,因为class存在永久代,因此静态方法锁相当于该类的一个全局锁
3.当作用在一个对象实例时monitor便是括号括起来的对象实例
锁的分类
1.自旋锁
当一个线程尝试获取某个锁时,如果该锁已经被其他线程占用,就一直循环检测锁是否被释放,而不是进入挂起或者睡眠状态
自旋锁适用于锁保护的临界区很小的情况,临界区很小的话,锁占用的时间就很短。自旋等待不能替代阻塞,虽然它可以避免线程切换带来的开销,但是它占用了CPU处理器的时间。如果持有锁的线程很快就释放了锁,那么自旋的效率就非常好,反之,自旋的线程就会白白消耗掉处理的资源,它不会做任何有意义的工作,典型的占着茅坑不拉屎,这样反而会带来性能上的浪费。所以说,自旋等待的时间(自旋的次数)必须要有一个限度,如果自旋超过了定义的时间仍然没有获取到锁,则应该被挂起。

自旋锁在JDK 1.4.2中引入,默认关闭,但是可以使用-XX:+UseSpinning开开启,在JDK1.6中默认开启。同时自旋的默认次数为10次,可以通过参数-XX:PreBlockSpin来调整。

如果通过参数-XX:PreBlockSpin来调整自旋锁的自旋次数,会带来诸多不便。假如将参数调整为10,但是系统很多线程都是等你刚刚退出的时候就释放了锁(假如多自旋一两次就可以获取锁),是不是很尴尬。于是JDK1.6引入自适应的自旋锁,让虚拟机会变得越来越聪明
2.适应性自旋锁
JDK 1.6引入了更加聪明的自旋锁,即自适应自旋锁。所谓自适应就意味着自旋的次数不再是固定的,它是由前一次在同一个锁上的自旋时间及锁的拥有者的状态来决定。那它如何进行适应性自旋呢?

线程如果自旋成功了,那么下次自旋的次数会更加多,因为虚拟机认为既然上次成功了,那么此次自旋也很有可能会再次成功,那么它就会允许自旋等待持续的次数更多。反之,如果对于某个锁,很少有自旋能够成功,那么在以后要或者这个锁的时候自旋的次数会减少甚至省略掉自旋过程,以免浪费处理器资源。

  • 2
    点赞
  • 5
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: CASsynchronized都是用于实现多线程同步的机制,但它们的实现方式和应用场景有所不同。 CAS(Compare And Swap)是一种乐观锁机制,它通过比较内存中的值和期望值是否相等来判断是否需要更新,如果相等则更新,否则不更新。CAS的优点是无锁,可以避免线程的阻塞和唤醒,适用于并发量较大的情况。 synchronized是一种悲观锁机制,它通过获取锁来保证同一时刻只有一个线程可以执行临界区代码,其他线程需要等待锁的释放才能执行。synchronized的优点是可以保证线程安全,但缺点是会造成线程的阻塞和唤醒,适用于并发量较小的情况。 因此,CAS适用于并发量较大的情况,可以提高程序的性能;而synchronized适用于并发量较小的情况,可以保证程序的正确性。 ### 回答2: CAS(Compare and Swap)和synchronized都属于并发编程中用于实现线程安全的技术,但它们有一些重要的区别。 首先,CAS是一种乐观锁技术,而synchronized是一种悲观锁技术。CAS通过对比内存中的值与期望值,根据比较结果来决定是否更新内存中的值,无锁期间没有线程阻塞。相比之下,synchronized通过获取对象的锁来保证线程的同步执行,如果无法获取锁,线程会进入阻塞状态,直到获取到锁才能继续执行。 其次,CAS是基于底层硬件支持的原子操作指令来实现的,效率更高。而synchronized是基于Java的关键字实现的,需要涉及到用户态和内核态的切换,效率相对较低。 此外,CAS操作可以针对单个变量进行,并且只有在比较结果一致的情况下才会进行更新,适合对于多个线程并发更新同一个变量的情况。而synchronized可以用于对代码块或方法进行同步,可以保护多个线程访问共享资源的一致性。 最后,CAS相对于synchronized更容易产生ABA问题。ABA问题是指在CAS操作中,如果变量的值在操作过程中被其他线程修改过多次,并且最后又变回原来的值,那么CAS操作会错误地认为没有被修改过。为了解决ABA问题,Java提供了AtomicStampedReference和AtomicMarkableReference等类。 总的来说,CAS适用于对单个变量进行高效的并发更新,而synchronized适用于保护共享资源的一致性以及对代码块或方法进行同步。根据实际需求和场景的不同,可以选择合适的技术来实现线程安全。 ### 回答3: CAS(Compare and Swap)和synchronized是两种用于实现线程安全的机制。 CAS是一种乐观锁机制,它使用原子操作来实现线程安全。CAS包含三个操作数:内存位置V、旧的预期值A和要更新的新值B。当执行CAS操作时,只有在内存位置的值与预期值相等时,CAS才会修改内存位置的值为新值B;否则,不做任何操作。由于CAS操作是原子的,所以能够保证只有一个线程能够成功地修改内存位置的值。但如果多个线程同时执行CAS操作,只有一个线程会成功,其他线程需要重试。 synchronized是一种悲观锁机制,它使用互斥锁来实现线程安全。在使用synchronized关键字修饰的代码块或方法中,同一时刻只有一个线程可以执行该代码块或方法。其他线程需要等待上一个线程执行完毕并释放锁后才能执行。synchronized能够保证代码块或方法的互斥访问,避免了并发访问的问题。但由于每次只能有一个线程执行,其他线程会进入阻塞状态,可能会造成性能问题。 总的来说,CAS是一种非阻塞的乐观锁机制,适用于竞争不激烈的情况,能够提高并发性能。而synchronized是一种阻塞的悲观锁机制,适用于竞争激烈的情况,能够保证线程安全,但可能降低并发性能。在实际开发中,我们需要根据具体的场景选择适合的机制来确保线程安全和并发性能。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值