java 实现排队_实验排队功能实现(JAVA)

1.功能要求

实验室有固定台数的设备供学生通过网络连接进行实验,一台设备只能同时被一个用户使用,一个用户只能占用一台设备。

下面是一个功能的简图:

20180111004532073004.png

2.实现方案

2.1 初始化

20180111004532087653.png

在项目启动之后,开始进行实验设备排队功能的初始化,需要初始化的有:

a,新建用于存放设备的队列,并从数据库中查出所有可正常使用的设备放入队列中;

b,新建一个用于排队的线程池,后面会说明用途;

c,新建一个用于存放排队用户的队列。

2.2 流程实现

20180111004532089607.png

Thread :当前的用户的请求线程;waitUsers:存放排队用户线程的队列;Exec:排队的线程池;threadA:在线程池中开启的排队线程;Equipment:存放设备的线程

3.具体实现

3.1.队列初始化

1 /**

2 * 初始化队列及线程池3 *@authoryangc4 *5 */

6 public classEquipmentQueue {7 //设备队列

8 public static BlockingQueueequipment;9 //请求队列

10 public static BlockingQueuewaitUsers;11 //线程池

12 public staticExecutorService exec;13

14 /**

15 * 初始化设备、请求队列及线程池16 */

17 public voidinitEquipmentAndUsersQueue(){18 exec =Executors.newCachedThreadPool();19 equipment=new LinkedBlockingQueue();20 //将空闲的设备放入设备队列中

21 setFreeDevices(exec);22 waitUsers=new LinkedBlockingQueue();23 }24

25 /**

26 * 将空闲的设备放入设备队列中27 *@paramexec28 */

29 private voidsetFreeDevices(ExecutorService exec) {30 //获取可用的设备

31 List equipments=getFreeEquipment();32 for (int i = 0; i < equipments.size(); i++) {33 Record dc=equipments.get(i);34 Equipment de=new Equipment(dc.getInt("id"),dc.getStr("quip_no"),dc.getStr("name"));35 try{36 equipment.put(de);37 } catch(InterruptedException e) {38 e.printStackTrace();39 }40 }41 }42

43 /**

44 * 获取可用的设备(从数据库中查出可用的设备)45 *@return

46 */

47 public ListgetFreeEquipment(){48 returnQuipPartsManager.manager.getFreeEquipment();49 }50 }

3.2.过滤实验请求

当用户的实验请求进入时,首先要判断用户在数据库中是否处于未退出的情况,如果处于未退出的状态,将状态改为已结束,然后重新进行排队。

每次请求实验时,会在数据库中保存一条排队数据,记录排队的用户、状态、使用时间等等信息。WaitUser是实验请求对应的类,主要字段有:Thread(存放实验请求的线程对象)、Session(实验请求对应的用户session)、Test(数据库中排队数据对应的类)。当请求进入后会在过滤器中将用户的请求对象放入到用请求队列中。

1 //判断当前的用户是否有未退出的实验并进行处理

2 TestManager.manager.setUserTestInfomation(session);3 //获取当前的线程

4 Thread thread=Thread.currentThread();5 //将当前用户为等待

6 Test test=TestManager.manager.SetUserTestStateForWait(session);7 //创建当前的用户请求对象

8 WaitUser waitUser=newWaitUser();9 waitUser.setThread(thread);10 waitUser.setSession(session);11 waitUser.setTest(test);12 //将当前用户请求对象放入队列中

13 EquipmentQueue.waitUsers.add(waitUser);

3.3.执行排队线程并挂起当前线程

在线程池中分配一个线程给当前的请求,并运行此线程,然后将请求线程挂起。

1 //在线程池中给当前的用户请求分配线程,运行等待分配设备

2 EquipmentQueue.exec.execute(waitUser);3 //暂停当前的用户请求,当whetherWait等于2时,说明设备绑定已经完成,无需将当前线程挂起

4 synchronized(thread){5 if(waitUser.getWhetherWait()!=2){6 thread.wait();7 }8 }

开始排队即运行WaitUser中的experiment方法,先从设备队列中获取一个设备,如果没有设备,当前线程将会进入堵塞状态,直到队列中放入设备;如果有设备,从请求队列中取出一个请求对象,设置请求与设备绑定。

335b83df261c422459d4afc29ba290e5.png

d1b641f023dd079c9e4a800b96607d9d.gif

