Java多線程模擬銀行系統(tǒng)存錢問題詳解
一、題目描述
題目:模擬一個簡單的銀行系統(tǒng),使用兩個不同的線程向同一個賬戶存錢。
實現:使用特殊域變量volatile實現同步。
二、解題思路
創(chuàng)建一個類: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 線程執(zhí)行改為:"+isFlag()); } } }
主線程得到的是false,新線程已經把值修改為true了。證實每個線程是存在緩存內存的
到此這篇關于Java多線程模擬銀行系統(tǒng)存錢問題詳解的文章就介紹到這了,更多相關Java多線程銀行存錢內容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關文章希望大家以后多多支持腳本之家!
相關文章
解決SpringBoot webSocket 資源無法加載、tomcat啟動報錯的問題
這篇文章主要介紹了解決SpringBoot webSocket 資源無法加載、tomcat啟動報錯的問題,本文給大家介紹的非常詳細,對大家的學習或工作具有一定的參考借鑒價值,需要的朋友可以參考下2020-11-11SpringBoot+SpringSecurity實現基于真實數據的授權認證
Spring Security是一個功能強大且高度可定制的身份驗證和訪問控制框架,Spring Security主要做兩個事情,認證、授權。這篇文章主要介紹了SpringBoot+SpringSecurity實現基于真實數據的授權認證,需要的朋友可以參考下2021-05-05