首页 > 其他分享 >ReentrantLock 简单使用

ReentrantLock 简单使用

时间:2024-07-13 18:56:43浏览次数:14  
标签:Account 简单 money ReentrantLock 使用 tryLock lock public

摘自:《Java 编程的逻辑》

Java 并发包中的提供了显式锁,它可以解决 synchronized 的一些限制。

Java 并发包中的显式锁接口和类位于包 java.util.concurrent.locks 下,主要接口和类有:

❑ 锁接口 Lock,主要实现类是 ReentrantLock;

❑ 读写锁接口 ReadWriteLock,主要实现类是 ReentrantReadWriteLock。

下面介绍接口 Lock 和实现类 ReentrantLock。

Lock 接口

public interface Lock {
    void lock();
    void lockInterruptibly() throws InterruptedException;
    boolean tryLock();
    boolean tryLock(long time, TimeUnit unit) throws InterruptedException;
    void unlock();
    Condition newCondition();
}

1)lock()/unlock():就是普通的获取锁和释放锁方法,lock() 会阻塞直到成功。

2)lockInterruptibly():与 lock() 的不同是,它可以响应中断(参见 82219997),如果被其他线程中断了,则抛出 InterruptedException。

3)tryLock():只是尝试获取锁,立即返回,不阻塞,如果获取成功,返回 true,否则返回 false。

4)tryLock(long time, TimeUnit unit):先尝试获取锁,如果能成功则立即返回 true,否则阻塞等待,但等待的最长时间由指定的参数设置,在等待的同时响应中断,如果发生了中断,抛出 InterruptedException,如果在等待的时间内获得了锁,返回 true,否则返回 false。

5)newCondition:新建一个条件,一个 Lock 可以关联多个条件。

可重入锁 ReentrantLock

1.基本用法

Lock 接口的主要实现类是 ReentrantLock,它的基本用法 lock/unlock 实现了与 synchronized 一样的语义,包括:

❑ 可重入,一个线程在持有一个锁的前提下,可以继续获得该锁;

❑ 可以解决竞态条件问题;

❑ 可以保证内存可见性。

ReentrantLock 有两个构造方法:

public ReentrantLock()
public ReentrantLock(boolean fair)

参数 fair 表示是否保证公平,不指定的情况下,默认为 false,表示不保证公平。所谓公平是指,等待时间最长的线程优先获得锁。保证公平会影响性能,一般也不需要,所以默认不保证,synchronized 锁也是不保证公平的。

使用显式锁,一定要记得调用 unlock。一般而言,应该将 lock 之后的代码包装到 try 语句内,在 finally 语句内释放锁。比如,使用 ReentrantLock 实现 Counter,代码可以为:

public class Counter {
    private final Lock lock = new ReentrantLock();
    private volatile int count;
    public void incr() {
        lock.lock();
        try {
            count++;
        } finally {
            lock.unlock();
        }
    }
    public int getCount() {
        return count;
    }
}

2.使用 tryLock 避免死锁

使用 tryLock(),可以避免死锁。在持有一个锁获取另一个锁而获取不到的时候,可以释放已持有的锁,给其他线程获取锁的机会,然后重试获取所有锁。

我们来看个例子,银行账户之间转账,用类 Account 表示账户。

代码清单 1 表示账户的类 Account

public class Account {
    private Lock lock = new ReentrantLock();
    private volatile double money;
    public Account(double initialMoney) {
        this.money = initialMoney;
    }
    public void add(double money) {
        lock.lock();
        try {
            this.money += money;
        } finally {
            lock.unlock();
        }
    }
    public void reduce(double money) {
        lock.lock();
        try {
            this.money -= money;
        } finally {
            lock.unlock();
        }
    }
    public double getMoney() {
        return money;
    }
    void lock() {
        lock.lock();
    }
    void unlock() {
        lock.unlock();
    }
    boolean tryLock() {
        return lock.tryLock();
    }
}

Account 里的 money 表示当前余额,add/reduce 用于修改余额。在账户之间转账,需要两个账户都锁定,如果不使用 tryLock,而直接使用 lock,则代码如代码清单 2 所示。

