银行业务调度系统_编码思路及代码

2024-06-16 09:48

本文主要是介绍银行业务调度系统_编码思路及代码,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

1,面试题目:银行业务调度系统

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

  • 银行内有6个业务窗口,1 — 4号窗口为普通窗口,5号窗口为快速窗口,6号窗口为VIP窗口。
  • 有三种对应类型的客户:VIP客户,普通客户,快速客户(办理如交水电费、电话费之类业务的客户)。
  • 异步随机生成各种类型的客户,生成各类型用户的概率比例为:

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

  • 客户办理业务所需时间有最大值和最小值,在该范围内随机设定每个VIP客户以及普通客户办理业务所需的时间,快速客户办理业务所需时间为最小值(提示:办理业务的过程可通过线程Sleep的方式模拟)。
  • 各类型客户在其对应窗口按顺序依次办理业务。 
  • 当VIP6号)窗口和快速业务(5号)窗口没有客户等待办理业务的时候,这两个窗口可以处理普通客户的业务,而一旦有对应的客户等待办理业务的时候,则优先处理对应客户的业务。 
  • 随机生成客户时间间隔以及业务办理时间最大值和最小值自定,可以设置。
  • 不要求实现GUI,只考虑系统逻辑实现,可通过Log方式展现程序运行结果。

 

2,面向对象的分析与设计——面向对象设计把握一个重要的经验:谁拥有数据,谁就对外提供操作这些数据的方法。

 

画类图分析:

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

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

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

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

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

    ②:如果我不是多次亲身经历银行的这种业务,再加之积累了大量面向对象的应用开发经验,我也不知道能否轻松进行这种设计,能否发掘出其中隐含的对象信息,我真说不出具体的经验是什么,就是日积月累出来的一种感觉。难道这就是传说中的:“只可意会,不可言传?”

 

1,NumberManager类

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

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

public class NumberManager {
  private int lastNumber = 0;
  private List queueNumbers = new ArrayList();

  public synchronized Integer generateNewNumber(){
    queueNumbers.add(++lastNumber);
    return lastNumber;
  }

  public synchronized Integer fetchNumber(){
    if(queueNumbers.size()>0){
      return (Integer)queueNumbers.remove(0);
    }

    else{
      return null;
    }
  }
}

2,NumberMachine类

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

  ②:将NumberMachine类设计成单例。

public class NumberMachine {

  private NumberMachine(){}
  private static NumberMachine instance = new NumberMachine();
  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;
  }
}

3,CustomerType枚举类

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

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

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

4,ServiceWindow类

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

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

