黑马程序员_银行业务调度系统

本文介绍了一个模拟银行业务调度系统的实现方案。系统包含6个业务窗口,支持VIP、快速及普通客户不同类型的服务。通过号码管理器生成不同类型的客户,并在对应窗口按顺序办理业务。文章提供了详细的代码实现。
---------- android培训java培训、java学习型技术博客,期待与您交流!----------

        银行业务调度系统,又是当年软通动力的一道面试题,话说张孝祥老师在边讲解边编写的过程中也出现了一些小差错,足见这个项目还是需要费些精力来完成的。不多说,做个总结。
        同样先来确定下,该银行业务调度系统模拟的具体需求。
1.银行内有6个业务窗口,1 - 4号窗口为普通窗口,5号窗口为快速窗口,6号窗口为VIP窗口。
2.有三种对应类型的客户:VIP客户,普通客户,快速客户(办理如交水电费、电话费之类业务的客户)。
3.异步随机生成各种类型的客户,生成各类型用户的概率比例为:
VIP客户 :普通客户 :快速客户  =  1 :6 :3。
4.客户办理业务所需时间有最大值和最小值,在该范围内随机设定每个VIP客户以及普通客户办理业务所需的时间,快速客户办理业务所需时间为最小值(提示:办理业务的过程可通过线程Sleep的方式模拟)。
5.各类型客户在其对应窗口按顺序依次办理业务。 
6.当VIP(6号)窗口和快速业务(5号)窗口没有客户等待办理业务的时候,这两个窗口可以处理普通客户的业务,而一旦有对应的客户等待办理业务的时候,则优先处理对应客户的业务。
7.随机生成客户时间间隔以及业务办理时间最大值和最小值自定,可以设置。
8.不要求实现GUI,只考虑系统逻辑实现,可通过Log方式展现程序运行结果。

        简略的说,就是有三种对应类型的客户:普通客户、快速客户和VIP客户。异步随机生成这三种类型的客户,各类型客户在其对应的服务窗口按顺序依次办理业务。但凡进过银行办业务的,都应该知道取号机的存在,前来办理业务的客户实际上就是由取号机产生的服务号码来表示的。所以本系统中需要一个号码管理器对象,让这个对象不断的生成号码,这也就等于随机生成了客户。由于有三类客户,每类客户的号码编排都是完全独立的,因此需要在本系统中创建三个号码管理器对象,各自管理一类客户的排队号码。这三个号码管理器对象统一由一个号码机器进行管理,这个号码机器在整个系统中始终只有一个,所以将其设计成单例。各个服务窗口通过号码管理器获取当前需要服务的号码。
        整个系统中各个类之间是怎样的关系,看下面的类图就比较清晰了。

        接下来看这些类的详细代码,由于张孝祥老师在编写过程中出现过一些小差错,所以事后我又重新梳理了一遍代码,并且加入了较为详尽的注释,所以就不再在代码外分析了。
        Constants类(其他类中用到的几个常量)
public class Constants {  
    public static int MAX_SERVICE_TIME = 10000; // 最长服务时间  
    public static int MIN_SERVICE_TIME = 1000;  // 最短服务时间  
  
    public static int COMMON_CUSTOMER_INTERVAL_TIME=1;  
}

        CustomerType类(客户类型,用到了枚举)
public enum CustomerType {  
    COMMON,EXPRESS,VIP;  
  
    @Override  
    public String toString() {  
        String name=null;  
        switch(this){  
        case COMMON:  
            name = "普通";  
            break;  
        case EXPRESS:  
            name = "快速";  
            break;  
        case VIP:  
            name = "VIP";  
            break;  
        }  
        return name;  
    }  
      
}

        NumberManager类(号码管理器)
public class NumberManager {  
    private int lastNumber = 1; //  上一个客户取走的业务号  
    private List<Integer> queueNumber = new ArrayList<Integer>();   // 待办业务队列  
      
    // 为来取号的客户生成新的业务号, 由于是多个线程对临界资源进行访问, 这个取号动作要加把锁  
    public synchronized Integer generateNewNumber(){
        queueNumber.add(lastNumber++);
        return lastNumber;
    }  
      
    // 提供给柜台用来取下一个办理业务的号, 由于是多个线程对临界资源进行访问, 这个取号动作要加把锁  
    public synchronized Integer fetchNInteger(){  
        if(queueNumber.size() > 0){  
            return queueNumber.remove(0);  
        }else{  
            return null;  
        }  
    }  
}

        NumberMachine类(号码机器即现实中的取号机)
public class NumberMachine {  
    // 单例对象设计模式  
    private static NumberMachine instance = null;  
    public static NumberMachine getInstance(){  
        if(instance == null){  
            instance = new NumberMachine();  
        }  
        return instance;  
    }  
    // 取号器中有三种业务号管理器  
    private NumberManager commonManager = new NumberManager();  // 普通业务  
    private NumberManager expressManager = new NumberManager(); // 快速业务  
    private NumberManager vipManager =new NumberManager();      // VIP业务  
    public NumberManager getCommonManager() {  
        return commonManager;  
    }  
    public NumberManager getExpressManager() {  
        return expressManager;  
    }  
    public NumberManager getVipManager() {  
        return vipManager;  
    }  
  
}

        ServericeWindow类(服务窗口)
public class ServiceWindow {  
    private CustomerType type = CustomerType.COMMON;  
    private int windowID = 1; // 窗口序号  
    public CustomerType getType() {  
        return type;  
    }  
    public void setType(CustomerType type) {  
        this.type = type;  
    }  
    public void setNumber(int number) {  
        this.windowID = number;  
    }  
      
    public void start(){  
        Executors.newSingleThreadExecutor().execute(new Runnable() {  
            public void run() {  
                // TODO Auto-generated method stub  
                while(true){  
                    switch(type){  
                    case COMMON:    //      普通客户窗口  
                        commonService();  
                        break;  
                    case EXPRESS:   // 快速客户窗口  
                        expressService();  
                        break;  
                    case VIP:  
                        vipService();  
                        break;  
                    }  
                }  
            }  
  
              
        });  
    }  
      
    private void commonService() {  
        String windowName = "第"+windowID+"号"+type+"窗口";  
        System.out.println(windowName+" 开始获取任务 !");  
        // 获取将要服务的客户业务号  
        Integer serviceNumber = NumberMachine.getInstance().getCommonManager().fetchNInteger();  
        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) {  
                // TODO Auto-generated catch block  
                e.printStackTrace();  
            }  
            System.out.println(windowName+"完成了为第"+serviceNumber+"个普通客户的服务,总共耗时 "+serviceTime/1000+" 秒!");  
        }else{  
            // 如果没有客户要办理业务, 就空闲1秒再去取下一个  
            System.out.println(windowName+"没有取到普通任务,空闲1秒");  
            try {  
                Thread.sleep(1000);  
            } catch (InterruptedException e) {  
                // TODO Auto-generated catch block  
                e.printStackTrace();  
            }  
        }  
    }  
      
    private void expressService(){  
        String windowName = "第"+windowID+"号"+type+"窗口";  
        System.out.println(windowName+" 开始获取任务 !");  
        // 获取将要服务的客户业务号  
        Integer serviceNumber = NumberMachine.getInstance().getExpressManager().fetchNInteger();  
        if(serviceNumber != null){  
            System.out.println(windowName+"开始为第"+serviceNumber+"个快速客户服务");  
            // 处理业务时间 就是MIN_SERVICE_TIME  
            try {  
                Thread.sleep(Constants.MIN_SERVICE_TIME);  
            } catch (InterruptedException e) {  
                // TODO Auto-generated catch block  
                e.printStackTrace();  
            }  
            System.out.println(windowName+"完成了为第"+serviceNumber+"个快速客户的服务,总共耗时 "+Constants.MIN_SERVICE_TIME/1000+" 秒");  
        }else{  
            // 如果没有客户要办理业务, 就会去获取普通客户业务号  
            System.out.println(windowName+"没有取到普通任务,尝试为普通客户服务");  
            commonService();  
        }  
    }  
      
    private void vipService(){  
        String windowName = "第"+windowID+"号"+type+"窗口";  
        System.out.println(windowName+" 开始获取任务 !");  
        // 获取将要服务的客户业务号  
        Integer serviceNumber = NumberMachine.getInstance().getVipManager().fetchNInteger();  
        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客户的服务");  
        }else{  
            // 如果没有客户要办理业务, 就会去获取普通客户业务号  
            System.out.println(windowName+"没有取到普通任务,尝试为普通客户服务");  
            commonService();  
        }  
    }  
}

        MainClass类(主函数,测试程序)
public class MainClass {  
    public static void main(String[] args) {  
        // 产生四个普通窗口  
        for(int i=1; i<=4; i++){  
            ServiceWindow commonWindow = new ServiceWindow();  
            commonWindow.setNumber(i);  
            commonWindow.setType(CustomerType.COMMON);  
            commonWindow.start();  
        }  
        // 产生一个快速窗口  
        ServiceWindow expressWindow = new ServiceWindow();  
        expressWindow.setNumber(1);  
        expressWindow.setType(CustomerType.EXPRESS);  
        expressWindow.start();  
        // 产生一个vip窗口  
        ServiceWindow vipWindow = new ServiceWindow();  
        vipWindow.setNumber(1);  
        vipWindow.setType(CustomerType.VIP);  
        vipWindow.start();  
          
        // 普通客户拿号  
        Executors.newScheduledThreadPool(1).scheduleAtFixedRate(  
                new Runnable() {  
                    public void run() {  
                        Integer serviceNumber = NumberMachine.getInstance().getCommonManager().generateNewNumber();  
                        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+" 号客户正在等待服务!");  
                    }  
                },   
                0,  
                Constants.COMMON_CUSTOMER_INTERVAL_TIME * 6,   
                TimeUnit.SECONDS  
                );  
    }  
  
}

        至此,整个银行业务调度系统便完成了。