代码清单 2 转账的错误写法

public class AccountMgr {
    public static class NoEnoughMoneyException extends Exception {}
    public static void transfer(Account from, Account to, double money)
            throws NoEnoughMoneyException {
        from.lock();
        try {
            to.lock();
            try {
                if(from.getMoney() >= money) {
                    from.reduce(money);
                    to.add(money);
                } else {
                    throw new NoEnoughMoneyException();
                }
            } finally {
                to.unlock();
            }
        } finally {
            from.unlock();
        }
    }
}

但这么写是有问题的,如果两个账户都同时给对方转账,都先获取了第一个锁,则会发生死锁。我们写段代码来模拟这个过程,如代码清单 3 所示。

代码清单 3 模拟账户转账的死锁过程

public static void simulateDeadLock() {
    final int accountNum = 10;
    final Account[] accounts = new Account[accountNum];
    final Random rnd = new Random();
    for(int i = 0; i < accountNum; i++) {
        accounts[i] = new Account(rnd.nextInt(10000));
    }
    int threadNum = 100;
    Thread[] threads = new Thread[threadNum];
    for(int i = 0; i < threadNum; i++) {
        threads[i] = new Thread() {
            public void run() {
                int loopNum = 100;
                for(int k = 0; k < loopNum; k++) {
                    int i = rnd.nextInt(accountNum);
                    int j = rnd.nextInt(accountNum);
                    int money = rnd.nextInt(10);
                    if(i ! = j) {
                        try {
                              transfer(accounts[i], accounts[j], money);
                        } catch (NoEnoughMoneyException e) {
                        }
                    }
                }
            }
        };
        threads[i].start();
    }
}

以上代码创建了 10 个账户,100 个线程,每个线程执行 100 次循环,在每次循环中,随机挑选两个账户进行转账。

我们使用 tryLock 来进行修改,先定义一个 tryTransfer 方法,如代码清单 4 所示。

代码清单 4 使用 tryLock 尝试转账

public static boolean tryTransfer(Account from, Account to, double money)
              throws NoEnoughMoneyException {
    if(from.tryLock()) {
        try {
            if(to.tryLock()) {
                try {
                    if(from.getMoney() >= money) {
                        from.reduce(money);
                        to.add(money);
                    } else {
                        throw new NoEnoughMoneyException();
                    }
                    return true;
                } finally {
                    to.unlock();
                }
            }
        } finally {
            from.unlock();
        }
    }
    return false;
}

如果两个锁都能够获得,且转账成功,则返回 true,否则返回 false。不管怎样,结束都会释放所有锁。transfer 方法可以循环调用该方法以避免死锁,代码可以为:

public static void transfer(Account from, Account to, double money)
        throws NoEnoughMoneyException {
    boolean success = false;
    do {
        success = tryTransfer(from, to, money);
        if(!success) {
            Thread.yield();
        }
    } while (!success);
}

除了实现 Lock 接口中的方法,ReentrantLock 还有一些其他方法,通过它们,可以获取关于锁的一些信息,这些信息可以用于监控和调试目的,具体可参看 API 文档,就不介绍了。

对比 ReentrantLock 和 synchronized

相比 synchronized, ReentrantLock 可以实现与 synchronized 相同的语义,而且支持以非阻塞方式获取锁,可以响应中断,可以限时,更为灵活。不过,synchronized 的使用更为简单,写的代码更少,也更不容易出错。

synchronized 代表一种声明式编程思维,程序员更多的是表达一种同步声明,由 Java 系统负责具体实现,程序员不知道其实现细节;显式锁代表一种命令式编程思维,程序员实现所有细节。

声明式编程的好处除了简单,还在于性能,在较新版本的 JVM 上,ReentrantLock 和 synchronized 的性能是接近的,但 Java 编译器和虚拟机可以不断优化 synchronized 的实现,比如自动分析 synchronized 的使用,对于没有锁竞争的场景,自动省略对锁获取/释放的调用。

