黑马程序员-----高薪面试题-银行业务调度系统

来源:互联网 发布:欧舒丹护手霜味道知乎 编辑:程序博客网 时间:2024/05/01 13:22

------<a href="http://www.itheima.com" target="blank">Java培训、Android培训、iOS培训、.Net培训</a>、期待与您交流! -------


一、需求

模拟实现银行业务调度系统逻辑,具体需求如下:

1、银行内有6个业务窗口,1 - 4号窗口为普通窗口,5号窗口为快速窗口,6号窗口为VIP窗口。

2、有三种对应类型的客户:VIP客户,普通客户,快速客户(办理如交水电费、电话费之类业务的客户)。

3、异步随机生成各种类型的客户,生成各类型用户的概率比例为:

        VIP客户 :普通客户 :快速客户  =  1 :6 :3。

4、客户办理业务所需时间有最大值和最小值,在该范围内随机设定每个VIP客户以及普通客户办理业务所需的时间,快速客户办理业务所需时间为最小值(提示:办理业务的过程可通过线程Sleep的方式模拟)。

5、各类型客户在其对应窗口按顺序依次办理业务。

6、 当VIP(6号)窗口和快速业务(5号)窗口没有客户等待办理业务的时候,这两个窗口可以处理普通客户的业务,而一旦有对应的客户等待办理业务的时候,则优先处理对应客户的业务。

7、随机生成客户时间间隔以及业务办理时间最大值和最小值自定,可以设置。

8、不要求实现GUI,只考虑系统逻辑实现,可通过Log方式展现程序运行结果

 二、思路 

系统分析(抽取对象和对象的功能): 

 1.从整个业务逻辑来说,整个系统分为 3 种对象,银行窗口、银行客户、银行自动排队机 

  2.整个业务用上面的对象来描述就是客户先到自动排队机按照自己的业务类别取号,拿到号后等着窗口呼叫 

 3.自动排队机可以给客户依顺序分配一个号码,然后把该客户添加到相应的业务种类的队列里,自动排队机还可 以给银行业务窗口分配对应类型的客户 

 4.银行客户窗口主要的功能就是从自动排队机里面获取一个客户,处理客户的业务,再从自动排队机里面获取一  个客户 

 步骤: 

 1.定义上述各个对象的接口和接口中的功能 

2.用具体的类实现这些接口 

 3.在 main 函数中使用这些对象实现业务 

1.创建 ACDSyste 对象 

2.创建窗口对象,但是 6 个窗口是异步工作的,所以需要6个线程来描述,所以窗口需要实现 Runnable 接口, 

因为每个窗口都要使用 ACDSystem 来呼叫客户,所以在创建窗口对象的时候需要传递一个 ACDSystem 对象。 

3.创建银行客户,并调用客户到自动排队机排队的方法 

 4.完善 main 函数中涉及到的类,和接口,这一步只是让他们语法上都对,实现 main 函数中用到的函数 

 5.从上至下完善 main 函数用到的类和类的函数,不对各个类中的内部函数实现 

  6.实现各个类中的内部函数 

 

总结: 

1.先从需求建立需求模型——分析需求 

2.再从需求模型建立系统模型——建立对象模型 

3.再把系统模型用代码实现——实现对象模型 

 4.根据变化重构——重构

三、代码实现

号码管理:

[java] view plaincopy
  1. <span style="font-size:14px;">package cn.itcast.interview2;  
  2. import java.util.ArrayList;  
  3. import java.util.List;  
  4.   
  5. public class NumberManager {  
  6.     //定义私有成员变量lastNumber,表示队列中最后一个数  
  7.     private int lastNumber = 0;  
  8.     //定义list集合queueNumber用于存放号码  
  9.     public List<Integer> queueNumber= new ArrayList<Integer>();  
  10.       
  11.     //定义方法generateNewNumber()用于产生号码  
  12.     //由于产生号码和获取号码必须保证同步,因此加入同步代码块  
  13.     public synchronized Integer generateNewNumber(){  
  14.         queueNumber.add(++lastNumber);  
  15.         return lastNumber;  
  16.     }  
  17.       
  18.     //定义方法fetchNumber()用于获取排列在最初的号码  
  19.     //由于产生号码和获取号码必须保证同步,因此加入同步代码块  
  20.     public synchronized Integer fetchNumber(){  
  21.           
  22.         if (queueNumber.size()>0){  
  23.             return (Integer)queueNumber.remove(0);  
  24.         }else{  
  25.             return null;  
  26.         }  
  27.     }  
  28.   
  29. }</span>  

