浅谈Java中的锁机制介绍

浅谈Java中的锁

一、    悲观锁和乐观锁

    悲观锁(Pessimistic Lock):顾名思义,就是很悲观,每次去拿数据的时候都认为别人会修改,所以每次在拿数据的时候都会上锁,这样别人想拿这个数据就会block直到它拿到锁。传统的关系型数据库里边就用到了很多这种锁机制,比如行锁,表锁等,读锁,写锁等,都是在做操作之前先上锁。for update就是一种。

    乐观锁(Optimistic Lock):顾名思义,就是很乐观,每次去拿数据的时候都认为别人不会修改,所以不会上锁,但是在更新的时候会判断一下在此期间别人有没有去更新这个数据,可以使用版本号等机制。乐观锁适用于多读的应用类型,这样可以提高吞吐量,像数据库如果提供类似于write_condition机制的其实都是提供的乐观锁。Java中常用的乐观锁方式有版本号、时间戳。

 

 

二、    公平锁和非公平锁

 

    锁的公平与非公平,是指线程请求获取锁的过程中,是否允许插队。在公平锁上,线程将按他们发出请求的顺序来获得锁;而非公平锁则允许在线程发出请求后立即尝试获取锁,如果可用则可直接获取锁,尝试失败才进行排队等待。

 

    公平锁:多个线程申请获取同一个锁,按照线程的申请顺序,排队获取锁。公平锁的好处是等待的线程不会被饿死,相应的缺陷就是整体吞吐量很低、效率很低。使用new ReentrantLock(true)可以构造一个公平锁。

 

    非公平锁:多个线程申请获取同一个锁,获取锁的顺序不按照申请顺序,抢占式的获取。非公平锁的好处是整体效率很高,但是可能会使有些线程一致在等待,造成饿死。使用Synchronized、new ReentrantLock()和new ReentrantLock(false)可以构建一个非公平锁。

 

 

三、    读写锁

 

    读写锁一次只有一个线程(writer线程)可以修改共享数据,但在许多情况下,任何数量的线程可以同时读取共享数据(reader线程)。Java中的读写锁通过ReentrantReadWriteLock实现。ReentrantReadWriteLock.ReadLock是读锁,它是共享锁。ReentrantReadWriteLock.WriteLock是写锁,它是独占锁。

 

四、    互斥锁

 

 

        互斥锁即一次只能有一个线程持有的锁。ReentrantLock和synchronized都是互斥锁。

 

    ReentrantLock是jdk5的新特性,采用ReentrantLock可以完全替代替换synchronized传统的锁机制,而且采用ReentrantLock的方式更加面向对象,也更加灵活

 

 

 

五、    共享锁和独占锁

 

    共享锁:允许多个线程同时获取锁,并发访问共享资源,如:ReadWriteLock。在实际使用过程中,线程A获取到了共享资源D的共享锁,其它线程只能获取D的共享锁,不能获取独占锁。

 

    独占锁:一次只能有一个线程获得锁,即只能被一个线程持有。在实际使用过程中,线程A获取到了共享资源D的独占锁,其它线程不能获取D的任何类型锁。(ReentrantLock就是以独占方式实现的互斥锁。)

 

 

 

六、    偏向锁、轻量级锁、重量级锁

 

    这三个锁都是针对synchronized来说的,具体的实现细节比较复杂,具体实现可查看上面的阻塞锁里提到的。

偏向锁:对于一段同步代码来说,锁偏向于第一次获取它的线程,如果继续执行的过程中,锁没有被其它线程持有,则持有偏向锁的线程将不需要同步,自动获取锁。加锁和解锁不需要额外的消耗,和执行非同步方法比仅存在纳秒级的差距。如果线程间存在锁竞争,会带来额外的锁撤销的消耗。适用于只有一个线程访问同步块场景。

 

    轻量级锁:当偏向锁被另一个线程持有的时候,偏向锁升级为轻量级锁,其它线程通过自旋转的方式尝试获取锁。竞争的线程不会阻塞,提高了程序的响应速度。如果始终得不到锁竞争的线程使用自旋会消耗CPU。适用于追求响应时间、同步块执行速度非常快。

 

    重量级锁:当轻量级锁被另一个线程持有的时候,轻量级锁升级为重量级锁。线程竞争不使用自旋,不会消耗CPU。缺点就是线程阻塞,响应时间缓慢。适用于追求吞吐量,同步块执行速度较长。

 

七、    自旋锁

 

    Java线程在得不到锁时不会立即阻塞,而是执行一个循环,不断的去尝试获取锁,这种技术就是自旋锁。它可以减少在获取锁的过程中,因为线程上下文的切换而导致的额外消耗。

 

八、    可重入锁

 

    可重入锁,也称为递归锁,即线程在获取到某方法的锁之后,如果在该方法内部调用其它方法,这个方法也需要获取锁,那么进入这个方法将自动获取锁,它可以在一定程度上避免死锁。

九、   可中断锁

    中断(Interrupt)一个线程意味着在该线程完成任务之前停止其正在进行的一切,有效地中止其当前的操作。线程是死亡、还是等待新的任务或是继续运行至下一步,就取决于这个程序。虽然初次看来它可能显得简单,但是,你必须进行一些预警以实现期望的结果。

从上面的介绍知道,interrupt()并不会使线程停止运行,那如何停止线程呢?

    中断线程最好的,最受推荐的方式是,使用共享变量(shared variable)发出信号,告诉线程必须停止正在运行的任务。线程必须周期性的核查这一变量(尤其在冗余操作期间),然后有秩序地中止任务。

    ReentrantLock可中断锁的介绍。可中断锁是通过ReentrantLock提供的lockInterruptibly()方法实现的。


  • 0
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值