您的位置:首页 > 职场人生

黑马程序员_Java高新技术_银行业务调度系统

2013-09-11 01:27 621 查看
-------
android培训、java培训、期待与您交流! ----------

需求分析:

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

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

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

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

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

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

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

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

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

面向对象的分析与设计

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

首先,经常在银行办理业务的人更有利于理解本系统,例如,我经常陪老婆跑银行,对银行的这个业务算是比较熟悉了,我知道每一个客户其实就是由银行的一个取号机器产生号码的方式来表示的。所以,我想到要有一个号码管理器对象,让这个对象不断地产生号码,就等于随机生成了客户。

由于有三类客户,每类客户的号码编排都是完全独立的,所以,我想到本系统一共要产生三个号码管理器对象,各自管理一类用户的排队号码。这三个号码管理器对象统一由一个号码机器进行管理,这个号码机器在整个系统中始终只能有一个,所以,它要被设计成单例。

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

各个窗口怎么知道该叫哪一个号了呢?它一定是问的相应的号码管理器,即服务窗口每次找号码管理器获取当前要被服务的号码。

类图分析



具体实现

NumberManager类和NumberMachine类

NumberManager类

定义一个用于存储上一个客户号码的成员变量和用于存储所有等待服务的客户号码的队列集合。

定义一个产生新号码的方法和获取马上要为之服务的号码的方法,这两个方法被不同的线程操作了相同的数据,所以,要进行同步。

package com.itheima.interview.bank;

import java.util.ArrayList;
import java.util.List;

public class NumberManager {

private int lastNumber=1;
private List<Integer> queueNumber=new ArrayList<Integer>();

public synchronized Integer generateNewManager(){
queueNumber.add(lastNumber);
return lastNumber++;
}

public synchronized Integer fetchServiceNumber(){
Integer number=null;
if(queueNumber.size()>0){
number=queueNumber.remove(0);
}
return number;//返回被取走的号
}
}


NumberMachine类

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

将NumberMachine类设计成单例。

package com.itheima.interview.bank;

public class NumberMachine {

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;
}

private static NumberMachine instance=new NumberMachine();
private NumberMachine(){}
public static NumberMachine getInstance(){
return instance;
}
}


ServiceWindow类和CustomerType枚举类

ServiceWindow类

定义一个start方法,内部启动一个线程,根据服务窗口的类别分别循环调用三个不同的方法。

定义三个方法分别对三种客户进行服务,为了观察运行效果,应详细打印出其中的细节信息。

package com.itheima.interview.bank;

import java.util.Random;
import java.util.concurrent.Executors;

/**
* 没有把VIP窗口和快速窗口做成子类,是因为实际业务中的普通窗口可以随时被设置为VIP窗口和快速窗口。
* */