1 public class WaitUser implementsRunnable{2 //当前请求的线程对象

3 privateThread thread;4 //当前用户的session对象

5 privateHttpSession session;6 //用于判断线程是否进入wait状态

7 private int whetherWait=0;8 //用户的实验对象

9 privateTest test;10

11 @Override12 public voidrun() {13 //当线程未中断时

14 while(!Thread.interrupted()){15 experiment();16 }17 }18

19 /**

20 * 将实验信息存入数据库,用户信息从session获取,将使用的设备从队列中删除,将设备对象存入session21 */

22 public voidexperiment(){23 try{24 //取出一个设备

25 Equipment equipment=EquipmentQueue.equipment.take();26 EquipmentQueue.equipment.remove(equipment);27 WaitUser waitUser=EquipmentQueue.waitUsers.take();28 EquipmentQueue.waitUsers.remove(waitUser);29 //将设备与用户绑定,状态设置为试验中

30 TestManager.manager.bindUserAndEquipment(equipment,waitUser);31 } catch(InterruptedException e) {32 System.err.println("---" +e.getMessage());33 }34 }35

36

37 publicTest getTest() {38 returntest;39 }40

41 public voidsetTest(Test test) {42 this.test =test;43 }44

45 public intgetWhetherWait() {46 returnwhetherWait;47 }48

49 public void setWhetherWait(intwhetherWait) {50 this.whetherWait =whetherWait;51 }52

53 publicHttpSession getSession() {54 returnsession;55 }56

57 public voidsetSession(HttpSession session) {58 this.session =session;59 }60

61 publicThread getThread() {62 returnthread;63 }64

65 public voidsetThread(Thread thread) {66 this.thread =thread;67 }68

69 }

WaitUser

3.4.释放请求线程

当设备绑定成功后,即可释放请求线程,这里有一个需要注意的问题,挂起请求线程与释放请求线程的先后关系(确保不会出现先释放后挂起的情况),把释放与挂起线程放到用显式锁修饰的代码块中,确保同时只会执行一处。当释放锁之后将状态WhetherWait的值设为2,标记此请求已经与设备绑定,不需要挂起。

1 Thread thread=waitUser.getThread();2 synchronized(thread) {3 waitUser.setWhetherWait(2);4 thread.notify();5 }

4.补充

4.1.排队时,给予客户端的反馈

每次请求实验会存入一条排队数据到数据库中,当完成设备绑定后会将排队数据的状态值设置为“正在使用”,可以在客户端执行一个定时任务,定时从数据库查询处于“排队中”的请求数量,这样就可以在客户端实时显示“排队中,您前面还有10位用户正在等待...”的效果。

4.2.用户进入实验后,长时间暂用而不使用的情况处理

当用户进入实验后,为防止用户长时间的占用实验设备(并没有在使用),需要执行一个定时任务,每间隔一段时间,弹窗确认用户是否在使用,如果用户没有回应,则视为用户已经离开,将用户与设备解绑并退出实验。

4.3.用户强行退出的处理(关闭页面/管理浏览器/关闭电脑...强行退出实验的情况)

当用户强行退出时,在数据库中用户与设备依然处于绑定状态。当用户进入实验后,需要在客户端执行一个定时任务,实时的更新用户的最新实验时间。然后在服务端运行一个定时任务,实时检查数据库中的排队数据,判断状态为“正在使用”的数据中是否有最新的实验时间与当前时间的差值是否大于客户端定时任务的间隔时间(考虑到时间的更新会有一定的延迟,可以适当留些余量)的数据,如果大于则设备与用户解绑。

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Java本身并不提供队列或排队功能,但可以使用Java中的多线程和线程池来实现任务排队。具体实现方法如下: 1. 创建一个任务队列,用来存储需要执行的任务。 2. 创建一个线程池,用来管理线程。可以使用Java中的ThreadPoolExecutor类来实现。 3. 将任务添加到任务队列中。 4. 当线程池中有空闲线程时,从任务队列中取出一个任务,将其分配给空闲线程执行。 5. 当线程池中没有空闲线程时,新的任务将被放入任务队列中等待执行。 6. 当所有的任务都执行完毕后,关闭线程池。 示例代码如下: ``` import java.util.concurrent.*; public class TaskQueue { public static void main(String[] args) { // 创建一个任务队列 BlockingQueue<Runnable> taskQueue = new LinkedBlockingQueue<>(); // 创建一个线程池,设置线程池大小为5 ThreadPoolExecutor executor = new ThreadPoolExecutor(5, 5, 0L, TimeUnit.MILLISECONDS, taskQueue); // 添加任务 for (int i = 0; i < 10; i++) { executor.execute(new Task(i)); } // 关闭线程池 executor.shutdown(); } } class Task implements Runnable { private int taskId; public Task(int taskId) { this.taskId = taskId; } @Override public void run() { System.out.println("Task " + taskId + " is running."); } } ``` 在这个示例代码中,我们创建了一个大小为5的线程池,并向任务队列中添加了10个任务。当线程池中有空闲线程时,会从任务队列中取出一个任务并执行。当所有的任务都执行完毕后,关闭线程池。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值