第73题JAVA高级技术-多线程7(模拟银行系统存钱2)

Posted 小虚竹

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了第73题JAVA高级技术-多线程7(模拟银行系统存钱2)相关的知识,希望对你有一定的参考价值。

回城传送–》《JAVA筑基100例》

文章目录

零、前言

​ 今天是学习 JAVA语言 打卡的第73天,每天我会提供一篇文章供群成员阅读( 不需要订阅付钱 ),读完文章之后,按解题思路,自己再实现一遍。在小虚竹JAVA社区 中对应的 【打卡贴】打卡,今天的任务就算完成了。

​ 因为大家都在一起学习同一篇文章,所以有什么问题都可以在群里问,群里的小伙伴可以迅速地帮到你,一个人可以走得很快,一群人可以走得很远,有一起学习交流的战友,是多么幸运的事情。

​ 学完后,自己写篇学习报告的博客,可以发布到小虚竹JAVA社区 ,供学弟学妹们参考。

​ 我的学习策略很简单,题海策略+ 费曼学习法。如果能把这100题都认认真真自己实现一遍,那意味着 JAVA语言 已经筑基成功了。后面的进阶学习,可以继续跟着我,一起走向架构师之路。

一、题目描述

题目:模拟一个简单的银行系统,使用两个不同的线程向同一个账户存钱。

实现:使用特殊域变量volatile实现同步。

二、解题思路

创建一个类:SynchronizedBankFrame,继承JFrame类

写一个内部类Bank

  • 定义一个account变量,来表示账户。

  • deposit():一个存钱的方法

  • getAccount():显示账户余额的方法。

写一个内部类Transfer,实现Runnable接口

在run方法中实现向账户存钱的功能。

volatile关键字为变量访问提供了一种免锁机制。使用volatile关键字修饰变量,每次使用该变量就要重新计算,而不是使用寄存器中的值。

volatile不会提供原子操作,也不能用来修饰final类型的变量。

三、代码详解

SynchronizedBankFrame

package com.xiaoxuzhu;
import java.awt.BorderLayout;
import java.awt.EventQueue;

import javax.swing.JFrame;
import javax.swing.JPanel;
import javax.swing.border.EmptyBorder;
import javax.swing.JButton;
import javax.swing.JScrollPane;
import javax.swing.JTextArea;
import java.awt.event.ActionListener;
import java.awt.event.ActionEvent;
import java.awt.GridLayout;
import javax.swing.JLabel;
import javax.swing.SwingConstants;
import java.awt.Font;
import javax.swing.UIManager;
/**
 * Description:
 *
 * @author xiaoxuzhu
 * @version 1.0
 *
 * <pre>
 * 修改记录:
 * 修改后版本	        修改人		修改日期			修改内容
 * 2022/5/14.1	    xiaoxuzhu		2022/5/14		    Create
 * </pre>
 * @date 2022/5/14
 */