//没有把VIP窗口和快速窗口做成子类,是因为实际业务中的普通窗口可以随时被设置为VIP窗口和快速窗口。
public class ServiceWindow {
  private static Logger logger = Logger.getLogger("cn.itcast.bankqueue");
  private CustomerType type = CustomerType.COMMON;
  private int number = 1;

  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(
      new Runnable(){
        public void run(){
          //下面这种写法的运行效率低,最好是把while放在case下面
          while(true){
            switch(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(){

    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,MainClass类

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

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

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(); 

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

          System.out.println
        }
      },
      0,
      Constants.COMMON_CUSTOMER_INTERVAL_TIME,


    
    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, 

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

}

6,Constants类

  ①:定义三个常量:MAX_SERVICE_TIME、MIN_SERVICE_TIME、COMMON_CUSTOMER_INTERVAL_TIME

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

这篇关于银行业务调度系统_编码思路及代码的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1066128

相关文章

不懂推荐算法也能设计推荐系统

本文以商业化应用推荐为例,告诉我们不懂推荐算法的产品,也能从产品侧出发, 设计出一款不错的推荐系统。 相信很多新手产品,看到算法二字,多是懵圈的。 什么排序算法、最短路径等都是相对传统的算法(注:传统是指科班出身的产品都会接触过)。但对于推荐算法,多数产品对着网上搜到的资源,都会无从下手。特别当某些推荐算法 和 “AI”扯上关系后,更是加大了理解的难度。 但,不了解推荐算法,就无法做推荐系

基于人工智能的图像分类系统

目录 引言项目背景环境准备 硬件要求软件安装与配置系统设计 系统架构关键技术代码示例 数据预处理模型训练模型预测应用场景结论 1. 引言 图像分类是计算机视觉中的一个重要任务,目标是自动识别图像中的对象类别。通过卷积神经网络(CNN)等深度学习技术,我们可以构建高效的图像分类系统,广泛应用于自动驾驶、医疗影像诊断、监控分析等领域。本文将介绍如何构建一个基于人工智能的图像分类系统,包括环境

水位雨量在线监测系统概述及应用介绍

在当今社会,随着科技的飞速发展,各种智能监测系统已成为保障公共安全、促进资源管理和环境保护的重要工具。其中,水位雨量在线监测系统作为自然灾害预警、水资源管理及水利工程运行的关键技术,其重要性不言而喻。 一、水位雨量在线监测系统的基本原理 水位雨量在线监测系统主要由数据采集单元、数据传输网络、数据处理中心及用户终端四大部分构成,形成了一个完整的闭环系统。 数据采集单元:这是系统的“眼睛”,

嵌入式QT开发:构建高效智能的嵌入式系统

摘要: 本文深入探讨了嵌入式 QT 相关的各个方面。从 QT 框架的基础架构和核心概念出发,详细阐述了其在嵌入式环境中的优势与特点。文中分析了嵌入式 QT 的开发环境搭建过程,包括交叉编译工具链的配置等关键步骤。进一步探讨了嵌入式 QT 的界面设计与开发,涵盖了从基本控件的使用到复杂界面布局的构建。同时也深入研究了信号与槽机制在嵌入式系统中的应用,以及嵌入式 QT 与硬件设备的交互,包括输入输出设

JAVA智听未来一站式有声阅读平台听书系统小程序源码

智听未来,一站式有声阅读平台听书系统 🌟&nbsp;开篇:遇见未来,从“智听”开始 在这个快节奏的时代,你是否渴望在忙碌的间隙,找到一片属于自己的宁静角落?是否梦想着能随时随地,沉浸在知识的海洋,或是故事的奇幻世界里?今天,就让我带你一起探索“智听未来”——这一站式有声阅读平台听书系统,它正悄悄改变着我们的阅读方式,让未来触手可及! 📚&nbsp;第一站:海量资源,应有尽有 走进“智听

活用c4d官方开发文档查询代码

当你问AI助手比如豆包,如何用python禁止掉xpresso标签时候,它会提示到 这时候要用到两个东西。https://developers.maxon.net/论坛搜索和开发文档 比如这里我就在官方找到正确的id描述 然后我就把参数标签换过来

poj 1258 Agri-Net(最小生成树模板代码)

感觉用这题来当模板更适合。 题意就是给你邻接矩阵求最小生成树啦。~ prim代码:效率很高。172k...0ms。 #include<stdio.h>#include<algorithm>using namespace std;const int MaxN = 101;const int INF = 0x3f3f3f3f;int g[MaxN][MaxN];int n

【区块链 + 人才服务】可信教育区块链治理系统 | FISCO BCOS应用案例

伴随着区块链技术的不断完善,其在教育信息化中的应用也在持续发展。利用区块链数据共识、不可篡改的特性, 将与教育相关的数据要素在区块链上进行存证确权,在确保数据可信的前提下,促进教育的公平、透明、开放,为教育教学质量提升赋能,实现教育数据的安全共享、高等教育体系的智慧治理。 可信教育区块链治理系统的顶层治理架构由教育部、高校、企业、学生等多方角色共同参与建设、维护,支撑教育资源共享、教学质量评估、

搭建Kafka+zookeeper集群调度

前言 硬件环境 172.18.0.5        kafkazk1        Kafka+zookeeper                Kafka Broker集群 172.18.0.6        kafkazk2        Kafka+zookeeper                Kafka Broker集群 172.18.0.7        kafkazk3

透彻!驯服大型语言模型(LLMs)的五种方法,及具体方法选择思路

引言 随着时间的发展,大型语言模型不再停留在演示阶段而是逐步面向生产系统的应用,随着人们期望的不断增加,目标也发生了巨大的变化。在短短的几个月的时间里,人们对大模型的认识已经从对其zero-shot能力感到惊讶,转变为考虑改进模型质量、提高模型可用性。 「大语言模型(LLMs)其实就是利用高容量的模型架构(例如Transformer)对海量的、多种多样的数据分布进行建模得到,它包含了大量的先验