简单总结下,能用 synchronized 就用 synchronized,不满足要求时再考虑 ReentrantLock。

标签:Account,简单,money,ReentrantLock,使用,tryLock,lock,public
From: https://www.cnblogs.com/Higurashi-kagome/p/18300492

相关文章

  • 使用llama.cpp量化模型
    文章目录概要整体实验流程技术细节小结概要大模型量化是指在保持模型性能尽可能不变的情况下,通过减少模型参数的位数来降低模型的计算和存储成本。本次实验环境为魔搭社区提供的免费GPU环境(24G),使用Llama.cpp进行4bit量化可以大幅减少大语言模型的内存占用,并提高推理......
  • 使用 pip 和 requirements.txt 安装 Python 包
    1.构建当前项目需要的软件包pip3installpipreqspip3installpip-toolspipreqs--savepath=requirements.in&&pip-compile1.根据配置文件批量安装软件包。requirements.txt在某些环境中,可使用pip3而不是pippipinstall-rrequirements.txt2.检查当前环境......
  • FOFA网络空间安全搜索引擎的使用
    一、FOFA是什么?FOFA是一款网络空间测绘的搜索引擎,旨在帮助用户以搜索的方式查找公网上的互联网资产。简单来说,FOFA的使用方式类似于谷歌或百度,用户可以输入关键词来匹配包含该关键词的数据。不同的是,这些数据不仅包括像谷歌或百度一样的网页,还包括像摄像头、打印机、数据库......
  • package.json 脚本配置使用环境文件
    脚本使用环境文件"scripts":{"dev":"vite","build:prod":"vitebuild","build:stage":"vitebuild--modestaging","preview":"vitepreview"}dev:运行开发服务器,默认使用.......
  • [JS] generator基本使用
    next方法与yield关键字generator函数可以返回一个迭代器,通过next方法切换generator的状态。generator函数被调用时并不会执行内部的语句,而是返回一个迭代器对象。迭代器对象首次调用next方法,才开始执行generator函数的语句。直到遇到yield语句,内部的执行中断,返回yield关键字右......
  • 哪些场景下适合使用人工智能作词软件来写歌词
    以下是一些适合使用人工智能作词软件的场景:软件我们选用“妙笔生词”智能写歌词软件(53650899)来操作。 1.创作灵感枯竭时:当创作者陷入思维困境,找不到新的创意和方向,人工智能作词软件可以快速提供一些灵感和初步的构思。2.提高创作效率:对于有大量作词需求,且时间紧迫的情况,例如......
  • `require.context` 参数和使用
    require.context是Webpack提供的一个方法,用于创建自己的(上下文)模块。这些模块可以动态导入符合特定条件的文件。在使用require.context时,你需要提供三个参数:directory:要搜索的文件夹路径(字符串)。useSubdirectories:是否递归地搜索子文件夹(布尔值)。regExp:匹配文件的......
  • 使用python绘制3D柱状图+完整代码展示
    一、首先进行代码效果图的展示        这是一个简单的3D模型图的展示,我们可以从官网上看到有类似的模型代码,但是大部分都没有加上全局系统配置,整体效果很单一,看不出来有什么特色,我们可以通过了解我们的python绘图工具pyecharts库。二、了解代码这个就是我们的全局......
  • 【Python】jupyter notebook平台的使用·
    目录一、安装Anaconda二、将BreadCancer.zip上传到jupyter notebook平台中三、了解BreadCancerClassifier.ipynb文件在jupyternotebook的单元格中的python代码,并运行。3.1 导入mainFun文件3.2 读入数据3.3开始训练3.4读入测试数据3.5 开始测试3.6 开始统计3......
  • 使用 lmdeploy 部署 internlm/internlm2_5-7b-chat
    使用lmdeploy部署internlm/internlm2_5-7b-chat0.引言1.lmdeploy性能2.lmdeploy支持的模型3.快速开始0.引言LMDeploy由MMDeploy和MMRazor团队联合开发,是涵盖了LLM任务的全套轻量化、部署和服务解决方案。这个强大的工具箱提供以下核心功能:高效的......