黑马程序员 银行业务调度系统

2024-06-11 00:18

本文主要是介绍黑马程序员 银行业务调度系统,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

---------------------- ASP.Net+Android+IO开发S、.Net培训、期待与您交流! ----------------------

一、项目需求

模拟实现银行业务调度系统逻辑,具体需求如下:
银行内有6个业务窗口,1 - 4号窗口为普通窗口,5号窗口为快速窗口,6号窗口为VIP窗口。
有三种对应类型的客户:VIP客户,普通客户,快速客户(办理如交水电费、电话费之类业务的客户)。
异步随机生成各种类型的客户,生成各类型用户的概率比例为:
        VIP客户 :普通客户 :快速客户  =  1 :6 :3。

客户办理业务所需时间有最大值和最小值,在该范围内随机设定每个VIP客户以及普通客户办理业务所需的时间,快速客户办理业务所需时间为最小值(提示:办理业务的过程可通过线程Sleep的方式模拟)。
各类型客户在其对应窗口按顺序依次办理业务。 
当VIP(6号)窗口和快速业务(5号)窗口没有客户等待办理业务的时候,这两个窗口可以处理普通客户的业务,而一旦有对应的客户等待办理业务的时候,则优先处理对应客户的业务。
随机生成客户时间间隔以及业务办理时间最大值和最小值自定,可以设置。
不要求实现GUI,只考虑系统逻辑实现,可通过Log方式展现程序运行结果。

二、面向对象的分析与设计

1、有三种对应类型的客户:VIP客户,普通客户,快速客户 ,异步随机生成各种类型的客户,各类型客户在其对应窗口按顺序依次办理业务 。

首先,经常在银行办理业务的人更有利于理解本系统,例如,我经常陪老婆跑银行,对银行的这个业务算是比较熟悉了,我知道每一个客户其实就是由银行的一个取号机器产生号码的方式来表示的。所以,我想到要有一个号码管理器对象,让这个对象不断地产生号码,就等于随机生成了客户。
由于有三类客户,每类客户的号码编排都是完全独立的,所以,我想到本系统一共要产生三个号码管理器对象,各自管理一类用户的排队号码。这三个号码管理器对象统一由一个号码机器进行管理,这个号码机器在整个系统中始终只能有一个,所以,它要被设计成单例。

2、各类型客户在其对应窗口按顺序依次办理业务 ,准确地说,应该是窗口依次叫号。

各个窗口怎么知道该叫哪一个号了呢?它一定是问的相应的号码管理器,即服务窗口每次找号码管理器获取当前要被服务的号码。
如果我不是多次亲身经历银行的这种业务,再加之积累了大量面向对象的应用开发经验,我也不知道能否轻松进行这种设计,能否发掘出其中隐含的对象信息,我真说不出具体的经验是什么,就是日积月累出来的一种感觉。难道这就是传说中的:“只可意会,不可言传?”

3、类之间的关系图

 

三、类的设计

1、NumberManager类

定义一个用于存储上一个客户号码的成员变量和用于存储所有等待服务的客户号码的队列集合。
定义一个产生新号码的方法和获取马上要为之服务的号码的方法,这两个方法被不同的线程操作了相同的数据,所以,要进行同步。

代码

[java] view plaincopyprint?
package bankqueue;  import java.util.ArrayList;  
import java.util.List;  public class NumberManager {  private int lastNumber = 0;  private List queueNumbers = new ArrayList();  public synchronized Integer generateNewNumber(){  queueNumbers.add(++lastNumber);  return lastNumber;  }  public synchronized Integer fetchNumber(){  if(queueNumbers.size()>0){  return (Integer)queueNumbers.remove(0);  }else{  return null;  }  }  
}  



2、NumberMachine类

定义三个成员变量分别指向三个NumberManager对象,分别表示普通、快速和VIP客户的号码管理器,定义三个对应的方法来返回这三个NumberManager对象。
将NumberMachine类设计成单例。

