当前位置: 移动技术网 > IT编程>开发语言>Java > 探索JAVA并发 - 可重入锁和不可重入锁

探索JAVA并发 - 可重入锁和不可重入锁

2019年12月21日  | 移动技术网IT编程  | 我要评论

张宪瑞,斗战神天坑树洞在哪,楚民网

本人免费整理了java高级资料,涵盖了java、redis、mongodb、mysql、zookeeper、spring cloud、dubbo高并发分布式等教程,一共30g,需要自己领取。
传送门:https://mp.weixin.qq.com/s/jzddfh-7ynudmkjt0irl8q

 

锁的简单应用

用lock来保证原子性(this.count++这段代码称为临界区)

什么是原子性,就是不可分,从头执行到尾,不能被其他线程同时执行。

可通过cas来实现原子操作

cas(compare and swap):

cas操作需要输入两个数值,一个旧值(期望操作前的值)和一个新值,在操作期间先比较下旧值有没有发生变化,如果没有发生变化,才交换成新值,发生了变化则不交换。

cas主要通过compareandswapxxx()方法来实现,而这个方法的实现需要涉及底层的unsafe类

unsafe类:java不能直接访问操作系统底层,而是通过本地方法来访问。unsafe类提供了硬件级别的原子操作

public class counter{
    private lock lock = new lock();
    private int count = 0;
    public int inc(){
        lock.lock();
        this.count++;
        lock.unlock();
        return count;
    }
}

 

 

不可重入锁

先来设计一种锁

public class lock{
    private boolean islocked = false;
    public synchronized void lock() throws interruptedexception{
        while(islocked){    
            wait();
        }
        islocked = true;
    }
    public synchronized void unlock(){
        islocked = false;
        notify();
    }
}

 

这其实是个不可重入锁,举个例子

public class count{
    lock lock = new lock();
    public void print(){
        lock.lock();
        doadd();
        lock.unlock();
    }
    public void doadd(){
        lock.lock();
        //do something
        lock.unlock();
    }
}

 

当调用print()方法时,获得了锁,这时就无法再调用doadd()方法,这时必须先释放锁才能调用,所以称这种锁为不可重入锁,也叫自旋锁。

可重入锁

设计如下:

public class lock{
    boolean islocked = false;
    thread  lockedby = null;
    int lockedcount = 0;
    public synchronized void lock()
            throws interruptedexception{
        thread thread = thread.currentthread();
        while(islocked && lockedby != thread){
            wait();
        }
        islocked = true;
        lockedcount++;
        lockedby = thread;
    }
    public synchronized void unlock(){
        if(thread.currentthread() == this.lockedby){
            lockedcount--;
            if(lockedcount == 0){
                islocked = false;
                notify();
            }
        }
    }
}

 

 

相对来说,可重入就意味着:线程可以进入任何一个它已经拥有的锁所同步着的代码块。

第一个线程执行print()方法,得到了锁,使lockedby等于当前线程,也就是说,执行的这个方法的线程获得了这个锁,执行add()方法时,同样要先获得锁,因不满足while循环的条件,也就是不等待,继续进行,将此时的lockedcount变量,也就是当前获得锁的数量加一,当释放了所有的锁,才执行notify()。

如果在执行这个方法时,有第二个线程想要执行这个方法,因为lockedby不等于第二个线程,导致这个线程进入了循环,也就是等待,不断执行wait()方法。只有当第一个线程释放了所有的锁,执行了notify()方法,第二个线程才得以跳出循环,继续执行。

这就是可重入锁的特点。

java中常用的可重入锁

synchronized

java.util.concurrent.locks.reentrantlock

  • atomicintegerfieldupdater:原子更新整型的字段的更新器
  • atomiclongfieldupdater:原子更新长整型字段的更新器
  • atomicstampedreference:原子更新带有版本号的引用类型。该类将整型数值与引用关联起来,可用于原子的更新数据和数据的版本号,可以解决使用cas进行原子更新时可能出现的aba问题。
  • atomicreference :原子更新引用类型
  • atomicreferencefieldupdater :原子更新引用类型里的字段
  • atomicmarkablereference:原子更新带有标记位的引用类型。可以原子更新一个布尔类型的标记位和应用类型
  • atomicintegerarray :原子更新整型数组里的元素
  • atomiclongarray :原子更新长整型数组里的元素
  • atomicreferencearray : 原子更新引用类型数组的元素
  • atomicbooleanarray :原子更新布尔类型数组的元素
  • atomicboolean :原子更新布尔类型
  • atomicinteger: 原子更新整型
  • atomiclong: 原子更新长整型

如对本文有疑问,请在下面进行留言讨论,广大热心网友会与你互动!! 点击进行留言回复

相关文章:

验证码:
移动技术网