public class SynchronizedBankFrame extends JFrame 

    /**
     *
     */
    private static final long serialVersionUID = 2671056183299397274L;
    private JPanel contentPane;
    private JTextArea thread1TextArea;
    private JTextArea thread2TextArea;

    /**
     * Launch the application.
     */
    public static void main(String[] args) 
        try 
            UIManager.setLookAndFeel("com.sun.java.swing.plaf.nimbus.NimbusLookAndFeel");
         catch (Throwable e) 
            e.printStackTrace();
        
        EventQueue.invokeLater(new Runnable() 
            public void run() 
                try 
                    SynchronizedBankFrame frame = new SynchronizedBankFrame();
                    frame.setVisible(true);
                 catch (Exception e) 
                    e.printStackTrace();
                
            
        );
    

    /**
     * Create the frame.
     */
    public SynchronizedBankFrame() 
        setTitle("使用volatile实现线程同步");
        setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
        setBounds(100, 100, 450, 300);
        contentPane = new JPanel();
        contentPane.setBorder(new EmptyBorder(5, 5, 5, 5));
        setContentPane(contentPane);
        contentPane.setLayout(new BorderLayout(0, 0));

        JPanel buttonPanel = new JPanel();
        contentPane.add(buttonPanel, BorderLayout.SOUTH);

        JButton startButton = new JButton("开始存钱");
        startButton.setFont(new Font("微软雅黑", Font.PLAIN, 16));
        startButton.addActionListener(new ActionListener() 
            public void actionPerformed(ActionEvent arg0) 
                do_button_actionPerformed(arg0);
            
        );
        buttonPanel.add(startButton);

        JPanel processPanel = new JPanel();
        contentPane.add(processPanel, BorderLayout.CENTER);
        processPanel.setLayout(new GridLayout(1, 2, 5, 5));

        JPanel thread1Panel = new JPanel();
        processPanel.add(thread1Panel);
        thread1Panel.setLayout(new BorderLayout(0, 0));

        JLabel thread1Label = new JLabel("一号线程");
        thread1Label.setFont(new Font("微软雅黑", Font.PLAIN, 16));
        thread1Label.setHorizontalAlignment(SwingConstants.CENTER);
        thread1Panel.add(thread1Label, BorderLayout.NORTH);

        JScrollPane thread1ScrollPane = new JScrollPane();
        thread1Panel.add(thread1ScrollPane, BorderLayout.CENTER);

        thread1TextArea = new JTextArea();
        thread1TextArea.setFont(new Font("微软雅黑", Font.PLAIN, 16));
        thread1ScrollPane.setViewportView(thread1TextArea);

        JPanel thread2Panel = new JPanel();
        processPanel.add(thread2Panel);
        thread2Panel.setLayout(new BorderLayout(0, 0));

        JLabel thread2Label = new JLabel("二号线程");
        thread2Label.setFont(new Font("微软雅黑", Font.PLAIN, 16));
        thread2Label.setHorizontalAlignment(SwingConstants.CENTER);
        thread2Panel.add(thread2Label, BorderLayout.NORTH);

        JScrollPane thread2ScrollPane = new JScrollPane();
        thread2Panel.add(thread2ScrollPane, BorderLayout.CENTER);

        thread2TextArea = new JTextArea();
        thread2TextArea.setFont(new Font("微软雅黑", Font.PLAIN, 16));
        thread2ScrollPane.setViewportView(thread2TextArea);
    

    protected void do_button_actionPerformed(ActionEvent arg0) 
        Bank bank = new Bank();
        Thread thread1 = new Thread(new Transfer(bank, thread1TextArea));
        thread1.start();
        Thread thread2 = new Thread(new Transfer(bank, thread2TextArea));
        thread2.start();
    

    private class Transfer implements Runnable 

        private Bank bank;
        private JTextArea textArea;

        public Transfer(Bank bank, JTextArea textArea) 
            this.bank = bank;
            this.textArea = textArea;
        

        public void run() 
            for (int i = 0; i < 10; i++) 
                bank.deposit(10);
                String text = textArea.getText();
                textArea.setText(text + "账户的余额是:" + bank.getAccount() + "\\n");
            
        
    

    private class Bank 
        private volatile int account = 100;// 将域变量用volatile修饰

        public void deposit(int money) // 向账户中存钱
            account += money;
        

        public int getAccount() // 获得账户余额
            return account;
        
    




多学一个知识点

每个线程是存在缓存内存的。且缓存内存是对其他线程不可见的。这就是内存不可见问题。

来验证下

package com.xiaoxuzhu;

/**
 * Description: 证明线程是存在缓存内存的
 *
 * @author xiaoxuzhu
 * @version 1.0
 *
 * <pre>
 * 修改记录:
 * 修改后版本	        修改人		修改日期			修改内容
 * 2022/5/14.1	    xiaoxuzhu		2022/5/14		    Create
 * </pre>
 * @date 2022/5/14
 */
public class VolatileDemo 


    public static void main(String[] args) 
        ThreadDemo threadDemo = new ThreadDemo();
        //启动线程  改值
        new Thread(threadDemo).start();

        while (true)
            if(threadDemo.isFlag())
                System.out.println("主线程读到的flag是true");
                break;
            
        
    

    static class  ThreadDemo implements Runnable
        private boolean flag = false;

        public boolean isFlag() 
            return flag;
        

        public void setFlag(boolean flag) 
            this.flag = flag;
        

        @Override
        public void run() 
            try 
                Thread.sleep(200);
             catch (InterruptedException e) 
                e.printStackTrace();
            
            flag = true;
            System.out.println("flag 线程执行改为:"+isFlag());
        
    


主线程得到的是false,新线程已经把值修改为true了。证实每个线程是存在缓存内存的

四、推荐专栏

《JAVA从零到壹》

《JAVA筑基100例》

五、示例源码下载

关注下面的公众号,回复筑基+题目号

筑基73

以上是关于第73题JAVA高级技术-多线程7(模拟银行系统存钱2)的主要内容,如果未能解决你的问题,请参考以下文章

第73题JAVA高级技术-多线程7(模拟银行系统存钱2)

第72题JAVA高级技术-多线程6(模拟银行系统存钱)

第72题JAVA高级技术-多线程6(模拟银行系统存钱)

用java编写多线程银行ATM 模拟程序

java并发编程--一道经典多线程题的2种解法

线程同步代码块:两个客户往一个银行存钱,每人存三十次一次存一百。 模拟银行存钱功能,时时银行现金数。