IT数码 购物 网址 头条 软件 日历 阅读 图书馆
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
图片批量下载器
↓批量下载图片,美女图库↓
图片自动播放器
↓图片自动播放器↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁
 
   -> Java知识库 -> Java并发编程之可重入锁ReentrantLock -> 正文阅读

[Java知识库]Java并发编程之可重入锁ReentrantLock

1、简介

可重入锁(递归锁),指的是同一线程 外层方法获得锁之后 ,内部方法仍然可以申请获取到该锁。
在 JAVA 环境下 ReentrantLock 和 synchronized 都是 可重入锁。

2、构造器

非公平锁:线程之间相互竞争获取锁,未获取到锁的阻塞等待锁释放。

    public ReentrantLock() {
        sync = new NonfairSync();
    }

示例 1 非公平锁

    public static void demo01() throws InterruptedException {
        ReentrantLock nonFairLock = new ReentrantLock();

        Thread t1 = new Thread(getRunnable(nonFairLock), "t1");
        Thread t2 = new Thread(getRunnable(nonFairLock), "t2");
        Thread t3 = new Thread(getRunnable(nonFairLock), "t3");

        Thread.sleep(3000);
        t1.start();
        t2.start();
        t3.start();
    }

    private static Runnable getRunnable(ReentrantLock nonFairLock) {
        return () -> {
            String thName = Thread.currentThread().getName();
            System.out.println("线程:" + thName + " 竞争锁");
            try {
                nonFairLock.lock();
                System.out.println("线程:" + thName + " 获取锁,执行任务");
                long start = System.currentTimeMillis();
                Thread.sleep((long) (Math.random() * 10000));
                System.out.println("线程:" + thName + " 完成任务耗时:" + (System.currentTimeMillis() - start));
            } catch (InterruptedException e) {
                e.printStackTrace();
            } finally {
                System.out.println("线程:" + thName + " 释放锁=============================");
                nonFairLock.unlock();
            }
        };
    }

在这里插入图片描述
可重入:同一线程可多次竞争锁

示例 2 可重入

    public static void demo02() throws InterruptedException {
        ReentrantLock nonFairLock = new ReentrantLock();

        Thread t1 = new Thread(getRunnable2(nonFairLock), "t1");
        Thread t2 = new Thread(getRunnable2(nonFairLock), "t2");
        Thread t3 = new Thread(getRunnable2(nonFairLock), "t3");

        Thread.sleep(3000);
        t1.start();
        t2.start();
        t3.start();
    }
    
    private static Runnable getRunnable2(ReentrantLock nonFairLock) {
        return () -> {
            String thName = Thread.currentThread().getName();
            System.out.println("线程:" + thName + " 竞争锁");
            for (int i = 0; i < 2; i++) {
                try {
                    nonFairLock.lock();
                    System.out.println("线程:" + thName + " 获取锁,执行任务");
                    long start = System.currentTimeMillis();
                    Thread.sleep((long) (Math.random() * 10000));
                    System.out.println("线程:" + thName + " 完成任务耗时:" + (System.currentTimeMillis() - start));
                } catch (InterruptedException e) {
                    e.printStackTrace();
                } finally {
                    System.out.println("线程:" + thName + " 释放锁=============================");
                    nonFairLock.unlock();
                }
            }
        };
    }

在这里插入图片描述
公平锁:锁释放,由等待时间最长的线程获取锁,有顺序。

示例 3 公平锁

    public ReentrantLock(boolean fair) {
        sync = fair ? new FairSync() : new NonfairSync();
    }
    public static void demo03() throws InterruptedException {
        ReentrantLock fairLock = new ReentrantLock(true);

        Thread t1 = new Thread(getRunnable3(fairLock), "t1");
        Thread t2 = new Thread(getRunnable3(fairLock), "t2");
        Thread t3 = new Thread(getRunnable3(fairLock), "t3");

        Thread.sleep(3000);
        t1.start();
        t2.start();
        t3.start();
    }

    private static Runnable getRunnable3(ReentrantLock fairLock) {
        return () -> {
            String thName = Thread.currentThread().getName();
            System.out.println("线程:" + thName + " 竞争锁");
            for (int i = 0; i < 2; i++) {
                try {
                    fairLock.lock();
                    System.out.println("线程:" + thName + " 获取锁,执行任务");
                    long start = System.currentTimeMillis();
                    Thread.sleep((long) (Math.random() * 10000));
                    System.out.println("线程:" + thName + " 完成任务耗时:" + (System.currentTimeMillis() - start));
                } catch (InterruptedException e) {
                    e.printStackTrace();
                } finally {
                    System.out.println("线程:" + thName + " 释放锁=============================");
                    fairLock.unlock();
                }
            }
        };
    }

在这里插入图片描述

3、主要方法

lock:获取锁,获取不到就阻塞线程

   public void lock() {
        sync.lock();
    }

unlock:释放锁

    public void unlock() {
        sync.release(1);
    }

tryLock:尝试获取锁,立即返回获取结果,不会阻塞线程