代码

[java] view plaincopyprint?
package bankqueue;  public class NumberMachine {  private NumberMachine(){}  private static NumberMachine instance = new NumberMachine();  public static NumberMachine getInstance(){  return instance;  }  private NumberManager commonManager = new NumberManager();  private NumberManager expressManager = new NumberManager();  private NumberManager vipManager = new NumberManager();  public NumberManager getCommonManager() {  return commonManager;  }  public NumberManager getExpressManager() {  return expressManager;  }  public NumberManager getVipManager() {  return vipManager;  }  
}  


3、CustomerType枚举类

系统中有三种类型的客户,所以用定义一个枚举类,其中定义三个成员分别表示三种类型的客户。
重写toString方法,返回类型的中文名称。这是在后面编码时重构出来的,刚开始不用考虑。

代码

[java]  view plain copy print ?
  1. package bankqueue;  
  2.   
  3. public enum CustomerType {  
  4.     COMMON,EXPRESS,VIP;  
  5.     public String toString(){  
  6.         String name = null;  
  7.         switch(this){  
  8.         case COMMON:  
  9.             name = "普通";  
  10.             break;  
  11.         case EXPRESS:  
  12.             name = "快速";  
  13.             break;  
  14.         case VIP:  
  15.             name = name();  
  16.             break;  
  17.         }  
  18.         return name;  
  19.     }  
  20. }  


4、ServiceWindow枚举类

定义一个start方法,内部启动一个线程,根据服务窗口的类别分别循环调用三个不同的方法。 
定义三个方法分别对三种客户进行服务,为了观察运行效果,应详细打印出其中的细节信息。

代码

