Java多线程银行账户同步:利用wait/notifyAll机制实现安全存取

Java多线程银行账户同步:利用wait/notifyAll机制实现安全存取

本文深入探讨了在java多线程环境中,如何有效实现共享银行账户的并发存取。通过一个模拟两人操作同一账户的场景,我们将详细介绍如何利用`synchronized`关键字以及`wait()`和`notifyall()`方法,在账户类中正确管理线程间的协作与资源同步,确保账户余额的线程安全,并避免常见的并发问题,从而构建一个健壮、高效的并发应用。

在多线程编程中,当多个线程共享同一个资源时,必须采取适当的同步机制来防止数据不一致和竞态条件。银行账户的存取操作是一个典型的共享资源问题,需要确保在并发环境下,账户余额能够被正确地更新,并且满足特定的业务规则(如最大存款额、最小取款额)。Java提供了synchronized关键字以及Object类的wait()、notify()和notifyAll()方法来实现线程间的同步与协作。

1. 核心概念回顾

在深入实现之前,我们先回顾几个关键的Java并发概念:

synchronized关键字:用于修饰方法或代码块,确保在任何时刻只有一个线程可以执行被synchronized保护的代码。它提供了互斥锁,防止多个线程同时修改共享数据。wait()方法:当一个线程在synchronized代码块中遇到某个条件不满足时(例如,账户余额不足或已达上限),可以调用wait()方法。调用wait()会使当前线程释放它所持有的锁,并进入等待状态,直到被其他线程唤醒。notify()和notifyAll()方法:用于唤醒在同一对象上调用了wait()方法的线程。notify()随机唤醒一个等待线程,而notifyAll()则唤醒所有等待线程。被唤醒的线程会尝试重新获取锁,并在获取到锁后从wait()的地方继续执行。

wait()、notify()和notifyAll()方法必须在synchronized方法或synchronized块中调用,并且它们作用的对象必须是当前线程所持有的锁对象。

2. 共享账户同步问题分析

假设我们有一个银行账户,由两个人(两个线程)共享。他们需要并发地进行存款(ingreso)和取款(retiro)操作。账户有最大存款限制(500欧元)和最小余额限制(1欧元)。当存款操作导致余额超出上限时,或者取款操作导致余额低于下限时,当前线程应等待,直到条件允许。当其他线程改变了账户状态后,应唤醒等待的线程。

立即学习“Java免费学习笔记(深入)”;

初始的尝试可能将wait()和notifyAll()放在线程类(Persona)中,试图让线程自身等待和唤醒。然而,这种做法是错误的。wait()和notifyAll()必须作用于共享资源的锁对象上,即本例中的Cuenta(账户)实例。如果将它们放在Persona类中,每个Persona实例都有自己的锁对象,线程之间无法通过同一个锁对象进行有效的通信和协调。正确的做法是,将这些同步和通信逻辑封装在共享资源(Cuenta)的方法中。

3. 解决方案设计与实现

我们将设计三个类来模拟这个场景:BPA(主应用类)、Persona(人物/线程类)和Cuenta(账户类)。

3.1 Cuenta类:共享账户

Cuenta类是共享资源,它的ingreso(存款)和retiro(取款)方法必须是synchronized的,以确保每次只有一个线程能够修改账户余额。同时,这些方法内部将包含条件判断和wait()/notifyAll()逻辑。

