黑马程序员---张老师银行业务调度系统学习

------- android培训java培训、期待与您交流! ----------

银行业务调度系统系统需求

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

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

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

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

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

4. 客户办理业务所需时间有最大值和最小值,在该范围内随机设定每个VIP客户以及普通客户办理业务所需的时间,快速客户办理业务所需时间为最小值(提示:办理业务的过程可通过线程Sleep的方式模拟)。 各类型客户在其对应窗口按顺序依次办理业务。 

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

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

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

需求分析:

需求中涉及的对象(现实中的):窗口客户。需求中描述客户的是2、3和6,客户办理业务其实银行 处理业务,所以可以把办理业务的方法归到窗口中。这两类对象是怎么建立联系的?是直接把客户作为窗口的参数传递进去吗?肯定不是,现实中他们是通过取号机器建立联系的,所以需求中隐含一个取号机器这个对象。所以,需求中涉及三个现实世界中的对象:窗口,客户取号机器。从现实空间映射到问题空间,就需要建立三个对应的类。

   所有类型的窗口,都能为普通客户提供服务,不过对于VIP和快速窗口,VIP和快速客户的优先级高,根据客户类型调用不同的处理方法。因为有优先级,所以给每个窗口添加个type属性,以标示其默认服务的客户类型;同时定义一个窗口ID,以给窗口标号。由这两个属性,我们就可以知道,这个窗口是VIP、普通还是快速窗口了。

  在模拟的时候,因为客户和窗口是多对一的关系,它唯一的行为就是等待办理业务,而这个时间其实就是窗口办理业务的时间。所以它不需要具体定义一个类,用数字表示就可以了。

  客户来到银行,不是先到窗口,而是先取号,而且是根据他需要VIP、普通还会快速服务来取号。所以取号机要对其进行分类,可以在取号机中完成所有的工作,但是因为对不同的客户,它要都建立客户与窗口间的联系,为客户提供号码,向窗口提供要服务的客户号码,虽然能在一个类中实现,但为了减少耦合,最好建立几个号码管理器,并把取号机设计为单例。

  建立一个集合属性存储客户和一个属性记录下一个要服务的客户号码,同时向外提供两个方法,分别使客户和窗口取得号码;因为多线程访问方法,所以这两个方法要加锁。

代码实现:

1.号码管理器

package cn.itcast.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)
			return queueNumber.remove(0);
		return number;
	}
}

2.取号机

package cn.itcast.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 NumberMachine(){}
	private static NumberMachine instance = new NumberMachine();
	public static NumberMachine getInstance(){
		return instance;
	}
}
3.服务窗口

package cn.itcast.bank;

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

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 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 serverTime = new Random().nextInt(maxRand) + 1 +Constants.MIN_SERVICE_TIME;
			try {
				Thread.sleep(serverTime);
			} catch (Exception e) {
				e.printStackTrace();
			}
			long useTime = (System.currentTimeMillis() - beginTime)/1000;
			System.out.println(windowName+"为第"+number+"个"+type+"客户完成服务,耗时"+useTime+"秒");
		}
		else{
			System.out.println(windowName+"没有取到服务任务");
			commonService();
		}
	}
	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 (Exception e) {
				e.printStackTrace();
			}
			long useTime = (System.currentTimeMillis() - beginTime)/1000;
			System.out.println(windowName+"为第"+number+"个"+type+"客户完成服务,耗时"+useTime+"秒");
		}
		else{
			System.out.println(windowName + "没有取到服务任务");
			commonService();
		}
	}

	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 serverTime = new Random().nextInt(maxRand) + 1 +Constants.MIN_SERVICE_TIME;
			try {
				Thread.sleep(serverTime);
			} catch (Exception e) {
				e.printStackTrace();
			}
			long useTime = (System.currentTimeMillis() - beginTime)/1000;
			System.out.println(windowName+"为第"+number+"个"+"普通"+"客户完成服务,耗时"+useTime+"秒");
		}
		else{
			System.out.println(windowName+"没有取到服务任务,休息一秒");
			try {
				Thread.sleep(1000);
			} catch (Exception e) {
				e.printStackTrace();
			}
		}
	}
}

4.常量类(为了满足条件6)
package cn.itcast.bank;

public class Constants {
	public static int MAX_SERVICE_TIME = 10000;
	public static int MIN_SERVICE_TIME = 1000;
	public static int COMMON_CUTOMER_INTERVAL_TIME = 1;	
}

5.客户类型枚举类

package cn.itcast.bank;

public enum CustomerType {
	COMMON,EXPRESS,VIP;
	public String toString(){
		switch(this){
		case COMMON:
			return "普通";
		case EXPRESS:
			return "快速";
		case VIP:
			return "VIP";
		}
		return null;
	}
}

6.启动运行类

import java.util.concurrent.*;

public class MainClass {
	public static void main(String[] args){
		for(int i=1;i<5;i++){
			ServiceWindow commonWindow = new ServiceWindow();
			commonWindow.setWindowId(i);
			commonWindow.start();
		}
		ServiceWindow expressWindow = new ServiceWindow();
		expressWindow.setType(CustomerType.EXPRESS);
		expressWindow.start();
		
		ServiceWindow vipWindow = new ServiceWindow();
		vipWindow.setType(CustomerType.VIP);
		vipWindow.start();
		
		Executors.newScheduledThreadPool(1).scheduleAtFixedRate(
				new Runnable(){
					public void run(){
						Integer number = NumberMachine.getInstance().getCommonManager().generateNewManager();
						System.out.println(number +"号普通客户等待服务!");
					}
				}, 
				0, 
				Constants.COMMON_CUTOMER_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_CUTOMER_INTERVAL_TIME*6, 
				TimeUnit.SECONDS
			);
		Executors.newScheduledThreadPool(1).scheduleAtFixedRate(
				new Runnable(){
					public void run(){
						Integer number = NumberMachine.getInstance().getVipManager().generateNewManager();
						System.out.println(number +"号VIP客户等待服务!");
					}
				}, 
				0, 
				Constants.COMMON_CUTOMER_INTERVAL_TIME*2, 
				TimeUnit.SECONDS
			);
	}
}
个人总结:

1.要树立良好的面向对象编程思想,记住一个重要原则,谁拥有数据,谁就对外提供操作这些数据的方法.

2.判定一个动作或一个方法属于哪一个类的时候,要看谁对这个方法的实现方式最清楚,

例如:人关门,关门这个方法是门的,而人只是调用了门的关门方法,传递了一个力,不是人在关门,而是门在关门.

3.在思考一个问题或需求的时候,最有效的解决方式是先画图,把各种对象,元素,方法都画出来,先思考好相互之间的所属关系及联系,

在图上标明,只要将设计思路确定以后,敲代码就是在实现设计思路了.敲程序前先画图!!!


------- android培训java培训、期待与您交流! ----------

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值