[java]  view plain copy print ?
  1. package bankqueue;  
  2.   
  3. import java.util.Random;  
  4. import java.util.concurrent.Executors;  
  5. import java.util.logging.Logger;  
  6.   
  7. /** 
  8.  * 没有把VIP窗口和快速窗口做成子类,是因为实际业务中的普通窗口可以随时被设置为VIP窗口和快速窗口。 
  9.  *  
  10.  */  
  11. public class ServiceWindow {  
  12.     private static Logger logger = Logger.getLogger("cn.itcast.bankqueue");  
  13.     private CustomerType type = CustomerType.COMMON;  
  14.     private int number = 1;  
  15.   
  16.     public CustomerType getType() {  
  17.         return type;  
  18.     }  
  19.   
  20.     public void setType(CustomerType type) {  
  21.         this.type = type;  
  22.     }  
  23.       
  24.     public void setNumber(int number){  
  25.         this.number = number;  
  26.     }  
  27.       
  28.     public void start(){Executors.newSingleThreadExecutor().execute(  
  29.         new Runnable(){  
  30.             public void run(){  
  31.                 //下面这种写法的运行效率低,最好是把while放在case下面  
  32.                 while(true){  
  33.                     switch(type){  
  34.                         case COMMON:  
  35.                             commonService();  
  36.                             break;  
  37.                         case EXPRESS:  
  38.                             expressService();  
  39.                             break;  
  40.                         case VIP:  
  41.                             vipService();  
  42.                             break;  
  43.                     }  
  44.                 }  
  45.             }  
  46.         });  
  47.     }  
  48.       
  49.     private void commonService(){  
  50.         String windowName = "第" + number + "号" + type + "窗口";         
  51.         System.out.println(windowName + "开始获取普通任务!");  
  52.         Integer serviceNumber = NumberMachine.getInstance().getCommonManager().fetchNumber();         
  53.         if(serviceNumber != null ){  
  54.             System.out.println(windowName + "开始为第" + serviceNumber + "号普通客户服务");          
  55.             int maxRandom = Constants.MAX_SERVICE_TIME - Constants.MIN_SERVICE_TIME;  
  56.             int serviceTime = new Random().nextInt(maxRandom)+1 + Constants.MIN_SERVICE_TIME;  
  57.       
  58.             try {  
  59.                 Thread.sleep(serviceTime);  
  60.             } catch (InterruptedException e) {  
  61.                 e.printStackTrace();  
  62.             }     
  63.             System.out.println(windowName + "完成为第" + serviceNumber + "号普通客户服务,总共耗时" + serviceTime/1000 + "秒");        
  64.         }else{  
  65.             System.out.println(windowName + "没有取到普通任务,正在空闲一秒");       
  66.             try {  
  67.                 Thread.sleep(1000);  
  68.             } catch (InterruptedException e) {  
  69.                 e.printStackTrace();  
  70.             }                 
  71.         }  
  72.     }  
  73.       
  74.     private void expressService(){  
  75.         Integer serviceNumber = NumberMachine.getInstance().getExpressManager().fetchNumber();  
  76.         String windowName = "第" + number + "号" + type + "窗口";     
  77.         System.out.println(windowName + "开始获取快速任务!");         
  78.         if(serviceNumber !=null){  
  79.             System.out.println(windowName + "开始为第" + serviceNumber + "号快速客户服务");              
  80.             int serviceTime = Constants.MIN_SERVICE_TIME;  
  81.             try {  
  82.                 Thread.sleep(serviceTime);  
  83.             } catch (InterruptedException e) {  
  84.                 e.printStackTrace();  
  85.             }         
  86.             System.out.println(windowName + "完成为第" + serviceNumber + "号快速客户服务,总共耗时" + serviceTime/1000 + "秒");        
  87.         }else{  
  88.             System.out.println(windowName + "没有取到快速任务!");                 
  89.             commonService();  
  90.         }  
  91.     }  
  92.       
  93.     private void vipService(){  
  94.   
  95.         Integer serviceNumber = NumberMachine.getInstance().getVipManager().fetchNumber();  
  96.         String windowName = "第" + number + "号" + type + "窗口";     
  97.         System.out.println(windowName + "开始获取VIP任务!");            
  98.         if(serviceNumber !=null){  
  99.             System.out.println(windowName + "开始为第" + serviceNumber + "号VIP客户服务");             
  100.             int maxRandom = Constants.MAX_SERVICE_TIME - Constants.MIN_SERVICE_TIME;  
  101.             int serviceTime = new Random().nextInt(maxRandom)+1 + Constants.MIN_SERVICE_TIME;  
  102.             try {  
  103.                 Thread.sleep(serviceTime);  
  104.             } catch (InterruptedException e) {  
  105.                 e.printStackTrace();  
  106.             }         
  107.             System.out.println(windowName + "完成为第" + serviceNumber + "号VIP客户服务,总共耗时" + serviceTime/1000 + "秒");       
  108.         }else{  
  109.             System.out.println(windowName + "没有取到VIP任务!");                
  110.             commonService();  
  111.         }     
  112.     }  
  113. }  


5、MainClass类

用for循环创建出4个普通窗口,再创建出1个快速窗口和一个VIP窗口。
接着再创建三个定时器,分别定时去创建新的普通客户号码、新的快速客户号码、新的VIP客户号码。

代码