public class ServiceWindow {

private CustomerType type = CustomerType.COMMON;
private int windowId=1;

public void setType(CustomerType type) {
this.type = type;
}

public void setWindowId(int windowId) {
this.windowId = windowId;
}

public void start(){
Executors.newSingleThreadExecutor().execute(new Runnable(){
public void run(){

while(true){
switch(type){

case COMMON:
commonService();
break;

case EXPRESS:
expressService();
break;

case VIP:
vipService();
break;

}

}

}
});
}

private void commonService() {
String windowName="第"+windowId+"号"+type+"窗口";
Integer number=NumberMachine.getInstance().getCommonManager().fetchServiceNumber();
System.out.println(windowName+"正在获取任务");
if(number!=null){
System.out.println(windowName+"为第"+number+"个"+"普通"+"客户服务");
long beginTime=System.currentTimeMillis();
int maxRand=Constants.MAX_SERVICE_TIME-Constants.MIN_SERVICE_TIME;
long serveTime=new Random().nextInt(maxRand)+1+Constants.MIN_SERVICE_TIME;

try {
Thread.sleep(serveTime);
} catch (InterruptedException e) {
e.printStackTrace();
}

long costTime=System.currentTimeMillis()-beginTime;
System.out.println(windowName+"为第"+number+"个"+"普通"+"客户完成服务,耗时"+costTime/1000+"秒");
}else{
System.out.println(windowName+"没有取到任务,先休息1秒钟");
try {
Thread.sleep(1000);
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}

private void expressService() {
String windowName="第"+windowId+"号"+type+"窗口";
Integer number=NumberMachine.getInstance().getExpressManager().fetchServiceNumber();
System.out.println(windowName+"正在获取任务");
if(number!=null){
System.out.println(windowName+"为第"+number+"个"+type+"客户服务");
long beginTime=System.currentTimeMillis();

try {
Thread.sleep(Constants.MIN_SERVICE_TIME);
} catch (InterruptedException e) {
e.printStackTrace();
}

long costTime=System.currentTimeMillis()-beginTime;
System.out.println(windowName+"为第"+number+"个"+type+"客户完成服务,耗时"+costTime/1000+"秒");
}else{
System.out.println(windowName+"没有取到任务,先休息1秒钟");
commonService();
}
}

private void vipService() {
String windowName="第"+windowId+"号"+type+"窗口";
Integer number=NumberMachine.getInstance().getVipManager().fetchServiceNumber();
System.out.println(windowName+"正在获取任务");
if(number!=null){
System.out.println(windowName+"为第"+number+"个"+type+"客户服务");
long beginTime=System.currentTimeMillis();
int maxRand=Constants.MAX_SERVICE_TIME-Constants.MIN_SERVICE_TIME;
long serveTime=new Random().nextInt(maxRand)+1+Constants.MIN_SERVICE_TIME;

try {
Thread.sleep(serveTime);
} catch (InterruptedException e) {
e.printStackTrace();
}

long costTime=System.currentTimeMillis()-beginTime;
System.out.println(windowName+"为第"+number+"个"+type+"客户完成服务,耗时"+costTime/1000+"秒");
}else{
System.out.println(windowName+"没有取到任务!");
commonService();
}
}

}


CustomerType枚举类

系统中有三种类型的客户,所以用定义一个枚举类,其中定义三个成员分别表示三种类型的客户。

重写toString方法,返回类型的中文名称。这是在后面编码时重构出来的,刚开始不用考虑。

package com.itheima.interview.bank;

public enum CustomerType {
COMMON,EXPRESS,VIP;

public String toString(){
switch(this){
case COMMON:
return "普通";
case EXPRESS:
return "快速";
case VIP:
return name();

}
return null;
}
}


MainClass类和Constants类

MainClass类

用for循环创建出4个普通窗口,再创建出1个快速窗口和一个VIP窗口。

接着再创建三个定时器,分别定时去创建新的普通客户号码、新的快速客户号码、新的VIP客户号码。

package com.itheima.interview.bank;

import java.util.concurrent.Executors;
import java.util.concurrent.TimeUnit;

public class MainClass {
public static void main(String[] args) {

//产生4个普通窗口
for(int i=1;i<5;i++){
ServiceWindow commonWindow = new ServiceWindow();
commonWindow.setWindowId(i);
commonWindow.start();
}

//产生1个快速窗口
ServiceWindow expressWindow = new ServiceWindow();
expressWindow.setType(CustomerType.EXPRESS);
expressWindow.start();

//产生1个VIP窗口
ServiceWindow vipWindow = new ServiceWindow();
vipWindow.setType(CustomerType.VIP);
vipWindow.start();

//普通客户取号
Executors.newScheduledThreadPool(1).scheduleAtFixedRate(
new Runnable(){
public void run(){
/**
* 采用logger方式,无法看到直观的运行效果,因为logger.log方法内部并不是直接把内容打印出出来,
* 而是交给内部的一个线程去处理,所以,打印出来的结果在时间顺序上看起来很混乱。
*/
//logger.info("第" + serviceNumber + "号普通客户正在等待服务!");

Integer number=NumberMachine.getInstance().getCommonManager().generateNewManager();
System.out.println(number+"号普通客户等待服务!");
}
},
0,
Constants.COMMON_CUSTOMER_INTERVAL_TIME,
TimeUnit.SECONDS
);

//快速客户取号
Executors.newScheduledThreadPool(1).scheduleAtFixedRate(
new Runnable(){
public void run(){
Integer number=NumberMachine.getInstance().getExpressManager().generateNewManager();
System.out.println(number+"号快速客户等待服务!");
}
},
0,
Constants.COMMON_CUSTOMER_INTERVAL_TIME*2,
TimeUnit.SECONDS
);

//VIP客户取号
Executors.newScheduledThreadPool(1).scheduleAtFixedRate(
new Runnable(){
public void run(){
Integer number=NumberMachine.getInstance().getVipManager().generateNewManager();
System.out.println(number+"号VIP客户等待服务!");
}
},
0,
Constants.COMMON_CUSTOMER_INTERVAL_TIME*6,
TimeUnit.SECONDS
);

}

}


Constants类

定义三个常量:

MAX_SERVICE_TIME

MIN_SERVICE_TIME

COMMON_CUSTOMER_INTERVAL_TIME

package com.itheima.interview.bank;

public class Constants {
public static int MAX_SERVICE_TIME=10000;//10秒
public static int MIN_SERVICE_TIME=1000;//1秒

/*每个普通窗口服务一个客户的平均时间为5秒,一共有4个这样的窗口,也就是说银行的所有普通窗口合起来
* 平均1.25秒内可以服务完一个普通客户,再加上快速窗口和VIP窗口也可以服务普通客户,所以,
* 1秒钟产生一个普通客户比较合理,*/
public static int COMMON_CUSTOMER_INTERVAL_TIME=1;
}


运行结果(部分):

第1号快速窗口正在获取任务

第3号普通窗口正在获取任务

第3号普通窗口没有取到任务,先休息1秒钟

第1号快速窗口没有取到任务,先休息1秒钟

第2号普通窗口正在获取任务

第2号普通窗口没有取到任务,先休息1秒钟

第1号普通窗口正在获取任务

第1号普通窗口没有取到任务,先休息1秒钟

第4号普通窗口正在获取任务

第4号普通窗口没有取到任务,先休息1秒钟

第1号VIP窗口正在获取任务

第1号VIP窗口没有取到任务!

第1号VIP窗口正在获取任务

第1号VIP窗口没有取到任务,先休息1秒钟

第1号快速窗口正在获取任务

第1号快速窗口没有取到任务,先休息1秒钟

1号普通客户等待服务!

1号快速客户等待服务!

1号VIP客户等待服务!

第1号普通窗口正在获取任务

2号普通客户等待服务!

第4号普通窗口正在获取任务

第4号普通窗口为第2个普通客户服务

第1号VIP窗口正在获取任务

第1号VIP窗口为第1个VIP客户服务

第1号普通窗口为第1个普通客户服务

第2号普通窗口正在获取任务

第2号普通窗口没有取到任务,先休息1秒钟

第3号普通窗口正在获取任务

第1号快速窗口正在获取任务

第3号普通窗口没有取到任务,先休息1秒钟

第1号快速窗口为第1个快速客户服务

3号普通客户等待服务!

2号快速客户等待服务!

第3号普通窗口正在获取任务

第1号快速窗口为第1个快速客户完成服务,耗时1秒

第1号快速窗口正在获取任务

第1号快速窗口为第2个快速客户服务

第2号普通窗口正在获取任务

第2号普通窗口没有取到任务,先休息1秒钟

第3号普通窗口为第3个普通客户服务

4号普通客户等待服务!

第2号普通窗口正在获取任务

第1号快速窗口为第2个快速客户完成服务,耗时1秒

第2号普通窗口为第4个普通客户服务

第1号快速窗口正在获取任务

第1号快速窗口没有取到任务,先休息1秒钟

第1号快速窗口正在获取任务

第1号快速窗口没有取到任务,先休息1秒钟

5号普通客户等待服务!

3号快速客户等待服务!

第1号快速窗口正在获取任务

第1号快速窗口为第3个快速客户服务

6号普通客户等待服务!

第1号快速窗口为第3个快速客户完成服务,耗时1秒

第1号快速窗口正在获取任务

第1号快速窗口没有取到任务,先休息1秒钟

第1号快速窗口正在获取任务

第1号快速窗口为第5个普通客户服务

第4号普通窗口为第2个普通客户完成服务,耗时4秒

第4号普通窗口正在获取任务

第4号普通窗口为第6个普通客户服务

7号普通客户等待服务!

2号VIP客户等待服务!

4号快速客户等待服务!

第1号VIP窗口为第1个VIP客户完成服务,耗时5秒

第1号VIP窗口正在获取任务

第1号VIP窗口为第2个VIP客户服务

第3号普通窗口为第3个普通客户完成服务,耗时4秒

第3号普通窗口正在获取任务

第3号普通窗口为第7个普通客户服务

8号普通客户等待服务!

9号普通客户等待服务!

5号快速客户等待服务!

第1号快速窗口为第5个普通客户完成服务,耗时3秒

第1号快速窗口正在获取任务

第1号快速窗口为第4个快速客户服务

-------
android培训、java培训、期待与您交流! ----------  详细请查看:http://edu.csdn.net/heima/
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: