Java并发编程Lock显示锁的了解

Lock是一个接口提供无条件的,可轮询的 ,定时的,可中断的锁获取操作,所有加锁和解锁的方法都是显示的。

Lock接口方法

  public abstract void lock(); //获取锁

这个就是平常用的最多的方法,用来获取锁,如果没有获取到锁则等待。不过为了避免产生死锁,一般都要跟try{}finally{} 一起使用

  lock.lock(); //获取锁
        try {
            //业务处理
        }finally {
            lock.unlock(); //释放锁
        }
    }
 public abstract void unlock(); //释放锁

用来释放锁,这个没什么好细说,用的时候,注意一定在finally包裹里使用,不然容易出现死锁。

    public abstract void lockInterruptibly()throws InterruptedException; //获取锁

这个方法是也是获取锁,跟lock() 跟相似,获取锁,获取不到,等待直到获取到锁为止。但是它是可以被中断的,这个方法直接throws InterruptedException 就是为了让调用者,自己去处理线程的中断。这个方法强调的是线程在获取锁的时候可以被中断,获取到锁的时候也可以被中断,详细看下面的例子。

 void method()throws InterruptedException{
try {
            //业务处理
        }finally {
            lock.unlock(); //释放锁
        }
}
public abstract boolean tryLock(); //获取锁,有返回值

这个方法与lock()主要的区别在,lock()方法获取不到锁,会等待,直到获取到锁为止。tryLock()只会获取一次,并将结果返回,获取到锁返回true,反之false。代码模板:

if(lock.tryLock()) {
            try {
            //上锁的业务处理   
            }finally {
                lock.unlock(); //释放锁
            }
        }else {
            //获取锁失败的业务处理
        }
  public void tryLock(long paramLong,TimeUnit paramTimeUnit) throws InterruptedException;

这个方法与tryLock()有点类似,在指定的时间内获取锁,如果没有获取到返回false,如果在指定的时间内,获取到锁则返回true。使用模板也与tryLock() 一样。

  public abstract Condition newCondition();

返回用来与此Lock实例一起使用的Condition实例。

Lock实现ReentrantLock 使用

lock() 方法使用

public class LockDemo {
    
    private List<String> list = new ArrayList<>();
    private final ReentrantLock lock = new ReentrantLock();
    
    public static void main(String[] args) throws InterruptedException {
        final LockDemo l = new LockDemo();
        for(int i = 0; i < 5 ; i++) {
          new Thread(()->{l.insert(Thread.currentThread());}).start(); 
        }
        Thread.sleep(2000); //为了看到list的结果,谁两秒钟
        System.out.println("执行完成了");
        l.list.forEach(System.out::println);
        
    }
    
   public   void insert(Thread thread) {
        lock.lock();
        try {
        System.out.println("此线程已经获取到锁了 "+thread.getName());
        list.add(thread.getName());
        }finally {
            lock.unlock();
            System.out.println("此线程已经释放锁了 "+thread.getName());
        }
    }
   }

}

执行结果

此线程已经获取到锁了 Thread-1
此线程已经释放锁了 Thread-1
此线程已经获取到锁了 Thread-0
此线程已经释放锁了 Thread-0
此线程已经获取到锁了 Thread-3
此线程已经释放锁了 Thread-3
此线程已经获取到锁了 Thread-4
此线程已经释放锁了 Thread-4
此线程已经获取到锁了 Thread-2
此线程已经释放锁了 Thread-2
执行完成了
Thread-1
Thread-0
Thread-3
Thread-4
Thread-2

可以看出获取锁的顺序跟list的顺序是一样。每个线程都是先获取到锁,释放锁在到下一个线程,不存在交叉的情况。

tryLock() 方法使用

public class LockDemo {
    private List<String> list = new ArrayList<>();
    private final ReentrantLock lock = new ReentrantLock();
    
    public static void main(String[] args) throws InterruptedException {
        final LockDemo l = new LockDemo();
        for(int i = 0; i < 5 ; i++) {
          new Thread(()->{l.tryInsert(Thread.currentThread());}).start(); 
        }
        Thread.sleep(2000);
        System.out.println("执行完成了");
        l.list.forEach(System.out::println);    
    }
   void tryInsert(Thread thread) {
       if(lock.tryLock()) {//tryLock()方法是有返回值的,它表示用来尝试获取锁,如果获取成功,则返回true,如果获取失败(
           System.out.println("此线程已经获取到锁了 "+thread.getName());                              //即锁已被其他线程获取),
           try {              //则返回false,也就说这个方法无论如何都会立即返回。在拿不到锁时不会一直在那等待。 这个方法就是线程 去竞争锁,获取到执行 不然到else里面去
           list.add(thread.getName());
           }finally {
               lock.unlock();
               System.out.println("此线程已经释放锁了 "+thread.getName());
           }
       }else {
           System.out.println("此线程获取锁失败 "+thread.getName());
       }
   }
}

执行的结果

此线程获取锁失败 Thread-0
此线程获取锁失败 Thread-3
此线程获取锁失败 Thread-1
此线程获取锁失败 Thread-2
此线程已经获取到锁了 Thread-4
此线程已经释放锁了 Thread-4
执行完成了
Thread-4

可以看出5个线程同时获取锁,只有一个线程成功的,其他的线程都进入else中。这个对应开头说的可轮询的,如果加上时间就是定时的。

lockInterruptibly()方法使用

public class LockDemo {

    private List<String> list = new ArrayList<>();
    private final ReentrantLock lock = new ReentrantLock();

    public static void main(String[] args) throws InterruptedException {
        final LockDemo l = new LockDemo();
        for (int i = 0; i < 5; i++) {
            Thread t = new Thread(() -> {
                l.interruptInsert(Thread.currentThread());
            });
            t.start();
            if (i % 2 == 0) {//对偶数进行中断,为了不让全部的线程都中断,list没有结果集
                Thread.sleep(10); //让线程获取去获取锁,在进行中断
                t.interrupt(); // 直接执行中断 
            }
        }
        Thread.sleep(5000);
        System.out.println("执行完成了");
        l.list.forEach(System.out::println);

    }

    void interruptInsert(Thread thread) {
        try {
            lock.lockInterruptibly();
            System.out.println("线程已经获取到锁了 : " + thread.getName());
            Thread.sleep(2000);
            list.add(thread.getName());
        } catch (InterruptedException e) {
            System.out.println("当前线程直接执行中断: " + thread.getName());
            e.printStackTrace();
        } finally {
            lock.unlock(); // 释放锁
            System.out.println("此线程已经释放锁了 " + thread.getName());
        }
    }
}

![image.png](https://upload-images.jianshu.io/upload_images/9213940-21a3f18890d75f77.png?imageMogr2/auto-orient/strip%7CimageView2/2/w/1240)

可以看出Thread-0,Thread-2 已经获取到锁了,还是被中断,Thread-4在等待获取锁,直接被中断的,终止等待。


          
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 206,968评论 6 482
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 88,601评论 2 382
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 153,220评论 0 344
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 55,416评论 1 279
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 64,425评论 5 374
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,144评论 1 285
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,432评论 3 401
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,088评论 0 261
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 43,586评论 1 300
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,028评论 2 325
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,137评论 1 334
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,783评论 4 324
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,343评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,333评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,559评论 1 262
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,595评论 2 355
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,901评论 2 345

推荐阅读更多精彩内容