银行业务调度系统的项目

这是我第一个博客,这个是从其他网站上找来的资料,并用自己的理解来写的,有很多不对的地方请包涵,最后我会附上 张孝祥-7K月薪面试题破解之二_银行业务调度系统视频教程 原文件。

银行业务调度系统的项目需求
模拟实现银行业务调度系统逻辑,具体需求如下:
*银行内有6个业务窗口,1 - 4号窗口为普通窗口,5号窗口为快速窗口,6号窗口为VIP窗口。
有三种对应类型的客户:VIP客户,普通客户,快速客户(办理如交水电费、电话费之类业务的客户)。
异步随机生成各种类型的客户,生成各类型用户的概率比例为:
VIP客户 :普通客户 :快速客户 = 1 :6 :3。
客户办理业务所需时间有最大值和最小值,在该范围内随机设定每个VIP客户以及普通客户办理业务所需的时间,快速客户办理业务所需时间为最小值(提示:办理业务的过程可通过线程Sleep的方式模拟)
各类型客户在其对应窗口按顺序依次办理业务。
当VIP(6号)窗口和快速业务(5号)窗口没有客户等待办理业务的时候,这两个窗口可以处理普通客户的业务,而一旦有对应的客户等待办理业务的时候,则优先处理对应客户的业务。
随机生成客户时间间隔以及业务办理时间最大值和最小值自定,可以设置。
不要求实现GUI,只考虑系统逻辑实现,可通过Log方式展现程序运行结果。*

分析该需求,java是面向对象,我们需要从该需求中找到相关的对象,然后在从其中找属性和方法。
结合实际,我们可以找到客户类,窗口类,机器类(该类是从实际情况中分析出来的)。
客户类属性:姓名
窗口类属性:窗口编号,窗口姓名。
机器类属性:该机器中有三个管理类,一个是普通号管理类,一个是快速管理类,一个是VIP管理类,管理类中包涵了来的客户的编号,和需要维护的客户队列。