package BPA;import java.util.Random;public class Cuenta {    private int saldo; // 当前余额    private final int saldoMax; // 最大余额    private final int saldoMin = 1; // 最小余额    public Cuenta(int saldoInicial, int saldoMaximo) {        this.saldo = saldoInicial;        this.saldoMax = saldoMaximo;    }    /**     * 取款操作     * @param nombre 执行操作的人名     * @param cuenta 账户实例 (虽然参数中包含,但实际操作的是当前对象)     */    public synchronized void retiro(String nombre, Cuenta cuenta) {        int dinero;        Random rnd = new Random();        // 随机生成取款金额,范围在0到349之间        dinero = (int) (rnd.nextDouble() * 350.0);        // 检查取款后是否会低于最小余额        while ((saldo - dinero)  saldoMax) {            System.out.println(" **** 账户已满!" + nombre + " 尝试存入: " + dinero + " 欧元. 当前余额: " + getSaldo() + " 欧元. ****");            System.out.println(" ---- " + nombre + " 正在等待取款,以便能存入更多钱 ---- ");            try {                // 余额已满,当前线程等待,并释放Cuenta对象的锁                wait();             } catch (InterruptedException e) {                System.out.println("存款等待被中断");                Thread.currentThread().interrupt(); // 重新设置中断标志            }            // 唤醒后,重新检查条件,防止虚假唤醒            dinero = (int) (rnd.nextDouble() * 350.0); // 重新生成存款金额        }        // 条件满足,执行存款        this.saldo = this.saldo + dinero;        System.out.println("Name: " + nombre + " 存入: " + dinero + " 欧元. 总余额: " + getSaldo() + " 欧元.");        // 唤醒所有在Cuenta对象上等待的线程        notifyAll();     }    public int getSaldo() {        return saldo;    }}

关键点说明:

九歌 九歌

九歌–人工智能诗歌写作系统

九歌 322 查看详情 九歌 ingreso和retiro方法都是synchronized的,确保了对saldo的原子操作。while循环用于检查条件(saldo – dinero saldoMax)。使用while而不是if是为了处理“虚假唤醒”(Spurious Wakeup)问题,即线程可能在没有被notify()或notifyAll()唤醒的情况下意外地从wait()返回。每次唤醒后,线程都会重新检查条件。wait():当条件不满足时,调用此方法使当前线程进入等待状态并释放Cuenta对象的锁。notifyAll():每次成功存取后,调用此方法唤醒所有在Cuenta对象上等待的线程,让他们有机会重新检查条件并执行操作。

3.2 Persona类:操作账户的线程

Persona类继承自Thread,代表一个操作账户的个体。每个Persona实例都会持有同一个Cuenta对象的引用。

package BPA;import java.util.logging.Level;import java.util.logging.Logger;public class Persona extends Thread {    String nombre;    private Cuenta cuenta;    public Persona(String nombre, Cuenta cuenta) {        this.nombre = nombre;        this.cuenta = cuenta;    }    @Override    public void run() {        while (true) {            // 线程循环进行存款和取款操作            cuenta.ingreso(nombre, cuenta);            cuenta.retiro(nombre, cuenta);            try {                // 为了模拟真实场景和避免CPU空转,线程可以短暂休眠                Thread.sleep(500); // 每次操作后休眠500毫秒            } catch (InterruptedException e) {                System.out.println(nombre + " 的操作被中断.");                Thread.currentThread().interrupt(); // 重新设置中断标志                break; // 线程中断时退出循环            }        }    }}

关键点说明:

Persona线程的run()方法中包含一个无限循环,不断尝试进行存款和取款。Thread.sleep()是为了模拟操作之间的时间间隔,防止线程过于频繁地尝试获取锁,从而提高可读性和观察性。run()方法本身不需要synchronized,因为对共享资源的同步已由Cuenta类的方法处理。

3.3 BPA类:主应用

BPA类是应用程序的入口点,负责创建账户实例和Persona线程,并启动它们。

package BPA;import java.util.logging.Level;import java.util.logging.Logger;public class BPA {    public static void main(String[] args) {        // 创建一个共享账户实例,初始余额40欧元,最大余额500欧元        Cuenta laCuenta = new Cuenta(40, 500);        // 创建两个Persona线程,共享同一个Cuenta实例        Persona Ramon = new Persona("Ramon", laCuenta);        Persona Quique = new Persona("Quique", laCuenta);        // 启动线程        Quique.start();        Ramon.start();        try {            // 主线程等待两个Persona线程执行完毕            // 在此示例中,Persona线程是无限循环,所以join()会一直等待            // 如果希望程序在特定条件或时间后停止,需要额外的中断逻辑            Quique.join();            Ramon.join();        } catch (InterruptedException ex) {                  System.out.println("主线程等待被中断");            Thread.currentThread().interrupt();        }    }}

关键点说明:

main方法中创建了一个Cuenta对象,并将其传递给两个Persona线程,确保它们操作的是同一个共享账户。start()方法启动线程,使其执行run()方法。join()方法使主线程等待子线程执行完毕。在这个无限循环的例子中,join()会一直等待,直到线程被外部中断。

4. 运行效果与注意事项

运行此程序,您将看到Ramon和Quique两个线程交替地进行存款和取款操作。当账户余额不足以取款或达到存款上限时,相应的线程将进入等待状态,直到另一个线程通过存取操作改变了账户状态并调用notifyAll()将其唤醒。

重要注意事项:

wait()和notifyAll()的调用对象:它们必须在锁对象上调用。在本例中,ingreso和retiro方法是synchronized的,它们锁定了this(即Cuenta实例本身)。因此,wait()和notifyAll()也必须在Cuenta实例上调用。while循环检查条件:务必使用while循环而不是if语句来检查wait()唤醒后的条件。这是为了避免“虚假唤醒”以及确保在多线程竞争中,条件在线程真正执行前仍然满足。中断处理:在wait()和sleep()方法中,InterruptedException是必须处理的。通常,捕获到此异常后,应重新设置线程的中断状态 (Thread.currentThread().interrupt();),并根据业务逻辑决定是继续执行还是退出。notify() vs notifyAll():在本场景中,notifyAll()是更安全的选项。因为可能有多种原因导致线程等待(余额不足或余额已满),notifyAll()可以唤醒所有等待的线程,让他们重新评估条件,避免死锁。如果只用notify(),可能会唤醒一个无法继续执行的线程,导致其他能执行的线程继续等待。线程生命周期:本示例中的Persona线程是无限循环的。在实际应用中,您可能需要设计一个机制来优雅地停止这些线程,例如通过设置一个共享的volatile标志位并在run()循环中检查它。

5. 总结

通过上述实现,我们成功地解决了Java多线程环境下共享银行账户的并发存取问题。核心在于理解synchronized关键字提供的互斥访问,以及wait()和notifyAll()方法实现的线程间协作。正确地将这些机制应用于共享资源(Cuenta对象)的方法中,并遵循while循环检查条件、处理中断等最佳实践,是构建健壮并发应用的关键。这个模式不仅适用于银行账户,也适用于任何需要生产者-消费者或有限缓冲区的并发场景。

以上就是Java多线程银行账户同步:利用wait/notifyAll机制实现安全存取的详细内容,更多请关注创想鸟其它相关文章!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 chuangxiangniao@163.com 举报,一经查实,本站将立刻删除。
发布者:程序猿,转转请注明出处:https://www.chuangxiangniao.com/p/1076209.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2025年12月2日 07:15:36
下一篇 2025年12月2日 07:15:57

相关推荐

  • AI写代码 教你用PHP加GitHub Copilot开发小工具

    使用GitHub Copilot可高效开发PHP小工具,如字符串反转功能,通过注释引导生成代码,但需审查安全性与逻辑正确性,结合Xdebug调试、输入验证和输出转义,确保代码质量与安全,不可盲目依赖AI。 AI写代码,用PHP加GitHub Copilot开发小工具,确实能极大提升效率,但也要注意代…

    2025年12月10日 好文分享
    000
  • 日历事件时间段重叠检测:原理与实现

    本教程详细讲解了如何准确判断两个日历事件的时间段是否存在重叠。通过定义事件的开始和结束时间,我们利用逻辑条件判断它们是否相互交叉。文章将提供核心算法原理、实用的代码示例以及处理常见边缘情况的注意事项,旨在帮助开发者高效地实现事件冲突检测和日程管理功能,确保时间安排的准确性与合理性。 在日程管理、资源…

    2025年12月10日
    000
  • 高效判断日历事件时间重叠的原理与实现

    本文深入探讨了日历或排程系统中事件时间重叠的检测方法。通过阐述事件重叠的定义,并提出一种简洁而鲁棒的核心逻辑条件,即当一个事件的开始时间早于另一个事件的结束时间,且另一个事件的开始时间早于当前事件的结束时间时,两者即发生重叠。文章提供了具体的代码示例,并讨论了在实际应用中需要考虑的边界条件和性能优化…

    2025年12月10日
    000
  • TYPO3 8.7 CLI 外部导入错误:权限与缓存问题解决方案

    在 TYPO3 8.7 中,当尝试通过 CLI 命令行工具,使用 external_import 扩展导入数据时,可能会遇到诸如 “User doesn’t have enough rights for synchronizing table…” 或 …

    2025年12月10日
    000
  • TYPO3 8.7:CLI 外部导入错误解决方案

    在 TYPO3 8.7 中,当尝试通过命令行界面 (CLI) 使用 external_import 导入数据时,可能会遇到诸如权限不足或缓存写入失败等错误。这些错误通常与 CLI 环境下缺少必要的后端认证初始化有关。以下将详细介绍如何解决这些问题。 问题描述 在使用自定义 Extbase 扩展,并通…

    2025年12月10日
    000
  • Symfony 如何将邮件消息转为数组

    将 symfony email 对象转换为数组主要用于日志记录、api 传输、数据持久化和测试验证;具体做法是通过提取收件人、主题、正文等核心属性,并遍历头部和附件信息,其中自定义头部需过滤冗余项,附件内容建议 base64 编码后存入数组,但大文件应考虑存储路径而非直接嵌入内容,最终生成一个结构清…

    2025年12月10日
    000
  • PHP函数怎样让函数返回 true 或 false PHP函数布尔值返回的简单实现方法​

    php函数返回true或false最直接的方式是使用return true;或return false;语句,适用于表示操作成功与否或条件是否满足的场景,例如表单验证、状态检查等,通过明确的布尔类型声明: bool可避免类型转换带来的陷阱,同时建议使用is、has、can等前缀命名函数以提高可读性,…

    2025年12月10日
    000
  • PHP函数怎样给函数添加简单的注释说明 PHP函数注释编写的基础方法教程​

    给php函数添加注释最推荐的方式是使用phpdoc风格的文档块,因为它不仅提供清晰的说明,还能被ide和文档工具解析,提升代码可维护性和团队协作效率;相比单行或多行注释,phpdoc通过@param、@return等标签结构化描述函数的参数、返回值和异常,支持智能提示和自动文档生成,有效避免代码与注…

    2025年12月10日
    000
  • CodeIgniter 中动态嵌入 YouTube 视频教程

    本文详细阐述了如何在 CodeIgniter 应用程序中,从数据库动态获取并嵌入 YouTube 视频。教程涵盖了 YouTube 嵌入链接的正确格式、数据存储策略、CodeIgniter 视图中的实现方法,并提供了关键注意事项,旨在帮助开发者确保视频内容的流畅播放和良好的用户体验。 在现代 web…

    2025年12月10日
    000
  • CodeIgniter中动态嵌入YouTube视频教程:构建与优化

    本教程旨在指导开发者如何在CodeIgniter应用中动态嵌入YouTube视频。我们将详细探讨YouTube视频嵌入的正确URL格式、如何从数据库中获取视频ID并构建动态 例如,如果一个YouTube视频的观看链接是 https://www.youtube.com/watch?v=dQw4w9Wg…

    2025年12月10日
    000
  • CodeIgniter中动态嵌入YouTube视频教程:解决连接与路径问题

    本教程旨在指导开发者如何在CodeIgniter应用中动态嵌入存储在数据库中的YouTube视频。核心内容包括理解YouTube视频的正确嵌入URL格式、在视图层使用 观看链接示例: https://www.youtube.com/watch?v=dQw4w9WgXcQ嵌入链接示例: https:/…

    2025年12月10日
    000
  • 如何动态地在CodeIgniter中嵌入YouTube视频

    本文旨在指导开发者如何在CodeIgniter应用中动态地嵌入来自数据库的YouTube视频链接。我们将详细探讨YouTube视频嵌入的正确URL格式、如何在数据库中有效存储视频信息,以及在CodeIgniter控制器和视图中处理和展示这些动态链接的多种策略,同时涵盖重要的注意事项和最佳实践,确保视…

    2025年12月10日
    000
  • PHP常用框架怎样配置与使用邮件发送功能 PHP常用框架邮件服务的集成方法

    邮件进垃圾箱主因是发件人身份未验证,需配置SPF、DKIM、DMARC以提升域名信誉,确保邮件不被标记为垃圾邮件。 在PHP主流框架中,配置和使用邮件发送功能通常围绕着一个统一的邮件服务抽象层展开。这层服务允许开发者通过简单的API调用来发送邮件,底层则支持多种邮件驱动(如SMTP、API服务商如M…

    2025年12月10日
    000
  • Symfony 如何将实体转换为数组

    推荐使用Symfony序列化组件将实体转换为数组,通过定义序列化组(如user:read)并利用SerializerInterface的normalize方法,可精准控制输出字段及处理关联关系;对于简单场景,也可在实体内手动实现toArray()方法。 将Symfony的实体(Entity)转换为数…

    2025年12月10日
    000
  • 日历事件重叠检测:核心逻辑与编程实践

    本教程深入探讨日历事件重叠的检测方法。通过阐释事件重叠的核心逻辑,并提供Python代码示例,指导读者如何精确判断两个时间段是否交叉。文章还涵盖了时间区间表示、日期时间处理及性能优化等关键实践,旨在为开发人员构建高效日历系统提供实用指南。 在构建日历或日程管理系统时,一个核心功能是识别事件之间是否存…

    2025年12月10日
    000
  • PHP函数怎样使用回调函数处理事件 PHP函数回调函数应用的实用技巧

    回调函数通过解耦核心逻辑与响应操作实现事件处理,如用户注册后触发邮件发送、日志记录等;使用EventDispatcher类注册和分发事件,支持匿名函数、具名函数、类方法作为回调;通过事件对象封装数据可提升类型安全与扩展性,并支持传播控制;需注意作用域、异常处理、性能及调试问题,合理使用日志、队列与优…

    2025年12月10日
    000
  • 如何判断日历事件的重叠与交叉

    本文深入探讨了日历应用中判断事件时间区间是否重叠的核心逻辑。通过分析事件的开始和结束时间,文章提供了两种主要的重叠判断条件:一种是检查一个事件的端点是否落在另一个事件内部,另一种是更通用的基于区间边界的逻辑。文中包含详细的代码示例和关于边界条件处理、零时长事件以及多事件场景的注意事项,旨在帮助开发者…

    2025年12月10日
    000
  • PHP连接MySQL时HY000/2002错误排查与解决

    本文详细探讨了PHP mysqli_connect() 函数在连接MySQL数据库时常见的 HY000/2002 错误,该错误通常指示连接超时或主机无响应。文章提供了系统化的排查步骤,包括优先使用 localhost 进行本地连接、实现健壮的错误处理机制、检查文件部署路径、验证MySQL服务状态及网…

    2025年12月10日
    000
  • 解决PHP中MySQL连接错误:无法连接到MySQL服务器

    本文旨在解决PHP应用中常见的“无法连接到MySQL服务器”错误,特别是当使用XAMPP环境时遇到的mysqli_connect(): (HY000/2002)连接失败问题。我们将深入探讨导致此类错误的核心原因,如主机地址配置不当、MySQL服务状态异常、文件放置位置错误等,并提供详细的解决方案、示…

    2025年12月10日
    000
  • 解决PHP MySQL连接错误:HY000/2002 故障排除与最佳实践

    本教程旨在解决PHP应用中常见的MySQL连接错误,特别是“HY000/2002: A connection attempt failed”问题。文章将深入探讨导致连接失败的常见原因,如主机地址配置不当、MySQL服务未运行以及文件部署位置错误,并提供详细的排查步骤、标准化的连接代码示例及错误处理机…

    2025年12月10日
    000

发表回复

登录后才能评论
关注微信