|  |  | 
 |  |  | import cn.lihu.jh.module.ecg.controller.admin.room.vo.MonitorInfoVO; | 
 |  |  | import cn.lihu.jh.module.ecg.dal.dataobject.checktype.CheckTypeDO; | 
 |  |  | import cn.lihu.jh.module.ecg.dal.dataobject.queue.QueueDO; | 
 |  |  | import cn.lihu.jh.module.ecg.dal.dataobject.queue.QueueStatisticDO; | 
 |  |  | import cn.lihu.jh.module.ecg.dal.dataobject.room.RoomDO; | 
 |  |  | import cn.lihu.jh.module.ecg.dal.mysql.call.CallMapper; | 
 |  |  | import cn.lihu.jh.module.ecg.dal.mysql.checktype.CheckTypeMapper; | 
 |  |  | 
 |  |  |     ConcurrentHashMap<String, BedBO> mapOpeningBed = new ConcurrentHashMap<>(); | 
 |  |  |  | 
 |  |  |     // 装机工位..不进入分流队列 | 
 |  |  |     Map<Integer, PriorityBlockingQueue<BedBO>> mapCheckTypeRoutingQueue = new HashMap(); | 
 |  |  |     ConcurrentHashMap<Integer, PriorityBlockingQueue<BedBO>> mapCheckTypeRoutingQueue = new ConcurrentHashMap<>(); | 
 |  |  |      | 
 |  |  |     Map<Integer, CheckTypeDO> mapCheckTypeVsReadyMax = null; | 
 |  |  |  | 
 |  |  | 
 |  |  |         } | 
 |  |  |  | 
 |  |  |         mapOpeningBed.remove( Utils.formatRoomBed(roomId, bedNo) ); | 
 |  |  |         removeRoutingQueue(bedBO); | 
 |  |  |         removeBedFromRoutingQueue(bedBO); | 
 |  |  |         return GlobalErrorCodeConstants.SUCCESS; | 
 |  |  |     } | 
 |  |  |  | 
 |  |  | 
 |  |  |         } | 
 |  |  |  | 
 |  |  |         bedBO.setStatus(BedStatusEnum.DOCTOR_PAUSE.getStatus()); | 
 |  |  |         removeRoutingQueue(bedBO); | 
 |  |  |         removeBedFromRoutingQueue(bedBO); | 
 |  |  |         return GlobalErrorCodeConstants.SUCCESS; | 
 |  |  |     } | 
 |  |  |  | 
 |  |  | 
 |  |  |         } | 
 |  |  |  | 
 |  |  |         bedBO.setStatus(BedStatusEnum.DOCTOR_ON.getStatus()); | 
 |  |  |         addRoutingQueue(bedBO); | 
 |  |  |         addBedToRoutingQueue(bedBO); | 
 |  |  |         return GlobalErrorCodeConstants.SUCCESS; | 
 |  |  |     } | 
 |  |  |  | 
 |  |  | 
 |  |  |         } | 
 |  |  |  | 
 |  |  |         bedBO.setStatus(BedStatusEnum.DOCTOR_ON.getStatus()); | 
 |  |  |         addRoutingQueue(bedBO); | 
 |  |  |         addBedToRoutingQueue(bedBO); | 
 |  |  |         return GlobalErrorCodeConstants.SUCCESS; | 
 |  |  |     } | 
 |  |  |  | 
 |  |  | 
 |  |  |             return ROOM_INVALID_STATUS; | 
 |  |  |         } | 
 |  |  |  | 
 |  |  |         removeRoutingQueue(bedBO); | 
 |  |  |         removeBedFromRoutingQueue(bedBO); | 
 |  |  |  | 
 |  |  |         bedBO.setStatus(BedStatusEnum.OPENING.getStatus()); | 
 |  |  |         return GlobalErrorCodeConstants.SUCCESS; | 
 |  |  | 
 |  |  |  | 
 |  |  |     // 常规 或者 领用 时 医生 取下一位患者 | 
 |  |  |     public ErrorCode nextPatient(Long roomId, String bedNo) { | 
 |  |  |         QueueDO firstReadyQueueItem = queueMapper.getFirstItemByBedAndStatus(roomId, bedNo, QueueStatusEnum.READY.getStatus()); | 
 |  |  |         QueueDO firstReadyQueueItem = queueMapper.getBedFirstQueueItemByStatus(roomId, bedNo, QueueStatusEnum.READY.getStatus()); | 
 |  |  |         // 该工位 没有 [候诊中中] 人员 | 
 |  |  |         if (null == firstReadyQueueItem) { | 
 |  |  |             return QUEUE_NOT_READY_PATIENT; | 
 |  |  | 
 |  |  |         firstReadyQueueItem.setStatus(QueueStatusEnum.ONSTAGE.getStatus()); | 
 |  |  |         queueMapper.updateById( firstReadyQueueItem ); | 
 |  |  |  | 
 |  |  |         // 分流队列中 该工位 候诊中中人的数量 减一 | 
 |  |  |         // 分流队列中 该工位 【候诊中】患者数量 减一 | 
 |  |  |         BedBO bo = mapOpeningBed.get( Utils.formatRoomBed( roomId, bedNo ) ); | 
 |  |  |         bo.readyPatNum.getAndDecrement(); // 可能已经【并发的】在 hurry-up 中改变了值 | 
 |  |  |         bo.readyPatNum--; // 可能已经在 hurry-up 中改变了值 | 
 |  |  |  | 
 |  |  |         refreshRoutingQueue(bo); | 
 |  |  |         refreshBedPriorityLevelInRoutingQueue(bo); | 
 |  |  |         return GlobalErrorCodeConstants.SUCCESS; | 
 |  |  |     } | 
 |  |  |  | 
 |  |  |     // 装机时 医生 取下一位患者 | 
 |  |  |     public ErrorCode nextInstallPatient(Long roomId, String bedNo) { | 
 |  |  |         // 从 DB 把 序号最小的 [已领用] [已召回-安装] 的人 设置为 [安装中] | 
 |  |  |         QueueDO firstItem = queueMapper.getFirstItemByRoomAndStatus(roomId, List.of(QueueStatusEnum.RECEIVED.getStatus(), QueueStatusEnum.RECALLED_INSTALL.getStatus())); | 
 |  |  |         QueueDO firstItem = queueMapper.getRoomFirstQueueItemByStatus(roomId, List.of(QueueStatusEnum.RECEIVED.getStatus(), QueueStatusEnum.RECALLED_INSTALL.getStatus())); | 
 |  |  |         // 该工位 没有 [已领用] | [已召回-安装] 人员 | 
 |  |  |         if (null == firstItem) { | 
 |  |  |             return QUEUE_NOT_READY_PATIENT; | 
 |  |  |         } | 
 |  |  |  | 
 |  |  |         firstItem.setStatus( QueueStatusEnum.INSTALLING.getStatus() ); | 
 |  |  |         firstItem.setBedNo( bedNo ); | 
 |  |  |         firstItem.setBedNo( bedNo );  // 转到 安装工位 | 
 |  |  |         queueMapper.updateById( firstItem ); | 
 |  |  |  | 
 |  |  |         // 检查项目.亲和性 处理逻辑 | 
 |  |  |         // 检查项目.亲和性 处理逻辑:  变为【安装中】后 拉 其他检查项来亲和 | 
 |  |  |         CheckTypeDO checkTypeDO = getCheckTypeItem( firstItem.getBookCheckType() ); | 
 |  |  |         if (checkTypeDO.getAffinityCheckTypes().length > 0) { | 
 |  |  |             procAffinityWhenSomeOneEntryInstalling(firstItem.getPatId(), roomId, bedNo, checkTypeDO.getAffinityCheckTypes() ); | 
 |  |  |         } | 
 |  |  |  | 
 |  |  |         // 装机工位 不进入 分流队列 | 
 |  |  |         // 分流队列中 该工位 [已领用]人的数量 减一 | 
 |  |  |         //BedQueueBO bo = mapBedVsQueue.get( Utils.formatRoomBed( roomId, bedNo ) ); | 
 |  |  |         //bo.queueNum.getAndDecrement(); // 可能已经【并发的】在 hurry-up 中改变了值 | 
 |  |  |         //refreshRoutingQueue(bo); | 
 |  |  |         // 装机工位 不参与分流 | 
 |  |  |  | 
 |  |  |         return GlobalErrorCodeConstants.SUCCESS; | 
 |  |  |     } | 
 |  |  | 
 |  |  |         bedBOList.forEach(item -> { | 
 |  |  |             mapOpeningBed.put( Utils.formatRoomBed(item.roomId, item.bedNo), item); | 
 |  |  |             if (BedStatusEnum.DOCTOR_ON.getStatus() == item.getStatus()) { | 
 |  |  |                 addRoutingQueue(item); | 
 |  |  |                 addBedToRoutingQueue(item); | 
 |  |  |             } | 
 |  |  |         }); | 
 |  |  |  | 
 |  |  | 
 |  |  |         BedBO bedBO = getBedQueueBO(roomId, bedNo); | 
 |  |  |  | 
 |  |  |         // 处理 过号-回来 的人 | 
 |  |  |         while (bedBO.readyPatNum.get() < bedBO.maxReadyPatNum) { | 
 |  |  |         while (bedBO.getReadyPatNum() < bedBO.getMaxReadyPatNum()) { | 
 |  |  |             // 查看 当前工位 是否有过号-回来的患者 | 
 |  |  |             Integer updateNum = queueMapper.queueRecalledPatient( | 
 |  |  |                     bedBO.getRoomId(), | 
 |  |  | 
 |  |  |  | 
 |  |  |             curSeqNum.getAndIncrement(); | 
 |  |  |  | 
 |  |  |             // 可能已经【并发的】在 nextPatient 中改变了值 | 
 |  |  |             bedBO.readyPatNum.incrementAndGet(); | 
 |  |  |             // 可能已经在 nextPatient 中改变了值 | 
 |  |  |             bedBO.readyPatNum++; | 
 |  |  |  | 
 |  |  |             // 可能已经【并发的】在 nextPatient 中改变了分流队列顺序 | 
 |  |  |             refreshRoutingQueue(bedBO); | 
 |  |  |             refreshBedPriorityLevelInRoutingQueue(bedBO); | 
 |  |  |         } | 
 |  |  |  | 
 |  |  |         // 处理 排队中 患者 | 
 |  |  | 
 |  |  |             if (null == bedBO) | 
 |  |  |                 return; | 
 |  |  |  | 
 |  |  |             int curQueueNum = bedBO.readyPatNum.get(); | 
 |  |  |             if (curQueueNum > bedBO.maxReadyPatNum) | 
 |  |  |             int curQueueNum = bedBO.getReadyPatNum(); | 
 |  |  |             if (curQueueNum > bedBO.getMaxReadyPatNum()) | 
 |  |  |                 throw new RuntimeException("hurryupOneCheckType: exceed max queue number!"); | 
 |  |  |  | 
 |  |  |             if (curQueueNum == bedBO.maxReadyPatNum) | 
 |  |  |             if (curQueueNum == bedBO.getMaxReadyPatNum()) | 
 |  |  |                 return; | 
 |  |  |  | 
 |  |  |             // 查看 是否有排队中的患者 | 
 |  |  | 
 |  |  |  | 
 |  |  |             curSeqNum.getAndIncrement(); | 
 |  |  |  | 
 |  |  |             // 可能已经【并发的】在 nextPatient 中改变了值 | 
 |  |  |             bedBO.readyPatNum.incrementAndGet(); | 
 |  |  |             // 可能已经在 nextPatient 中改变了值 | 
 |  |  |             bedBO.readyPatNum++; | 
 |  |  |  | 
 |  |  |             // 可能已经【并发的】在 nextPatient 中改变了分流队列顺序 | 
 |  |  |             refreshRoutingQueue(bedBO); | 
 |  |  |             refreshBedPriorityLevelInRoutingQueue(bedBO); | 
 |  |  |         } | 
 |  |  |     } | 
 |  |  |  | 
 |  |  | 
 |  |  |     } | 
 |  |  |  | 
 |  |  |     private void hurryupOneBedCheckTypeList(BedBO bedBO) { | 
 |  |  |         while (bedBO.readyPatNum.get() < bedBO.maxReadyPatNum) { | 
 |  |  |         while (bedBO.getReadyPatNum() < bedBO.getMaxReadyPatNum()) { | 
 |  |  |             // 抢占 是否有排队中的患者 | 
 |  |  |             Integer updateNum = queueMapper.preemptWaitingPatientWithBedCheckTypes( | 
 |  |  |                     bedBO.getRoomId(), | 
 |  |  | 
 |  |  |  | 
 |  |  |             curSeqNum.getAndIncrement(); | 
 |  |  |  | 
 |  |  |             bedBO.readyPatNum.incrementAndGet(); | 
 |  |  |             bedBO.readyPatNum++; | 
 |  |  |         } | 
 |  |  |  | 
 |  |  |         // 改变了分流队列顺序 | 
 |  |  |         refreshRoutingQueue(bedBO); | 
 |  |  |         refreshBedPriorityLevelInRoutingQueue(bedBO); | 
 |  |  |     } | 
 |  |  |  | 
 |  |  |     /** | 
 |  |  | 
 |  |  |         return ret; | 
 |  |  |     } | 
 |  |  |  | 
 |  |  |     private void addRoutingQueue(BedBO bedBO) { | 
 |  |  |     private void addBedToRoutingQueue(BedBO bedBO) { | 
 |  |  |         // 装机工位 不进 分流队列,不接收 【排队中】患者分流 | 
 |  |  |         if (Objects.equals(BedOpTypeEnum.OP_INSTALL.getOpType(), bedBO.getOpType())) | 
 |  |  |             return; | 
 |  |  | 
 |  |  |         queueStatusList.add(QueueStatusEnum.READY.getStatus()); //[候诊中] | 
 |  |  |         List<QueueDO> queueDOList = queueMapper.getBedQueueByStatus(bedBO.getRoomId(), bedBO.getBedNo(), queueStatusList); | 
 |  |  |         bedBO.setMaxReadyPatNum( getBedReadyMax(bedBO) ); | 
 |  |  |         bedBO.setReadyPatNum(new AtomicInteger(queueDOList.size())); | 
 |  |  |         bedBO.setReadyPatNum( queueDOList.size() ); | 
 |  |  |  | 
 |  |  |         Integer[] checkTypes = bedBO.getCheckTypes(); | 
 |  |  |         Arrays.stream(checkTypes).forEach(checkType -> { | 
 |  |  | 
 |  |  |         }); | 
 |  |  |     } | 
 |  |  |  | 
 |  |  |     private void removeRoutingQueue(BedBO bedBO) { | 
 |  |  |     private void removeBedFromRoutingQueue(BedBO bedBO) { | 
 |  |  |         // 装机工位 不进 分流队列,不接收 【排队中】患者分流 | 
 |  |  |         if (Objects.equals(BedOpTypeEnum.OP_INSTALL.getOpType(), bedBO.getOpType())) | 
 |  |  |             return; | 
 |  |  | 
 |  |  |         }); | 
 |  |  |     } | 
 |  |  |  | 
 |  |  |     private void refreshRoutingQueue(BedBO bedBO) { | 
 |  |  |     private void refreshBedPriorityLevelInRoutingQueue(BedBO bedBO) { | 
 |  |  |         // 装机工位 不进 分流队列,不接收 【排队中】患者分流 | 
 |  |  |         if (Objects.equals(BedOpTypeEnum.OP_INSTALL.getOpType(), bedBO.getOpType())) | 
 |  |  |             return; | 
 |  |  | 
 |  |  |         Integer[] checkTypes = bedBO.getCheckTypes(); | 
 |  |  |         Arrays.stream(checkTypes).forEach(checkType -> { | 
 |  |  |             PriorityBlockingQueue<BedBO> routingQueue = mapCheckTypeRoutingQueue.get(checkType); | 
 |  |  |             routingQueue.remove(bedBO); | 
 |  |  |             routingQueue.offer(bedBO); | 
 |  |  |             if (null != routingQueue) { | 
 |  |  |                 routingQueue.remove(bedBO); | 
 |  |  |                 routingQueue.offer(bedBO); | 
 |  |  |             } | 
 |  |  |         }); | 
 |  |  |     } | 
 |  |  |  | 
 |  |  | 
 |  |  |         for (int i = 0; i < affinityItems.size(); i++) { | 
 |  |  |             QueueDO queueItem = affinityItems.get(i); | 
 |  |  |             queueItem.setStatus(QueueStatusEnum.AFFINITY_INSTALL.getStatus()); //改变 排队状态 | 
 |  |  |             queueItem.setAffinityItem(1); | 
 |  |  |             queueItem.setRoomId(roomId_operator); | 
 |  |  |             queueItem.setBedNo(bedNo_operator);  // 跟随到 装机 工位 | 
 |  |  |             queueMapper.updateById(queueItem); |