这个分析是慢慢的积累出来的,也有一些课程来,比如UML中类的课程就讲了怎样找类和类的属性、方法。而且找类后在实际编写代码时都有可能在分出类来,这些类是实际需要。
客户类:由于该程序很小,客户类中没有其他的东西,只有一个名字属性,并且不需要客户该对象。这里我们可以创建一个enum,该类中有三个COMMON,FAST,VIP分别表示普通客户,快速客户,vip客户。

 public enum Customer {
    COMMON, FAST, VIP;
    public String toString() {
        String name = null;
        switch (this) {
        case COMMON:
            name = "普通";
        case FAST:
            name = "快速";
        case VIP:
            name = name();
        }
        return name;
    }
}
  该enum中重写了toString()方法,这是用户返回一组汉字。
  这样客户类创建好了

  窗口类中属性窗口编号,窗口名称
  方法:叫号方法
  从实际中,我们可以发现银行的窗口叫号,然后客户才来
  public class WindonService {
    private Customer name = Customer.COMMON;        //窗口名称
    private int number = 1;     //窗口编号


    /**
     * 要完成的任务
     */
    public void start(){
        //开启线程来完成
        Executors.newSingleThreadExecutor().execute(new Runnable() {
            @Override
            public void run() {
                while(true){
                    switch (name) {
                    case COMMON:
                        commonService();
                        break;

                    case FAST:
                        fastService();
                        break;

                    case VIP:
                        vipService();
                        break;
                    }
                }
            }
        });

    }

    /**
     *vip窗口叫号服务
     */
    protected void vipService() {
        String windonservice = "第" + number + "号" + name + "窗口";
        System.out.println(windonservice + ",开始获取vip服务!");
        //取号
        Integer serviceNumber = NumberMachine.getIntance().getVipManager().fetchNumber();
        //判断号码是否为空,如果为空,就等待一段时间
        if (serviceNumber != null) {
            System.out.println(windonservice+ "正在为第" + serviceNumber+"vip客户服务");
            //计算服务时间
            //服务时间
            int randomTime = Constans.MAX_TIME - Constans.MIN_TIME;
            int ServiceTime = new Random().nextInt(randomTime)+1 + Constans.MIN_TIME;
            try {
                Thread.sleep(ServiceTime);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
            System.out.println(windonservice + "为第" + serviceNumber + "vip客户服务" + ServiceTime/1000 +"时间");
        }else {
            System.out.println("没有取到vip任务,等待一秒钟");
            //取到普通号码,则服务普通号码
            commonService();
        }
    }




    /**
     * 快速窗口叫号服务
     */
    protected void fastService() {
        String windonservice = "第" + number + "号" + name + "窗口";
        System.out.println(windonservice + ",开始获取快速服务!");
        //取号
        Integer serviceNumber = NumberMachine.getIntance().getFastManager().fetchNumber();
        //判断号码是否为空,如果为空,就等待一段时间
        if (serviceNumber != null) {
            System.out.println(windonservice+ "正在为第" + serviceNumber+"快速客户服务");
            //快速服务时间为最小时间
            int serviceTime = Constans.MIN_TIME;
            try {
                Thread.sleep(serviceTime);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
            System.out.println(windonservice + "为第" + serviceNumber + "快速客户服务" + serviceTime/1000 +"时间");
        }else {
            System.out.println("没有取到快速任务,等待一秒钟");
            //快速窗口没有取到快速客户的号码,则试着去取普通号码
            //取到普通号码,则服务普通号码
            commonService();
        }
    }

    /**
     * 普通窗口的服务
     */
    protected void commonService() {
        String windonservice = "第" + number + "号" + name + "窗口";
        System.out.println(windonservice + ",开始获取普通服务!");
        //取号
        Integer serviceNumber = NumberMachine.getIntance().getCommonManager().fetchNumber();
        //判断号码是否为空,如果为空,就等待一段时间
        if (serviceNumber != null) {
            System.out.println(windonservice+ "正在为第" + serviceNumber+"普通客户服务");
            //计算服务时间
            //服务时间
            int randomTime = Constans.MAX_TIME - Constans.MIN_TIME;
            int ServiceTime = new Random().nextInt(randomTime)+1 + Constans.MIN_TIME;
            try {
                Thread.sleep(ServiceTime);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
            System.out.println(windonservice + "为第" + serviceNumber + "普通客户服务" + ServiceTime/1000 +"时间");
        }else {
            System.out.println("没有取到普通任务,等待一秒钟");
            try {
                Thread.sleep(1000);         //等待一秒钟
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        }
    }





    // ---------------------------------------------
    public Customer getName() {
        return name;
    }


    public void setName(Customer name) {
        this.name = name;
    }


    public int getNumber() {
        return number;
    }


    public void setNumber(int number) {
        this.number = number;
    }

}

该类中我没有进行优化。所有每个窗口进行叫号,这样,因为有三个窗口,这样我们进行一下区分,进行start()方法,进入start()中后,会进行判断那一个窗口,并进行相应的方法。
普通窗口叫号时,先取出号码,并判断是否有该号码,没有则等待一分钟在叫,有那么就行该客户服务。该客户服务有一个时间,根据需求,需要在最小时间和最大时间之间的随机值。随后就结束服务。
快速窗口叫号时,先取出号码,并判断是否有该号码,没有,这时需要去询问是否有普通客户,此时直接调用普通窗口叫号服务,有那么就行该客户服务。该客户服务有一个时间,根据需求,直接是最小时间。
快速窗口叫号时,先取出号码,并判断是否有该号码,没有,这时需要去询问是否有普通客户,此时直接调用普通窗口叫号服务,有那么就行该客户服务。该客户服务有一个时间,根据需求,需要在最小时间和最大时间之间的随机值。随后就结束服务。

此时该服务类就可以完成。其中会发现如何去得到一个号码呢。该号码是有机器产生的,这时就需要创建机器类,机器类中来创建一个号码管理器,该号码管理器是管理每个到来客户的一个信息的,比如普通客户到时机器到了几号了,同时要维护每次到来客户的队列。该号码管理器能够产生两个方法,取号,产生号码方法。如下:

public class NumberManager {
    private Integer lastNumber = 0;
    private List queueNumber = new ArrayList();

    /**
     * 生成号码
     */
    public synchronized Integer generateNumber(){
        queueNumber.add(++lastNumber);
        return lastNumber;
    }
    /**
     * 取号码
     * @return
     */
    public synchronized Integer fetchNumber(){
        if (queueNumber.size() > 0) {
            return queueNumber.remove(0);
        }else {
            return null;
        }
    }
}

机器管理类只需维护三个号码管理器就可以了,在实际中机器只有一台,那么该处用到单例,单例的方法如下:
private Class(){}
private static Class mInstance;
public static Class getInstance(){
if(mInstance == null){//Class为类名
mInstance = new Class();
}
return mInstance;
}

机器管理类如下:

public class NumberMachine {
    /**
     * 由于叫号机器只有一台,必须给其创建单利
     * 机器可以生成三种不同号码,普通号码,快速号码,vip号码
     */
    private NumberMachine(){}
    private static NumberMachine mInstance;

    public static NumberMachine getIntance(){
        if (mInstance== null) {
            mInstance = new NumberMachine();
        }
        return mInstance;
    }

    private NumberManager commonManager = new NumberManager();
    private NumberManager fastManager = new NumberManager();
    private NumberManager vipManager = new NumberManager();

    public NumberManager getCommonManager() {
        return commonManager;
    }
    public NumberManager getFastManager() {
        return fastManager;
    }
    public NumberManager getVipManager() {
        return vipManager;
    }

}

这里有个常量类:

public class Constans {
    public static final int MAX_TIME = 10000;           //最大服务时间是10秒
    public static final int MIN_TIME = 1000;            //最小服务时间是1秒
}

那么这时完成大部分功能了,有一个功能没有完成,那是怎样随机产生客户,这个会在主方法中完成:
主方法要完成两个功能:
分别创建6个窗口;
按要求产生客户;

按比例产生号码,我想到了使用scheduleAtFixedRate,该方法是周期性的完成任务。其中调用时,有四个参数,一个参数是Runnable,是要完成需要执行的方法,initialDelay 初始延迟时间,period周期,TimeUnit周期的时间单位。
Executors.newScheduledThreadPool(1).scheduleAtFixedRate(new Runnable() {
@Override
public void run() {
}
}, 0, 1, TimeUnit.SECONDS);
这是该方法的调用。

    public static void main(String[] args) {
        /**
         * 创建出6个窗口来
         * 然后机器运行起来,生成号码
         */
        for (int i = 1; i < 5; i++) {   //创建4个普通窗口
            WindonService commonService = new WindonService();
            commonService.setName(Customer.FAST);
            commonService.start();
        }

        WindonService fastService = new WindonService();    //创建1个快速窗口
        fastService.setName(Customer.VIP);
        fastService.start();

        WindonService vipService = new WindonService(); //创建1个vip窗口
        vipService.setNumber(1);
        vipService.start();

        Executors.newScheduledThreadPool(1).scheduleAtFixedRate(new Runnable() {
            @Override
            public void run() {
                Integer serviceNumber = NumberMachine.getIntance().getCommonManager().generateNumber();
                System.out.println("第" + serviceNumber + "号普通客户正在等待服务!");
            }
        }, 0, 1, TimeUnit.SECONDS);

        Executors.newScheduledThreadPool(1).scheduleAtFixedRate(new Runnable() {
            @Override
            public void run() {
                Integer serviceNumber = NumberMachine.getIntance().getFastManager().generateNumber();
                System.out.println("第" + serviceNumber + "号快速客户正在等待服务!");
            }
        }, 0, 3, TimeUnit.SECONDS);

        Executors.newScheduledThreadPool(1).scheduleAtFixedRate(new Runnable() {
            @Override
            public void run() {
                Integer serviceNumber = NumberMachine.getIntance().getVipManager().generateNumber();
                System.out.println("第" + serviceNumber + "号vip客户正在等待服务!");
            }
        }, 0, 6, TimeUnit.SECONDS);
    }

总结:
第一结合实际我们要找到有哪几个类,并且确定该类要完成的那个属性和方法。那个接下来的工作就很简单了。
那面我就是找到几个类,最后在主方法中进行创建相关的对象。这里还需要有一些经验的地方是scheduleAtFixedRate,该方法的使用。
大家可以下在下列视屏看

参考:

张孝祥-7K月薪面试题破解之二_银行业务调度系统视频教程

你可能感兴趣的:(java)