------- android培训、java培训、期待与您交流! ----------
java专题之银行业务调度系统
1.首先让我们来看看具体的业务逻辑
模拟实现银行业务调度系统逻辑,具体需求如下:
银行内有6个业务窗口,1 - 4号窗口为普通窗口,5号窗口为快速窗口,6号窗口为VIP窗口。
有三种对应类型的客户:VIP客户,普通客户,快速客户(办理如交水电费、电话费之类业务的客户)。
异步随机生成各种类型的客户,生成各类型用户的概率比例为:
VIP客户 :普通客户 :快速客户 = 1 :6 :3。
客户办理业务所需时间有最大值和最小值,在该范围内随机设定每个VIP客户以及普通客户办理业务所需的时间,快速客户办理业务所需时间为最小值(提示:办理业务的过程可通过线程Sleep的方式模拟)。
各类型客户在其对应窗口按顺序依次办理业务。
当VIP(6号)窗口和快速业务(5号)窗口没有客户等待办理业务的时候,这两个窗口可以处理普通客户的业务,而一旦有对应的客户等待办理业务的时候,则优先处理对应客户的业务。
随机生成客户时间间隔以及业务办理时间最大值和最小值自定,可以设置。
不要求实现GUI,只考虑系统逻辑实现,可通过Log方式展现程序运行结果。
第一步先尝试着使用面向对象的编程思想对具体的业务逻辑进行分析,看看需要使用到哪些对象,以及哪些方法。
1).客户到银行办理相关的业务,就必须到一个客户服务机前获取相关的客户号,之后凭借这个服务号排队等待服务,从这句话中我们可以抽取到我们必须要有一个客户服务机对象,所
以我们现在就确定了第一个对象 “客户服务机” ,又因为客户服务机只有一台,不能让别人来创建,所以我们必须使用到单例模式,创建一个 NumberMachine 的类,并让该类实现单例模式
2).我们再来看一下客户,客户分为三类,即普通客户,vip客户,快速客户,他们的比例为6 : 1 : 3,他们先是到客户服务机取得服务号,之后又到对应的窗口中办理业务,所以还要有一个
号码管理器,用它来管理着三种类型的服务号,并且号码管理器必须存在于NumberMachine中,号码管理器中还必须封装方法,调用这个方法可以得到服务号,也就是取得服务号的行为,所以
我们还要在号码管理器中定义一个集合,用来存储这些服务号,另外为了方便服务窗口叫号,服务窗口怎么知道到底该叫多少号,这还是要问号码管理器,所以还必须定义另外一个方法
,可以让服务窗口知道该叫多少号,最后因为客户分三种,所以还要在号码管理器中定义一个属性,用于区分这些号码管理器对象。第二个类 NumberManger 就出现了
3).第三个要思考的类就是服务窗口了,服务窗口和客户一样分为三类,每个服务窗口应该有自己的Id号和类型这一属性,并且能设置和获取他们,这是因为根据生活的客观规律,当大量的
客户去银行办理业务时,如果普通客户非常多,而vip客户和快速客户没有时,vip服务窗口和快速服务窗口可以临时改为普通窗口进行业务办理,一旦当这些窗口来了对应的客户时,则
优先办理这些对应的客户。又因为三种不同的窗口办理业务的内容不同,所以我们要定义三个不同的方法来供窗口调用,这就是第三个类 ServiceWindow
4).最后一个要考虑的也就是客户了客户只有三种,不能多,也不能少,所以这时我们就应该想到要用枚举了,得定义一个 CustomerType 这个枚举
2.面向对象的思考结束后,就是具体的代码实现了
1)NumberMachine
public class NumberMachine {
private NumberMachine(){} //将构造方法私有化
private static NumberMachine instance = new NumberMachine(); //创建一个该类本身的对象,并用static修饰
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;
}
}
2)NumberManger
import java.util.ArrayList;
import java.util.List;
public class NumberManager {
private int lastNumber = 0; //定义一个属性,由于表示上一个客户的服务号
private List queueNumbers = new ArrayList(); //创建一个集合对象,用于存储客户的服务号码
public synchronized Integer generateNewNumber(){ //定义一个生成号码的方法,并将其返回给客户,之后将号码存入集合,因为涉及到多线程访问同一资源,所以必须加上synchronized关键字
queueNumbers.add(++lastNumber);
return lastNumber;
}
public synchronized Integer fetchNumber(){ //该方法用于服务窗口不清楚应缴多少号时调用,返回集合中第一个对象,同样也是多线程,所以要加synchronized修饰
if(queueNumbers.size()>0){
return (Integer)queueNumbers.remove(0);
}else{
return null;
}
}
}
3)CustomerType
public enum CustomerType {
COMMON,EXPRESS,VIP;
public String toString(){ //重写toString方法,为对象设置名字
String name = null;
switch(this){
case COMMON:
name = "普通";
break;
case EXPRESS:
name = "快速";
break;
case VIP:
name = name();
break;
}
return name;
}
}
4)ServiceWindow
import java.util.Random;
import java.util.concurrent.Executors;
import java.util.logging.Logger;
/**
* 没有把VIP窗口和快速窗口做成子类,是因为实际业务中的普通窗口可以随时被设置为VIP窗口和快速窗口。
* */
public class ServiceWindow {
private static Logger logger = Logger.getLogger("cn.itcast.bankqueue");
private CustomerType type = CustomerType.COMMON; //因为普通窗口有四个,所以初始化将其设置为普通窗口类型
private int number = 1; //定义number属性,用于给窗口打上编号
public CustomerType getType() { //拿到服务窗口的类型
return type;
}
public void setType(CustomerType type) { //设置服务窗口的类型
this.type = type;
}
public void setNumber(int number){ //设置窗口的编号
this.number = number;
}
public void start(){ //调用这个方法时,服务窗口开始工作
Executors.newSingleThreadExecutor().execute( //Executors是jdk1.5之后新增加的一个类,总之调用这段代码会创建一个新线程
new Runnable(){
public void run(){
//下面这种写法的运行效率低,最好是把while放在case下面
while(true){
switch(type){ //根据type属性判断执行哪个方法
case COMMON:
commonService();
break;
case EXPRESS:
expressService();
break;
case VIP:
vipService();
break;
}
}
}
}
);
}
private void commonService(){ //该方法用于为普通客户服务的窗口调用
String windowName = "第" + number + "号" + type + "窗口";
System.out.println(windowName + "开始获取普通任务!");
Integer serviceNumber = NumberMachine.getInstance().getCommonManager().fetchNumber();
if(serviceNumber != null ){
System.out.println(windowName + "开始为第" + serviceNumber + "号普通客户服务");
int maxRandom = Constants.MAX_SERVICE_TIME - Constants.MIN_SERVICE_TIME; //拿到服务最长时间值减去最短时间值之差
int serviceTime = new Random().nextInt(maxRandom)+1 + Constants.MIN_SERVICE_TIME; //用一个随机数产生器生成一个随机数,该数在指定范围内
try {
Thread.sleep(serviceTime); //该线程睡眠一段时间
} catch (InterruptedException e) {
e.printStackTrace();
}
System.out.println(windowName + "完成为第" + serviceNumber + "号普通客户服务,总共耗时" + serviceTime/1000 + "秒");
}else{
System.out.println(windowName + "没有取到普通任务,正在空闲一秒");
try {
Thread.sleep(1000); //没有取得任务则睡眠一秒
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}
private void expressService(){ //该方法用于为快速客户服务的窗口调用
Integer serviceNumber = NumberMachine.getInstance().getExpressManager().fetchNumber();
String windowName = "第" + number + "号" + type + "窗口";
System.out.println(windowName + "开始获取快速任务!");
if(serviceNumber !=null){
System.out.println(windowName + "开始为第" + serviceNumber + "号快速客户服务");
int serviceTime = Constants.MIN_SERVICE_TIME;
try {
Thread.sleep(serviceTime); //快速服务窗口的服务时间是最短的。
} catch (InterruptedException e) {
e.printStackTrace();
}
System.out.println(windowName + "完成为第" + serviceNumber + "号快速客户服务,总共耗时" + serviceTime/1000 + "秒");
}else{
System.out.println(windowName + "没有取到快速任务!");
commonService(); //如果没有取得快速任务则该窗口可以作为普通窗口为普通客户服务
}
}
private void vipService(){ //该方法用于为vip客户服务的窗口调用
Integer serviceNumber = NumberMachine.getInstance().getVipManager().fetchNumber();
String windowName = "第" + number + "号" + type + "窗口";
System.out.println(windowName + "开始获取VIP任务!");
if(serviceNumber !=null){
System.out.println(windowName + "开始为第" + serviceNumber + "号VIP客户服务");
int maxRandom = Constants.MAX_SERVICE_TIME - Constants.MIN_SERVICE_TIME;
int serviceTime = new Random().nextInt(maxRandom)+1 + Constants.MIN_SERVICE_TIME;
try {
Thread.sleep(serviceTime);
} catch (InterruptedException e) {
e.printStackTrace();
}
System.out.println(windowName + "完成为第" + serviceNumber + "号VIP客户服务,总共耗时" + serviceTime/1000 + "秒");
}else{
System.out.println(windowName + "没有取到VIP任务!");
commonService(); //如果没有取得快速任务则该窗口可以作为普通窗口为普通客户服务
}
}
}
5)Constants
/*
*该类是为了更好的管理程序中使用的数据,而建立起来的一个类
*当其他类大量的用到该类中的数据时,我们就要考虑建立一个类,来管理这些数据,当我们要修改这些数据时,我们直接修类中的相关变量即可,这样就增加了程序的维护性
**/
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;
}
最后如果程序想要运行还需要一个主函数
6)MainClass
import java.util.concurrent.Executors;
import java.util.concurrent.TimeUnit;
import java.util.logging.Logger;
public class MainClass {
private static Logger logger = Logger.getLogger("cn.itcast.bankqueue");
public static void main(String[] args) {
//产生4个普通窗口
for(int i=1;i<5;i++){
ServiceWindow window = new ServiceWindow();
window.setNumber(i);
window.start(); //普通窗户开始工作
}
//产生1个快速窗口
ServiceWindow expressWindow = new ServiceWindow();
expressWindow.setType(CustomerType.EXPRESS);
expressWindow.start(); //快速窗口开始工作
//产生1个VIP窗口
ServiceWindow vipWindow = new ServiceWindow();
vipWindow.setType(CustomerType.VIP);
vipWindow.start(); //vip窗户开始工作
//普通客户拿号
Executors.newScheduledThreadPool(1).scheduleAtFixedRate( //jdk1.5的新技术,创建一个线程池,该线程池中只有一个线程对象,调用该线程的scheduleAtFixedRate方法可以对线程指定时间执行
new Runnable(){
public void run(){
Integer serviceNumber = NumberMachine.getInstance().getCommonManager().generateNewNumber(); //模拟普通客户通过generateNewNumber()方法拿到号码
/**
* 采用logger方式,无法看到直观的运行效果,因为logger.log方法内部并不是直接把内容打印出出来,
* 而是交给内部的一个线程去处理,所以,打印出来的结果在时间顺序上看起来很混乱。
*/
//logger.info("第" + serviceNumber + "号普通客户正在等待服务!");
System.out.println("第" + serviceNumber + "号普通客户正在等待服务!");
}
},
0,
Constants.COMMON_CUSTOMER_INTERVAL_TIME,
TimeUnit.SECONDS);
//快速客户拿号
Executors.newScheduledThreadPool(1).scheduleAtFixedRate(
new Runnable(){
public void run(){
Integer serviceNumber = NumberMachine.getInstance().getExpressManager().generateNewNumber();
System.out.println("第" + serviceNumber + "号快速客户正在等待服务!");
}
},
0,
Constants.COMMON_CUSTOMER_INTERVAL_TIME * 2,
TimeUnit.SECONDS);
//VIP客户拿号
Executors.newScheduledThreadPool(1).scheduleAtFixedRate(
new Runnable(){
public void run(){
Integer serviceNumber = NumberMachine.getInstance().getVipManager().generateNewNumber();
System.out.println("第" + serviceNumber + "号VIP客户正在等待服务!");
}
},
0,
Constants.COMMON_CUSTOMER_INTERVAL_TIME * 6,
TimeUnit.SECONDS);
}
}
至此,张孝祥老师视频指导的项目基本完成,完成这个项目有利于我们对面向对象编程的理解,一切事物皆可封装成对象,在这个简单的小项目中,号码机器,号码管理器,服务窗口
客户类型甚至连需要用到的普通的常量都被封装成对象,真正实现了面向对象的编程.