`

基于优先级队列的事件驱动模拟(Event-Driven Simulation)

阅读更多
1. 引言:

    利用模拟来研究一个不少于两个(n>=2)出纳窗口的银行中每个客户到达(Arrival)银行和离开(Departue)银行的情况。
一般每个出纳窗口在某一时刻值能接待一位客户,在客户很多时需在每个窗口顺序排队。如果客户到达银行时某个窗口
正空闲,客户就可以立刻到该窗口办理业务;如果所有窗口都不空闲,则客户要排在最短的队列后面。我们需要通过计算
每位客户的平均等待时间和每位出纳员的繁忙度(时间百分比)的出服务的效率。

2. 设计过程:

(1) 模拟设计
      
1> 在运行过程中,模拟将跟踪各个客户的到达和离去。

       设客户到达时间为time(Arrival), 离去时间为time(Depature),窗口的服务时间ServiceTime,则有:

       time(Depature) = time(Arrival) + ServiceTime

       考虑另一种情况,设两个出纳窗口都不空闲。客户必须排在某一个队列中等待服务。模拟可以让客户选择出纳窗口
,并在窗口更新"下次空闲时间"标牌。银行模拟的关键部分是两个客户事件:到达事件Arrival和离开事件Depature。假设
客户的等待时间为WaitTime,则有:

       time(Depature) = time(Arrival) + WaitTime + ServiceTime

       当一位客户到达银行时,就会产生一个到达事件Arrival。
       当一位客户检查并选定出纳窗口后,就将定义一个离开事件来描述该客户什么时候离开。

事件类代码如下:

package boke.eventdriven;

/**
 * TODO 事件类型
 * 
 * @author 毛正吉
 * 
 */
public enum EventType {
	// 到达和离开事件
	Arrival, Departure

}

--------------------------------

package boke.eventdriven;

/**
 * TODO 事件类
 * 
 * @since jdk1.5及其以上
 * @author 毛正吉
 * @version 1.0
 * @date 2010.05.24
 * 
 */
public class Event {
	private int time; // 客户到达 | 离开时间
	EventType eType; // 事件类型
	private int customerID; // 客户编号,编号为1,2,3...
	private int tellerID; // 出纳窗口编号,编号为1,2,3,...
	private int waitTime; // 客户等待时间
	private int serviceTime; // 客户服务时间

	/**
	 * 构造方法
	 * 
	 * @param _time
	 * @param _eType
	 * @param _customerID
	 * @param _tellerID
	 * @param _waitTime
	 * @param _serviceTime
	 */
	public Event(int _time, EventType _eType, int _customerID, int _tellerID,
			int _waitTime, int _serviceTime) {
		time = _time;
		eType = _eType;
		customerID = _customerID;
		tellerID = _tellerID;
		waitTime = _waitTime;
		serviceTime = _serviceTime;
	}

	/** ************ getter和setter方法 ************* */
	public int getTime() {
		return time;
	}

	public void setTime(int time) {
		this.time = time;
	}

	public EventType getEType() {
		return eType;
	}

	public void setEType(EventType type) {
		eType = type;
	}

	public int getCustomerID() {
		return customerID;
	}

	public void setCustomerID(int customerID) {
		this.customerID = customerID;
	}

	public int getTellerID() {
		return tellerID;
	}

	public void setTellerID(int tellerID) {
		this.tellerID = tellerID;
	}

	public int getWaitTime() {
		return waitTime;
	}

	public void setWaitTime(int waitTime) {
		this.waitTime = waitTime;
	}

	public int getServiceTime() {
		return serviceTime;
	}

	public void setServiceTime(int serviceTime) {
		this.serviceTime = serviceTime;
	}

}

2> 在模拟中,还要统计每一出纳窗口信息:
   处理过的客户总数;
   客户的总等待时间;
   总的服务时间;
   什么时候出纳窗口空闲
------------------------------

package boke.eventdriven;

/**
 * TODO 出纳窗口类
 * 
 * @since jdk1.5及其以上
 * @author 毛正吉
 * @version 1.0
 * @date 2010.05.24
 * 
 */
public class TellerStatus {
	public int finishService; // 什么时候出纳窗口空闲
	public int totalCustomerCount; // 服务过的客户总数
	public int totalCustomerWait; // 客户总的等待时间
	public int totalService; // 总的服务时间
}


3> 模拟为每一位客户生成一个到达事件和离开事件。所有的事件都以时间为标记,并放入一个优先级队列中。
在优先级队列中优先权最高的事件是时间标记最早的事件。这样一种表结构使得我们能够以一个事件递增的顺序
从队列中移去客户的到达和离开事件。
----------------------------------------

package boke.eventdriven;

/**
 * TODO 优先级队列
 * 
 * @since jdk1.5及其以上
 * @author 毛正吉
 * @version 1.0
 * @date 2010.05.24
 * 
 */
public class PQueue {
	/**
	 * 
	 * @param args
	 * @throws PQueueException
	 */
	public static void main(String[] args) throws PQueueException {
		PQueue pq = new PQueue();
		for (int i = 0; i <= 10; i++) {
			Event e = new Event(i, EventType.Arrival, i, i, i, i);
			pq.enPQueue(e);
		}

		System.out.println(pq.length());

		while (!pq.isEmpty()) {
			Event e = pq.delPQueue();
			System.out.println(e.getEType() + " " + e.getTime());
		}

		System.out.println(pq.length());
	}

	private static final int MAX_EVENT_SIZE = 1000; // 队列最大长度
	private Event[] events; // 队列元素
	private int count; // 队列长度

	public PQueue() {
		events = new Event[MAX_EVENT_SIZE];
		count = 0;
	}

	/**
	 * 入队
	 * 
	 * @param evt
	 * @throws PQueueException
	 */
	public void enPQueue(Event evt) throws PQueueException {
		if (!isFull()) {
			events[count] = evt;
			count++;
		} else {
			throw new PQueueException("PQueue is full!");
		}
	}

	/**
	 * 出队:标记时间最短的事件有最大的优先权
	 * 
	 * @return
	 * @throws PQueueException
	 */
	public Event delPQueue() throws PQueueException {
		if (!isEmpty()) {
			int min = events[0].getTime();
			int minindex = 0;

			for (int i = 1; i < count; i++) {
				if (events[i].getTime() < min) {
					min = events[i].getTime();
					minindex = i;
				}
			}

			Event et = events[minindex];
			events[minindex] = events[count - 1];
			count--;
			return et;
		} else {
			throw new PQueueException("PQueue is empty!");
		}
	}

	/**
	 * 队列置空
	 */
	public void makeEmpey() {
		count = 0;
	}

	/**
	 * 队列是否为空
	 */
	public boolean isEmpty() {
		return count == 0;
	}

	/**
	 * 队列是否为满
	 */
	public boolean isFull() {
		return count == MAX_EVENT_SIZE;
	}

	/**
	 * 队列的长度
	 */
	public int length() {
		return count;
	}
}

-------------------------------
package boke.eventdriven;

/**
 * TODO 优先级队列异常类
 * 
 * @since jdk1.5及其以上
 * @author 毛正吉
 * @version 1.0
 * @date 2010.05.24
 * 
 */
public class PQueueException extends Exception {
	public PQueueException() {
		super("PQueueException");
	}

	public PQueueException(String msg) {
		super(msg);
	}
}
---------------------------------

2. 模拟建立、运行

1> 下一次客户何时到达?当前客户的服务时间有多长?

    银行模拟将使用一个随机数发生器来生成。该发生器假定在一定的取值范围内取任何值的概率都是相等的。
如果当前的到达时间发生在T分钟,下一次到达时间将随机地发生在T+arriveLow 到 T+arriveHigh的范围内,客户
需要的服务时间则在serviceLow到serviceHigh的范围内。
 
    看函数nextArrivalTime() 和 getServiceTime()

2> 在模拟过程中,客户要得到空闲时间最小的出纳窗口?

    看函数nextAvailableTeller()

3> 模拟运行

    看函数runSimulation()

-----------------------------
package boke.eventdriven;

import java.util.Random;

/**
 * TODO 模拟 类
 * 
 * @since jdk1.5及其以上
 * @author 毛正吉
 * @version 1.0
 * @date 2010.05.24
 * 
 */
public class Simulation {
	private int simulationLength; // 模拟的时间长度
	private int numTellers; // 出纳窗口个数
	private int nextCustomer; // 下一位客户的编号
	private int arrivalLow, arrivalHigh; // 下一次到达的时间范围
	private int serviceLow, serviceHigh; // 服务的时间范围
	private TellerStatus[] tstat = new TellerStatus[11]; // 最多10个出纳窗口
	private PQueue pq = new PQueue(); // 优先级队列
	private Random rnd = new Random(); // 用于到达和服务的时间

	/**
	 * 构造方法
	 * 
	 * @param simulationLength
	 * @param numTellers
	 * @param arrivalLow
	 * @param arrivalHigh
	 * @param serviceLow
	 * @param serviceHigh
	 * @throws PQueueException
	 */
	public Simulation(int simulationLength, int numTellers, int arrivalLow,
			int arrivalHigh, int serviceLow, int serviceHigh)
			throws PQueueException {
		Event firstEvent = new Event(0, EventType.Arrival, 1, 0, 0, 0); // 第一个到达事件
		for (int i = 0; i <= 10; i++) { // 初始化出纳窗口信息
			tstat[i] = new TellerStatus();
			tstat[i].finishService = 0;
			tstat[i].totalService = 0;
			tstat[i].totalCustomerWait = 0;
			tstat[i].totalCustomerCount = 0;
		}
		this.nextCustomer = 1; // 下一位客户的编号从1开始

		this.simulationLength = simulationLength; // 输入的模拟时间长度(以分钟计算)
		this.numTellers = numTellers; // 输入的出纳窗口个数
		this.arrivalLow = arrivalLow; // 输入下一次到达的时间范围
		this.arrivalHigh = arrivalHigh;
		this.serviceLow = serviceLow; // 输入服务的时间范围
		this.serviceHigh = serviceHigh;
		pq.enPQueue(firstEvent);
	}

	/**
	 * 确定一下次到达的随机时间
	 * 
	 * @return
	 */
	private int nextArrivalTime() {
		return arrivalLow + rnd.nextInt((arrivalHigh - arrivalLow + 1));
	}

	/**
	 * 确定客户服务的随机时间
	 * 
	 * @return
	 */
	private int getServiceTime() {
		return serviceLow + rnd.nextInt((serviceHigh - serviceLow + 1));
	}

	/**
	 * 产生一个可用的出纳窗口
	 * 
	 * @return
	 */
	private int nextAvailableTeller() {
		int minfinish = simulationLength; // 初始时假定所有的出纳窗口在下班时关闭
		int minfinishindex = rnd.nextInt(numTellers) + 1; // 给下班前到达但在下班后得到服务的客户提供一个随机的出纳窗口编号

		for (int i = 1; i <= numTellers; i++) { // 找一个可用的窗口
			if (tstat[i].finishService < minfinish) { // 寻找窗口空闲时间最小者
				minfinish = tstat[i].finishService;
				minfinishindex = i;
			}
		}
		return minfinishindex; // 返回窗口空闲时间最小者的窗口号码
	}

	/**
	 * 模拟的主函数
	 * 
	 * @throws PQueueException
	 */
	public void runSimulation() throws PQueueException {
		while (!pq.isEmpty()) {
			// 出队
			Event e = pq.delPQueue();

			/* ###################################### */
			System.out.println("客户" + e.getCustomerID() + "在时间" + e.getTime()
					+ "分" + e.getEType());
			/* ###################################### */
			System.out.println("######################################");
			System.out.println("CustomerID = " + e.getCustomerID());
			System.out.println("TellerID = " + e.getTellerID());
			System.out.println("WaitTime = " + e.getWaitTime());
			System.out.println("ServiceTime = " + e.getServiceTime());
			System.out.println("######################################");

			// 整个模拟的时间长度
			simulationLength = (e.getTime() <= simulationLength) ? simulationLength
					: e.getTime();

			// 1.到达 (Arrival)
			// (1)当前到达的客户负责生成下一个到达事件
			int nextTime = e.getTime() + nextArrivalTime();// 计算下次到达时间

			if (nextTime > simulationLength) { // 超出模拟时间长度,什么也不做
				continue;
			} else { // 产生下一个客户到达事件,并放入优先级队列
				nextCustomer++;
				Event newEvent = new Event(nextTime, EventType.Arrival,
						nextCustomer, 0, 0, 0);
				pq.enPQueue(newEvent);

				/* ###################################### */
				System.out.print("客户" + e.getCustomerID() + "为" + "客户"
						+ newEvent.getCustomerID() + "在第" + newEvent.getTime()
						+ "分钟到达产生一个到达事件,");
			}

			int serviceTime = getServiceTime(); // 客户所需服务时间
			int tellerID = nextAvailableTeller(); // 可为客户服务的出纳窗口
			int waitTime = 0; // 客户等待时间

			if (tstat[tellerID].finishService == 0) {
				tstat[tellerID].finishService = e.getTime();
			} else {
				waitTime = tstat[tellerID].finishService - e.getTime();
			}

			// (2)更新出纳窗口的各项信息
			tstat[tellerID].finishService += serviceTime;
			tstat[tellerID].totalCustomerCount++;
			tstat[tellerID].totalCustomerWait += waitTime;
			tstat[tellerID].totalService += serviceTime;

			// (3) 定义离开事件,并把它放入优先级队列中
			Event newEvent = new Event(tstat[tellerID].finishService,
					EventType.Departure, e.getCustomerID(), tellerID, waitTime,
					serviceTime);
			pq.enPQueue(newEvent);

			// 2.离开
			tellerID = e.getTellerID();
			if (e.getTime() == tstat[tellerID].finishService) {
				tstat[tellerID].finishService = 0;
			}

			/* ###################################### */
			System.out.println("并为自己在第" + newEvent.getTime() + "分钟"
					+ newEvent.getEType() + "产生一个离开事件");

			/* ###################################### */
			System.out.println("######################################");
			System.out.println("CustomerID = " + newEvent.getCustomerID());
			System.out.println("TellerID = " + newEvent.getTellerID());
			System.out.println("WaitTime = " + newEvent.getWaitTime());
			System.out.println("ServiceTime = " + newEvent.getServiceTime());
			System.out.println("######################################");
		}
	}

	/**
	 * 模拟输出函数
	 */
	public void printSimulationResults() {
		int cumCustomers = 0; // 客户总数
		int cumWait = 0; // 客户总的等待时间
		for (int i = 1; i <= numTellers; i++) {
			cumCustomers += tstat[i].totalCustomerCount;
			cumWait += tstat[i].totalCustomerWait;
		}

		System.out.println();
		System.out
				.println("**************************** Simulation Summary ******************************");
		System.out.println("模拟时间长度 : " + this.simulationLength + "分钟");
		System.out.println("客户总数 :" + cumCustomers);

		// 计算平均等待时间
		float avgCustWait = (float) ((cumWait) / cumCustomers + 0.5);
		System.out.println("平均等待时间 : " + avgCustWait + "分钟");

		// 打印各个出纳窗口信息
		for (int i = 1; i <= numTellers; i++) {
			// 窗口平均服务时间
			float tellerWork = tstat[i].totalService / simulationLength;
			// 服务时间百分比
			float tellerWorkPercent = (float) (tellerWork * 100.0 + 0.5);
			System.out.println("出纳窗口" + i + ": 服务时间百分比 : " + tellerWorkPercent);
		}

	}
}
3. 模拟的输出:

package boke.eventdriven;

/**
 * TODO 模拟 类客户端应用
 * 
 * @since jdk1.5及其以上
 * @author 毛正吉
 * @version 1.0
 * @date 2010.05.24
 * 
 */
public class SimulationApp {

	/**
	 * @param args
	 * @throws PQueueException
	 */
	public static void main(String[] args) throws PQueueException {
		// 模拟参数给出如下
		int simulationLength = 30; // 输入的模拟时间长度(以分钟计算)
		int numTellers = 2; // 输入的出纳窗口个数
		int arrivalLow = 6; // 输入下一次到达的时间范围
		int arrivalHigh = 10;
		int serviceLow = 18; // 输入服务的时间范围
		int serviceHigh = 20;
		Simulation s = new Simulation(simulationLength, numTellers, arrivalLow,
				arrivalHigh, serviceLow, serviceHigh);

		// 运行模拟程序
		s.runSimulation();

		// 输出结果
		s.printSimulationResults();
	}

}
-----------------------------------------
客户1在时间0分Arrival
######################################
CustomerID = 1
TellerID = 0
WaitTime = 0
ServiceTime = 0
######################################
客户1为客户2在第7分钟到达产生一个到达事件,并为自己在第18分钟Departure产生一个离开事件
######################################
CustomerID = 1
TellerID = 1
WaitTime = 0
ServiceTime = 18
######################################
客户2在时间7分Arrival
######################################
CustomerID = 2
TellerID = 0
WaitTime = 0
ServiceTime = 0
######################################
客户2为客户3在第14分钟到达产生一个到达事件,并为自己在第25分钟Departure产生一个离开事件
######################################
CustomerID = 2
TellerID = 2
WaitTime = 0
ServiceTime = 18
######################################
客户3在时间14分Arrival
######################################
CustomerID = 3
TellerID = 0
WaitTime = 0
ServiceTime = 0
######################################
客户3为客户4在第23分钟到达产生一个到达事件,并为自己在第38分钟Departure产生一个离开事件
######################################
CustomerID = 3
TellerID = 1
WaitTime = 4
ServiceTime = 20
######################################
客户1在时间18分Departure
######################################
CustomerID = 1
TellerID = 1
WaitTime = 0
ServiceTime = 18
######################################
客户4在时间23分Arrival
######################################
CustomerID = 4
TellerID = 0
WaitTime = 0
ServiceTime = 0
######################################
客户2在时间25分Departure
######################################
CustomerID = 2
TellerID = 2
WaitTime = 0
ServiceTime = 18
######################################
客户3在时间38分Departure
######################################
CustomerID = 3
TellerID = 1
WaitTime = 4
ServiceTime = 20
######################################
客户4在时间45分Departure
######################################
CustomerID = 4
TellerID = 2
WaitTime = 0
ServiceTime = 22
######################################

**************************** Simulation Summary ******************************
模拟时间长度 : 45分钟
客户总数 :4
平均等待时间 : 2.5分钟
出纳窗口1: 服务时间百分比 : 0.5
出纳窗口2: 服务时间百分比 : 0.5



      
分享到:
评论

相关推荐

    Designing Event-Driven Systems

    Designing Event-Driven Systems_Concepts and Patterns for Streaming Services with Apache Kakfa, published by OReilly

    C语言 实现 Socket 范例,包括event-driven

    4. **Event-Driven Socket**:事件驱动编程是一种高效处理并发I/O的方式,它利用事件循环机制来处理多个连接请求。在C语言中,可以使用第三方库如libevent或libev来实现。事件驱动Socket的基本思想是,当某个事件...

    Formalization and Verification of Event-driven Process chain

    Formalization and Verification of Event-driven Process chain

    Event-Driven Programming:

    教程强调了几个核心概念,比如事件队列、处理器设计模式(Handler Design Pattern)、无头处理器模式(Headless Handlers Pattern)、扩展处理器模式(Extended Handlers Pattern)、事件对象(Event Objects)、...

    基于java的开发源码-Message-Driven Bean EJB实例源代码.zip

    基于java的开发源码-Message-Driven Bean EJB实例源代码.zip 基于java的开发源码-Message-Driven Bean EJB实例源代码.zip 基于java的开发源码-Message-Driven Bean EJB实例源代码.zip 基于java的开发源码-Message-...

    Event-Driven Architecture Overview

    事件驱动架构(Event-Driven Architecture, EDA)是一种软件架构模式,它强调以事件为中心的设计,使得各个组件或服务通过异步通信来响应事件。事件可以是业务流程中发生的任何值得注意的事情,既可以是系统内部产生...

    Object-Oriented.Programming.Languages.And.Event-Driven.Programming.epub )

    After a brief second chapter on event-driven programming (EDP), subsequent chapters are built around case studies in each of the languages Smalltalk, C++, Java, C#, and Python. Included in each case ...

    event-driven_io-源码.rar

    在计算机编程领域,事件驱动I/O(Event Driven Input/Output)是一种高效且灵活的处理并发请求的编程模型,它广泛应用于网络服务、图形用户界面(GUI)以及现代微服务架构中。本资料"event-driven_io-源码.rar"提供...

    事件驱动的渗透测试扫描器_Event-driven_pentest_scanner_nacs.zip

    事件驱动的渗透测试扫描器_Event-driven_pentest_scanner_nacs

    VMware7.0虚拟机硬盘无法编辑,无法连接到Profile-Driven Storage Service

    在使用VMware vSphere 7.0时,可能会遇到虚拟机硬盘无法编辑或者无法连接到Profile-Driven Storage Service的问题。这种情况通常与存储配置、vSphere Client连接问题、虚拟机设置或者服务状态有关。以下是一些可能...

    论文阅读笔记Deep learning for event-driven stock prediction1

    该论文提出了一种基于深度学习的事件驱动股票预测方法,使用了 Novel Neural Tensor Network 来训练 event embeddings,并使用 CNN 模型来预测股市波动。 知识点: 1. 事件驱动股票预测:使用深度学习方法来预测...

    AN_Event-Driven_Arduino

    ### Event-Driven Arduino编程知识点详解 #### 一、引言 本文档主要介绍如何通过现代状态机技术将事件驱动的编程范式应用于Arduino图形化软件开发。具体来说,您将学习如何利用开源QP™/C++主动对象框架构建响应...

    拦截器与冲突解决

    然而,在使用`&lt;mvc:annotation-driven /&gt;`元素时,有时会出现与自定义拦截器的冲突问题。这个问题通常出现在当我们试图同时配置基于注解的控制器处理和自定义拦截器时,Spring可能无法正确地处理这些组件的执行顺序...

    测试驱动开发Test-Driven+Development+By+Example(中英文)

    测试驱动开发(Test-Driven Development,简称TDD)是一种软件开发方法,强调在编写实际代码之前先编写测试用例。这种做法旨在提高代码质量、可维护性和减少缺陷。《Test-Driven Development By Example》是一本由...

    VMware vSphere 磁盘无法增加容量,报错:无法连接到Profile-Driven Storage Service

    VMware vSphere 磁盘无法增加容量,报错:无法连接到Profile-Driven Storage Service

    1 - Basic Event-Driven User Interface_interface_labview_

    本文将围绕"Basic Event-Driven User Interface"这一主题,深入探讨LabVIEW中的事件驱动编程机制及其应用。 首先,理解事件驱动编程的核心概念是关键。事件驱动编程是一种编程范式,它依赖于事件的发生来触发程序的...

Global site tag (gtag.js) - Google Analytics