取号机:
[java] view plaincopy
  1. package cn.itcast.interview2;  
  2.   
  3. public class NumberMechine {  
  4.       
  5.     /*号码机是用于给不同柜台产生号码用的,由于号码机的唯一性,这里考虑使用单例设计模式*/  
  6.       
  7.     //首先,构造函数私有化  
  8.     private NumberMechine(){}  
  9.       
  10.     //其次,创建本类对象  
  11.     private static NumberMechine instance = new NumberMechine();  
  12.       
  13.     //最后,提供获得实例的方法  
  14.     public static NumberMechine getInstance(){  
  15.         return instance;  
  16.     }  
  17.       
  18.     //定义三个窗口的对象,并提供get方法  
  19.     private NumberManager commonManager = new NumberManager();  
  20.     private NumberManager expressManager = new NumberManager();  
  21.     private NumberManager vipManager = new NumberManager();  
  22.       
  23.     public NumberManager getCommonManager() {  
  24.         return commonManager;  
  25.     }  
  26.     public NumberManager getExpressManager() {  
  27.         return expressManager;  
  28.     }  
  29.     public NumberManager getVipManager() {  
  30.         return vipManager;  
  31.     }  
  32.   
  33. }  

客户类型类:

[java] view plaincopy
  1. package cn.itcast.interview2;  
  2.   
  3. public enum CustomerType {  
  4.     /*三种顾客类型分别是,普通客户,快速客户,VIP客户*/  
  5.     COMMON,  
  6.     EXPRESS,  
  7.     VIP;  
  8.       
  9.     public String toString(){  
  10.         String name = null;  
  11.         switch(this){  
  12.           
  13.             case COMMON:   
  14.                 name = "普通";  
  15.                 break;  
  16.             case EXPRESS:  
  17.                 name = "快速";  
  18.                 break;  
  19.             case VIP:   
  20.                 name = "VIP";  
  21.                 break;  
  22.               
  23.         }  
  24.         return name;          
  25.           
  26.     }  
  27.   
  28. }  

常量时间定义:

[java] view plaincopy
  1. package cn.itcast.interview2;  
  2.   
  3.   
  4. public class Constants {  
  5.     public static int MAX_SERVICE_TIME = 10000//10秒!  
  6.     public static int MIN_SERVICE_TIME = 1000//1秒!  
  7.       
  8.     /*每个普通窗口服务一个客户的平均时间为5秒,一共有4个这样的窗口,也就是说银行的所有普通窗口合起来 
  9.      * 平均1.25秒内可以服务完一个普通客户,再加上快速窗口和VIP窗口也可以服务普通客户,所以, 
  10.      * 1秒钟产生一个普通客户比较合理,*/     
  11.     public static int COMMON_CUSTOMER_INTERVAL_TIME = 1;      
  12. }  

服务窗口:对象理解:一个窗口就是一个对象,所以需要6个这样的对象。
    但有三类窗口,即里面要判断三种创建对象的方式。
    窗口本身的数据:窗口类型及几号,可变客户号码【窗口的叫号方法(关联取号机器)】,服务时间,等待时间。