---------- android培训java培训、java学习型技术博客,期待与您交流!----------
内容概要:本文是一篇关于使用RandLANet模型对SensatUrban数据集进行点云语义分割的实战教程,系统介绍了从环境搭建、数据准备、模型训练与测试到精度评估的完整流程。文章详细说明了在Ubuntu系统下配置TensorFlow 2.2、CUDA及cuDNN等深度学习环境的方法,并指导用户下载和预处理SensatUrban数据集。随后,逐步讲解RandLANet代码的获取与运行方式,包括训练、测试命令的执行与参数含义,以及如何监控训练过程中的关键指标。最后,教程涵盖测试结果分析、向官方平台提交结果、解读评估报告及可视化效果等内容,并针对常见问题提供解决方案。; 适合人群:具备一定深度学习基础,熟悉Python编程和深度学习框架,从事计算机视觉或三维点云相关研究的学生、研究人员及工程师;适合希望动手实践点云语义分割项目的初学者与进阶者。; 使用场景及目标:①掌握RandLANet网络结构及其在点云语义分割任务中的应用;②学会完整部署一个点云分割项目,包括数据处理、模型训练、测试与性能评估;③为参与相关竞赛或科研项目提供技术支撑。; 阅读建议:建议读者结合提供的代码链接和密码访问完整资料,在本地或云端环境中边操作边学习,重点关注数据格式要求与训练参数设置,遇到问题时参考“常见问题与解决技巧”部分及时排查。
内容概要:本文详细介绍了三相异步电机SVPWM-DTC(空间矢量脉宽调制-直接转矩控制)的Simulink仿真实现方法,结合DTC响应快与SVPWM谐波小的优点,构建高性能电机控制系统。文章系统阐述了控制原理,包括定子磁链观测、转矩与磁链误差滞环比较、扇区判断及电压矢量选择,并通过SVPWM技术生成固定频率PWM信号,提升系统稳态性能。同时提供了完整的Simulink建模流程,涵盖电机本体、磁链观测器、误差比较、矢量选择、SVPWM调制、逆变器驱动等模块的搭建与参数设置,给出了仿真调试要点与预期结果,如电流正弦性、转矩响应快、磁链轨迹趋圆等,并提出了模型优化与扩展方向,如改进观测器、自适应滞环、弱磁控制和转速闭环等。; 适合人群:电气工程、自动化及相关专业本科生、研究生,从事电机控制算法开发的工程师,具备一定MATLAB/Simulink和电机控制理论基础的技术人员。; 使用场景及目标:①掌握SVPWM-DTC控制策略的核心原理与实现方式;②在Simulink中独立完成三相异步电机高性能控制系统的建模与仿真;③通过仿真验证控制算法有效性,为实际工程应用提供设计依据。; 阅读建议:学习过程中应结合文中提供的电机参数和模块配置逐步搭建模型,重点关注磁链观测、矢量选择表和SVPWM调制的实现细节,仿真时注意滞环宽度与开关频率的调试,建议配合MATLAB官方工具箱文档进行参数校准与结果分析。
已经博主授权,源码转载自 https://pan.quark.cn/s/bf1e0d5b9490 本文重点阐述了Vue2.0多Tab切换组件的封装实践,详细说明了通过封装Tab切换组件达成多Tab切换功能,从而满足日常应用需求。 知识点1:Vue2.0多Tab切换组件的封装* 借助封装Tab切换组件,达成多Tab切换功能* 支持tab切换、tab定位、tab自动化仿React多Tab实现知识点2:TabItems组件的应用* 在index.vue文件中应用TabItems组件,借助name属性设定tab的标题* 通过:isContTab属性来设定tab的内容* 能够采用子组件作为tab的内容知识点3:TabItems组件的样式* 借助index.less文件来设定TabItems组件的样式* 设定tab的标题样式、背景色彩、边框样式等* 使用animation达成tab的切换动画知识点4:Vue2.0多Tab切换组件的构建* 借助运用Vue2.0框架,达成多Tab切换组件的封装* 使用Vue2.0的组件化理念,达成TabItems组件的封装* 通过运用Vue2.0的指令和绑定机制,达成tab的切换功能知识点5:Vue2.0多Tab切换组件的优势* 达成多Tab切换功能,满足日常应用需求* 支持tab切换、tab定位、tab自动化仿React多Tab实现* 能够满足多样的业务需求,具备良好的扩展性知识点6:Vue2.0多Tab切换组件的应用场景* 能够应用于多样的业务场景,例如:管理系统、电商平台、社交媒体等* 能够满足不同的业务需求,例如:多Tab切换、数据展示、交互式操作等* 能够与其它Vue2.0组件结合运用,达成复杂的业务逻辑Vue2.0多Tab切换组件的封装实例提供了...
代码下载地址: https://pan.quark.cn/s/41cd695ddf65 `htmldiff` 是一个以 Ruby 语言为基础构建的库,其主要功能是在 HTML 文档中展示文本之间的差异。 该库的一个显著特点在于它不仅能够识别出不同之处,还会借助 HTML 标签来呈现这些差异,从而让用户能够直观地观察到文本的变化情况。 这种特性使得 `htmldiff` 在版本控制、文档对比或任何需要展示文本变动场景的应用中显得尤为有用。 `htmldiff` 的核心作用是对比两个字符串,并生成一个 HTML 输出结果,这个结果会明确地指出哪些部分被添加、哪些部分被删除以及哪些部分被修改。 此外,通过运用 CSS,用户可以进一步调整差异展示的样式,使其与项目或网站的现有设计风格相协调。 在使用 `htmldiff` 之前,需要先完成该库的安装。 如果项目已经配置了 Ruby 环境和 Gemfile,可以在 Gemfile 文件中添加 `gem htmldiff` 语句,随后执行 `bundle install` 命令进行安装。 如果没有 Gemfile 文件,也可以直接采用 `gem install htmldiff` 命令来进行全局安装。 在编程实现时,可以通过调用 `Htmldiff.diff` 方法来对比两个字符串,并获取相应的 HTML 输出。 例如:```rubyrequire htmldiffstr1 = "这是一个示例文本。 "str2 = "这是一个示例文本,现在有更多内容。 "diff_html = Htmldiff.diff(str1, str2)puts diff_html```上述代码将会输出两个字符串之间的差异,其中新增的内容会被 `<ins>` 标签所包围,而...
源码地址: https://pan.quark.cn/s/4b03c5611266 依据所提供的资料,可以判定这份资料是关于《电子技术基础模拟部分》第五版教科书第七章节的习题解析,由湖南人文科技学院通信与控制工程系的田汉平教师提供。 尽管具体内容未予展示,但能够围绕模拟电子技术的基础理论、第七章节所涉及的核心概念以及潜在的习题种类等方面来展开相关知识点的阐述。 ### 模拟电子技术概述模拟电子技术是电子工程学科中的一个关键领域,主要探讨模拟信号的产生、转换、传输和处理等议题。 模拟信号是指时间与幅度上均呈现连续变化的电信号。 模拟电路的设计与剖析是模拟电子技术的核心,它涵盖了放大器、振荡器、滤波器等电路的设计原理及其应用。 ### 第七章核心知识点猜测#### 1. 放大电路分析与设计- **基本放大电路**:共射极、共基极和共集电极放大电路的特性及其应用场景。 - **多级放大电路**:掌握如何将多个放大电路串联,以提升增益或优化频率响应。 - **差分放大电路**:用于抑制共模信号,放大差模信号,是精密仪器和测量设备中的关键构成部分。 #### 2. 反馈电路与稳定性- **反馈的基本概念**:正反馈与负反馈的区分,以及它们在电路中的应用场景。 - **深度负反馈**:解析深度负反馈状态下的放大器性能改进,包括增益稳定性和带宽的拓宽。 - **振荡电路**:理解LC振荡器、RC振荡器的工作机制及应用领域。 #### 3. 功率放大器- **A类、B类、AB类功率放大器**:熟练掌握不同类型功率放大器的特性、效率及其适用环境。 - **热效应与保护措施**:讨论在功率放大器设计过程中需要关注的散热问题及相应的防护措施。 #### 4. 集成运算放大器的应用- **理想运放模型**:熟...
下载方式:https://pan.quark.cn/s/71eedd1f6247 ADC0832被视作一种常见的8位模数转换器(Analog-to-Digital Converter),在电子设计领域中具有广泛的应用。 该种芯片能够将连续的模拟信号转变为离散的数字信号,从而方便微处理器或其他数字系统进行进一步处理。 在Proteus环境中,通过仿真ADC0832的设计,有助于我们深入理解并验证基于模拟输入信号的数字化操作。 1. **ADC0832的核心构造与运作机制** ADC0832属于一种8位逐次逼近型模数转换器,其内部集成了一个比较器、一个计数器以及一个寄存器。 当输入模拟电压被施加至输入端时,内部比较器会持续调整数字代码,直至输出端的电压与输入电压达到一致。 这一过程即为逐次逼近,其最终输出的数字代码代表了输入模拟电压的等效数值。 2. **模数转换流程** - **采样与保持**:在模数转换开始之前,模拟信号必须被采样并保持一段时期,以确保在转换期间输入电压保持稳定。 - **启动转换**:通过向芯片发送启动信号(例如START或CONVST引脚的高电平脉冲),转换过程得以启动。 - **逐位比较**:内部比较器从最高位开始,依次进行比较,并根据比较结果更新数字代码。 - **转换终止**:当所有8位都完成比较后,转换过程结束,并通过中断标志或状态引脚向系统发出通知。 - **数据读取**:借助SPI、I2C或并行接口,从ADC的输出寄存器中获取转换结果。 3. **Proteus仿真平台** Proteus是一款功能强大的电子电路仿真软件,能够支持多种微控制器及外围设备的仿真,其中包括ADC0832。 在Proteus中,用户可以构建ADC0832与微控制器的...
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值