[java]  view plain copy print ?
  1. package bankqueue;  
  2.   
  3. import java.util.concurrent.Executors;  
  4. import java.util.concurrent.TimeUnit;  
  5. import java.util.logging.Logger;  
  6.   
  7. class MainClass {  
  8.       
  9.     private static Logger logger = Logger.getLogger("cn.itcast.bankqueue");  
  10.       
  11.   
  12.     public static void main(String[] args) {  
  13.         //产生4个普通窗口   
  14.         for(int i=1;i<5;i++){  
  15.             ServiceWindow window =  new ServiceWindow();  
  16.             window.setNumber(i);  
  17.             window.start();  
  18.         }  
  19.       
  20.         //产生1个快速窗口   
  21.         ServiceWindow expressWindow =  new ServiceWindow();  
  22.         expressWindow.setType(CustomerType.EXPRESS);  
  23.         expressWindow.start();  
  24.           
  25.         //产生1个VIP窗口        
  26.         ServiceWindow vipWindow =  new ServiceWindow();  
  27.         vipWindow.setType(CustomerType.VIP);  
  28.         vipWindow.start();        
  29.           
  30.         //普通客户拿号   
  31.         Executors.newScheduledThreadPool(1).scheduleAtFixedRate(  
  32.                 new Runnable(){  
  33.                     public void run(){  
  34.                         Integer serviceNumber = NumberMachine.getInstance().getCommonManager().generateNewNumber();  
  35.                         /** 
  36.                          * 采用logger方式,无法看到直观的运行效果,因为logger.log方法内部并不是直接把内容打印出出来, 
  37.                          * 而是交给内部的一个线程去处理,所以,打印出来的结果在时间顺序上看起来很混乱。 
  38.                          */  
  39.                         //logger.info("第" + serviceNumber + "号普通客户正在等待服务!");  
  40.                         System.out.println("第" + serviceNumber + "号普通客户正在等待服务!");                         
  41.                     }  
  42.                 },  
  43.                 0,  
  44.                 Constants.COMMON_CUSTOMER_INTERVAL_TIME,   
  45.                 TimeUnit.SECONDS);  
  46.           
  47.         //快速客户拿号   
  48.         Executors.newScheduledThreadPool(1).scheduleAtFixedRate(  
  49.                 new Runnable(){  
  50.                     public void run(){  
  51.                         Integer serviceNumber = NumberMachine.getInstance().getExpressManager().generateNewNumber();  
  52.                         System.out.println("第" + serviceNumber + "号快速客户正在等待服务!");  
  53.                     }  
  54.                 },  
  55.                 0,  
  56.                 Constants.COMMON_CUSTOMER_INTERVAL_TIME * 2,   
  57.                 TimeUnit.SECONDS);  
  58.           
  59.         //VIP客户拿号   
  60.         Executors.newScheduledThreadPool(1).scheduleAtFixedRate(  
  61.                 new Runnable(){  
  62.                     public void run(){  
  63.                         Integer serviceNumber = NumberMachine.getInstance().getVipManager().generateNewNumber();  
  64.                         System.out.println("第" + serviceNumber + "号VIP客户正在等待服务!");  
  65.                     }  
  66.                 },  
  67.                 0,  
  68.                 Constants.COMMON_CUSTOMER_INTERVAL_TIME * 6,   
  69.                 TimeUnit.SECONDS);  
  70.     }  
  71.   
  72. }  


6、Constants类

定义三个常量:
MAX_SERVICE_TIME
MIN_SERVICE_TIME
COMMON_CUSTOMER_INTERVAL_TIME

代码

[java]  view plain copy print ?
  1. package bankqueue;  
  2.   
  3. public class Constants {  
  4.     public static int MAX_SERVICE_TIME = 10000//10秒!  
  5.     public static int MIN_SERVICE_TIME = 1000//1秒!  
  6.       
  7.     /*每个普通窗口服务一个客户的平均时间为5秒,一共有4个这样的窗口,也就是说银行的所有普通窗口合起来 
  8.      * 平均1.25秒内可以服务完一个普通客户,再加上快速窗口和VIP窗口也可以服务普通客户,所以, 
  9.      * 1秒钟产生一个普通客户比较合理,*/     
  10.     public static int COMMON_CUSTOMER_INTERVAL_TIME = 1;      
  11. }  
  12. ---------------------- ASP.Net+Android+IOS开发、.Net培训、期待与您交流! ----------------------

这篇关于黑马程序员 银行业务调度系统的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1049672

相关文章

Mac系统下卸载JAVA和JDK的步骤