[java] view plaincopy
  1. package cn.itcast.interview2;  
  2.   
  3. import java.util.Random;  
  4. import java.util.concurrent.Executors;  
  5. import java.util.logging.Logger;  
  6.   
  7. public class ServiceWindow {  
  8.     private static Logger logger = Logger.getLogger("cn.itcast.interview2");  
  9.     private CustomerType type = CustomerType.COMMON;  
  10.     private int number = 1;  
  11.       
  12.     public void setType(CustomerType type){  
  13.         this.type = type;  
  14.     }  
  15.   
  16.     public CustomerType getType() {  
  17.         return type;  
  18.     }  
  19.   
  20.     public void setNumber(int number) {  
  21.         this.number = number;  
  22.     }  
  23.       
  24.     public void start(){  
  25.         Executors.newSingleThreadExecutor().execute(  
  26.                 new Runnable(){  
  27.                     public void run(){  
  28.                         while(true){  
  29.                             switch(type){  
  30.                                 case COMMON:  
  31.                                     commonServer();  
  32.                                     break;  
  33.                                 case EXPRESS:  
  34.                                     expressServer();  
  35.                                     break;  
  36.                                 case VIP:  
  37.                                     vipServer();  
  38.                                     break;  
  39.                             }  
  40.                         }  
  41.                     }  
  42.                 });  
  43.     }  
  44.   
  45.   
  46.   
  47.     private void commonServer() {  
  48.         String windowName = "第"+number+"号"+type+"窗口";  
  49.         System.out.println(windowName+"为您服务,开始获取普通任务!");  
  50.         Integer serviceNumber = NumberMechine.getInstance().getCommonManager().fetchNumber();  
  51.         if (serviceNumber!=null){  
  52.             System.out.println(windowName + "开始为第"+serviceNumber+"号客户服务");  
  53.             int maxRandom = Constants.MAX_SERVICE_TIME - Constants.MIN_SERVICE_TIME;  
  54.             int serviceTime = new Random().nextInt(maxRandom) + 1 + Constants.MIN_SERVICE_TIME;  
  55.             try {  
  56.                 Thread.sleep(serviceTime);  
  57.             } catch (InterruptedException e) {  
  58.                 // TODO 自动生成的 catch 块  
  59.                 e.printStackTrace();  
  60.             }  
  61.             System.out.println(windowName + "完成为第" + serviceNumber + "号普通客户服务,总共耗时" + serviceTime/1000 + "秒");    
  62.         }else{  
  63.             System.out.println("没有取到普通任务,一秒后重新取任务!");  
  64.             try {  
  65.                 Thread.sleep(1000);  
  66.             } catch (InterruptedException e) {  
  67.                 // TODO 自动生成的 catch 块  
  68.                 e.printStackTrace();  
  69.             }  
  70.         }  
  71.           
  72.           
  73.     }  
  74.       
  75.     protected void expressServer() {  
  76.         String windowName = "第"+number+"号"+type+"窗口";  
  77.         System.out.println(windowName+"为您服务,开始获取快速任务!");  
  78.         Integer serviceNumber = NumberMechine.getInstance().getExpressManager().fetchNumber();  
  79.         if (serviceNumber!=null){  
  80.             System.out.println(windowName + "开始为第"+serviceNumber+"号客户服务");  
  81.               
  82.             int serviceTime =Constants.MIN_SERVICE_TIME;  
  83.             try {  
  84.                 Thread.sleep(serviceTime);  
  85.             } catch (InterruptedException e) {  
  86.                 // TODO 自动生成的 catch 块  
  87.                 e.printStackTrace();  
  88.             }  
  89.             System.out.println(windowName + "完成为第" + serviceNumber + "号快速客户服务,总共耗时" + serviceTime/1000 + "秒");    
  90.         }else{  
  91.             System.out.println("没有取到快速任务,正在尝试获取普通任务!");  
  92.             commonServer();  
  93.         }  
  94.           
  95.     }  
  96.     protected void vipServer() {  
  97.         String windowName = "第"+number+"号"+type+"窗口";  
  98.         System.out.println(windowName+"为您服务,开始获取VIP任务!");  
  99.         Integer serviceNumber = NumberMechine.getInstance().getVipManager().fetchNumber();  
  100.         if (serviceNumber!=null){  
  101.             System.out.println(windowName + "开始为第"+serviceNumber+"号客户服务");  
  102.             int maxRandom = Constants.MAX_SERVICE_TIME - Constants.MIN_SERVICE_TIME;  
  103.             int serviceTime = new Random().nextInt(maxRandom) + 1 + Constants.MIN_SERVICE_TIME;  
  104.             try {  
  105.                 Thread.sleep(serviceTime);  
  106.             } catch (InterruptedException e) {  
  107.                 // TODO 自动生成的 catch 块  
  108.                 e.printStackTrace();  
  109.             }  
  110.             System.out.println(windowName + "完成为第" + serviceNumber + "号VIP客户服务,总共耗时" + serviceTime/1000 + "秒");   
  111.         }else{  
  112.             System.out.println("没有取到VIP任务,正在尝试获取普通任务!");  
  113.             commonServer();  
  114.         }  
  115.           
  116.     }     
  117.   
  118. }  