public boolean tryLock() {
        return sync.nonfairTryAcquire(1);
    }

设置超时时间参数,该参数时间内获取不到锁就等待,直到超过该参数时间

    public boolean tryLock(long timeout, TimeUnit unit)
            throws InterruptedException {
        return sync.tryAcquireNanos(1, unit.toNanos(timeout));
    }

示例 4 tryLock

    public static void demo04() throws InterruptedException {
        ReentrantLock nonFairLock = new ReentrantLock();

        Thread t1 = new Thread(getRunnable4(nonFairLock), "t1");
        Thread t2 = new Thread(getRunnable4(nonFairLock), "t2");
        Thread t3 = new Thread(getRunnable4(nonFairLock), "t3");

        Thread.sleep(3000);
        t1.start();
        t2.start();
        t3.start();
    }

    private static Runnable getRunnable4(ReentrantLock nonFairLock) {
        return () -> {
            String thName = Thread.currentThread().getName();
            System.out.println("线程:" + thName + " 竞争锁");
            try {
                while (true) {
                    if (nonFairLock.tryLock()) {
                        System.out.println("线程:" + thName + " 获取锁,执行任务");
                        long start = System.currentTimeMillis();
                        Thread.sleep((long) (Math.random() * 10000));
                        System.out.println("线程:" + thName + " 完成任务耗时:" + (System.currentTimeMillis() - start));
                        break;
                    } else {
                        System.out.println("线程:" + thName + " 锁已被持有,先干点别的吧");
                        Thread.sleep(5000);
                    }
                }
            } catch (InterruptedException e) {
                e.printStackTrace();
            } finally {
                System.out.println("线程:" + thName + " 释放锁=============================");
                nonFairLock.unlock();
            }
        };
    }

在这里插入图片描述

示例 5 设置超时时间

    private static Runnable getRunnable4(ReentrantLock nonFairLock) {
        return () -> {
            String thName = Thread.currentThread().getName();
            System.out.println("线程:" + thName + " 竞争锁");
            try {
                while (true) {
                    if (nonFairLock.tryLock(5000, TimeUnit.MILLISECONDS)) {
                        System.out.println("线程:" + thName + " 获取锁,执行任务");
                        long start = System.currentTimeMillis();
                        Thread.sleep((long) (Math.random() * 10000));
                        System.out.println("线程:" + thName + " 完成任务耗时:" + (System.currentTimeMillis() - start));
                        break;
                    } else {
                        System.out.println("线程:" + thName + " 锁已被持有,先干点别的吧");
                    }
                }
            } catch (InterruptedException e) {
                e.printStackTrace();
            } finally {
                System.out.println("线程:" + thName + " 释放锁=============================");
                nonFairLock.unlock();
            }
        };
    }

在这里插入图片描述
lockInterruptibly:持有锁线程中断,释放锁,不会继续占有

    public void lockInterruptibly() throws InterruptedException {
        sync.acquireInterruptibly(1);
    }

示例 6 lockInterruptibly

   public static void demo05() throws InterruptedException {
        ReentrantLock nonFairLock = new ReentrantLock();

        Thread t1 = new Thread(getRunnable5(nonFairLock), "t1");
        Thread t2 = new Thread(getRunnable5(nonFairLock), "t2");

        Thread.sleep(3000);
        t1.start();
        t2.start();

        Thread.sleep(1000);
        t2.interrupt();
    }

    private static Runnable getRunnable5(ReentrantLock nonFairLock) {
        return () -> {
            String thName = Thread.currentThread().getName();
            System.out.println("线程:" + thName + " 竞争锁");
            try {
                nonFairLock.lockInterruptibly();
                System.out.println("线程:" + thName + " 获取锁,执行任务");
                long start = System.currentTimeMillis();
                Thread.sleep((long) (Math.random() * 10000));
                System.out.println("线程:" + thName + " 完成任务耗时:" + (System.currentTimeMillis() - start));
            } catch (InterruptedException e) {
                System.out.println("线程:" + thName + " 线程中断");
            } finally {
                System.out.println("线程:" + thName + " 释放锁=============================");
                nonFairLock.unlock();
            }
        };
    }

在这里插入图片描述

  Java知识库 最新文章
计算距离春节还有多长时间
系统开发系列 之WebService(spring框架+ma
springBoot+Cache(自定义有效时间配置)
SpringBoot整合mybatis实现增删改查、分页查
spring教程
SpringBoot+Vue实现美食交流网站的设计与实
虚拟机内存结构以及虚拟机中销毁和新建对象
SpringMVC---原理
小李同学: Java如何按多个字段分组
打印票据--java
上一篇文章      下一篇文章      查看所有文章
加:2022-04-09 18:09:53  更:2022-04-09 18:10:06 
 
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁

360图书馆 购物 三丰科技 阅读网 日历 万年历 2024年11日历 -2024/11/24 5:44:43-

图片自动播放器
↓图片自动播放器↓
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
图片批量下载器
↓批量下载图片,美女图库↓
  网站联系: qq:121756557 email:121756557@qq.com  IT数码