《Mac系统下卸载JAVA和JDK的步骤》JDK是Java语言的软件开发工具包,它提供了开发和运行Java应用程序所需的工具、库和资源,:本文主要介绍Mac系统下卸载JAVA和JDK的相关资料,需... 目录1. 卸载系统自带的 Java 版本检查当前 Java 版本通过命令卸载系统 Java2. 卸载自定

基于Python实现一个简单的题库与在线考试系统

《基于Python实现一个简单的题库与在线考试系统》在当今信息化教育时代,在线学习与考试系统已成为教育技术领域的重要组成部分,本文就来介绍一下如何使用Python和PyQt5框架开发一个名为白泽题库系... 目录概述功能特点界面展示系统架构设计类结构图Excel题库填写格式模板题库题目填写格式表核心数据结构

Linux系统中的firewall-offline-cmd详解(收藏版)

《Linux系统中的firewall-offline-cmd详解(收藏版)》firewall-offline-cmd是firewalld的一个命令行工具,专门设计用于在没有运行firewalld服务的... 目录主要用途基本语法选项1. 状态管理2. 区域管理3. 服务管理4. 端口管理5. ICMP 阻断

Windows 系统下 Nginx 的配置步骤详解

《Windows系统下Nginx的配置步骤详解》Nginx是一款功能强大的软件,在互联网领域有广泛应用,简单来说,它就像一个聪明的交通指挥员,能让网站运行得更高效、更稳定,:本文主要介绍W... 目录一、为什么要用 Nginx二、Windows 系统下 Nginx 的配置步骤1. 下载 Nginx2. 解压

如何确定哪些软件是Mac系统自带的? Mac系统内置应用查看技巧

《如何确定哪些软件是Mac系统自带的?Mac系统内置应用查看技巧》如何确定哪些软件是Mac系统自带的?mac系统中有很多自带的应用,想要看看哪些是系统自带,该怎么查看呢?下面我们就来看看Mac系统内... 在MAC电脑上,可以使用以下方法来确定哪些软件是系统自带的:1.应用程序文件夹打开应用程序文件夹

windows系统上如何进行maven安装和配置方式

《windows系统上如何进行maven安装和配置方式》:本文主要介绍windows系统上如何进行maven安装和配置方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不... 目录1. Maven 简介2. maven的下载与安装2.1 下载 Maven2.2 Maven安装2.

使用Python实现Windows系统垃圾清理

《使用Python实现Windows系统垃圾清理》Windows自带的磁盘清理工具功能有限,无法深度清理各类垃圾文件,所以本文为大家介绍了如何使用Python+PyQt5开发一个Windows系统垃圾... 目录一、开发背景与工具概述1.1 为什么需要专业清理工具1.2 工具设计理念二、工具核心功能解析2.

Linux系统之stress-ng测压工具的使用

《Linux系统之stress-ng测压工具的使用》:本文主要介绍Linux系统之stress-ng测压工具的使用,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录一、理论1.stress工具简介与安装2.语法及参数3.具体安装二、实验1.运行8 cpu, 4 fo

ubuntu20.0.4系统中安装Anaconda的超详细图文教程

《ubuntu20.0.4系统中安装Anaconda的超详细图文教程》:本文主要介绍了在Ubuntu系统中如何下载和安装Anaconda,提供了两种方法,详细内容请阅读本文,希望能对你有所帮助... 本文介绍了在Ubuntu系统中如何下载和安装Anaconda。提供了两种方法,包括通过网页手动下载和使用wg

ubuntu系统使用官方操作命令升级Dify指南

《ubuntu系统使用官方操作命令升级Dify指南》Dify支持自动化执行、日志记录和结果管理,适用于数据处理、模型训练和部署等场景,今天我们就来看看ubuntu系统中使用官方操作命令升级Dify的方... Dify 是一个基于 docker 的工作流管理工具,旨在简化机器学习和数据科学领域的多步骤工作流。