main类:创建各类窗口,模拟拿号的比例1:6:3
[java] view plaincopy
  1. package cn.itcast.interview2;  
  2.   
  3. import java.util.concurrent.Executors;  
  4. import java.util.concurrent.TimeUnit;  
  5.   
  6. public class MainClass {  
  7.   
  8.     public static void main(String[] args) {  
  9.         //产生4个普通窗口  
  10.         for (int i = 0; i < 5; i++){  
  11.             ServiceWindow window = new ServiceWindow();  
  12.             window.setNumber(i);  
  13.             window.start();  
  14.         }  
  15.           
  16.         //产生一个快速窗口  
  17.         ServiceWindow expressWindow = new ServiceWindow();  
  18.         expressWindow.setType(CustomerType.EXPRESS);  
  19.         expressWindow.start();  
  20.           
  21.         //产生一个vip窗口  
  22.         ServiceWindow vipWindow = new ServiceWindow();  
  23.         vipWindow.setType(CustomerType.VIP);  
  24.         vipWindow.start();  
  25.           
  26.         //普通客户拿号  
  27.         Executors.newScheduledThreadPool(1).scheduleAtFixedRate(  
  28.                 new Runnable(){  
  29.                     public void run(){  
  30.                         Integer serviceNumber = NumberMechine.getInstance().getCommonManager().generateNewNumber();  
  31.                         System.out.println("第" + serviceNumber + "号普通客户正在等待服务!");  
  32.                     }  
  33.                 },  
  34.                     0,  
  35.                     Constants.COMMON_CUSTOMER_INTERVAL_TIME,  
  36.                     TimeUnit.SECONDS  
  37.                 );  
  38.           
  39.         //快速客户拿号  
  40.         Executors.newScheduledThreadPool(1).scheduleAtFixedRate(  
  41.                 new Runnable(){  
  42.                     public void run(){  
  43.                         Integer serviceNumber = NumberMechine.getInstance().getExpressManager().generateNewNumber();  
  44.                         System.out.println("第" + serviceNumber + "号快速客户正在等待服务!");  
  45.                     }  
  46.                 },  
  47.                     0,  
  48.                     Constants.COMMON_CUSTOMER_INTERVAL_TIME*3,  
  49.                     TimeUnit.SECONDS  
  50.                 );  
  51.   
  52.         //vip客户拿号  
  53.         Executors.newScheduledThreadPool(1).scheduleAtFixedRate(  
  54.                 new Runnable(){  
  55.                     public void run(){  
  56.                         Integer serviceNumber = NumberMechine.getInstance().getVipManager().generateNewNumber();  
  57.                         System.out.println("第" + serviceNumber + "号vip客户正在等待服务!");  
  58.                     }  
  59.                 },  
  60.                     0,  
  61.                     Constants.COMMON_CUSTOMER_INTERVAL_TIME*6,  
  62.                     TimeUnit.SECONDS  
  63.                 );  
  64.           
  65.     }  
  66.   

0 0
原创粉丝点击