Java如何使用ReentrantLock实现长轮询
作者:NichenFly 发布时间:2023-11-26 00:09:08
标签:Java,ReentrantLock,长轮询
Java代码
1. ReentrantLock
加锁阻塞,一个condition对应一个线程,以便于唤醒时使用该condition一定会唤醒该线程
/**
* 获取探测点数据,长轮询实现
* @param messageId
* @return
*/
public JSONObject getToutData(String messageId) {
Message message = toutMessageCache.get(messageId);
if (message == null) {
// 等待
lock.lock();
try {
Condition condition = lock.newCondition();
conditionMap.put(messageId + "_data", condition);
condition.await(CONNECTION_HOLD_TIMEOUT, TimeUnit.SECONDS); // 等待60s
} catch (InterruptedException e) {
// 等待超时, do nothing
} finally {
lock.unlock();
}
}
// 再次尝试获取
message = toutMessageCache.get(messageId);
if (message == null) {
// 如果还没有, 返回空对象
return null;
}
byte[] bytes = message.getDataBytes();
if (bytes == null) {
return null;
}
String resStr = new String(bytes, StandardCharsets.UTF_8);
// log.info("resStr: {}", resStr);
JSONObject resObj;
try {
resObj = new JSONObject(resStr);
resObj.put("invokeTime", DateUtil.format(new Date(resObj.getLong("invokeTime")), DatePattern.NORM_DATETIME_MS_PATTERN));
} catch (Exception e) {
resObj = new JSONObject();
}
return resObj;
}
2. 回调
当异步数据返回,使用上一步的condition唤醒线程
public void callback(Message message) {
String messageId = message.getId();
toutMessageCache.put(message.getId(), message);
String messageDataId = messageId + "_data";
if (conditionMap.containsKey(messageDataId)) {
lock.lock();
try {
Condition condition = conditionMap.get(messageDataId);
condition.signal();
} catch (Exception e) {
e.printStackTrace();
} finally {
lock.unlock();
conditionMap.remove(messageDataId);
}
}
}
3. 唤醒
执行回调操作
public void distribute(Message message, ChannelHandlerContext ctx) {
MessageType messageType = message.getMessageType();
switch (messageType) {
case TOUT_DATA_RESPONSE:
// 数据响应
toutService.callback(message);
break;
}
}
4. 调用
调用时,判断返回的值是否为空,如果为空,与前端约定,当返回该状态值时,应再次发起相同请求
/**
* 获取探测数据(使用长轮询实现)
* @param linkId
* @return
*/
@GetMapping("/data")
public ResultVO getToutData(String linkId) {
JSONObject resObj = toutService.getToutData(linkId);
if (resObj == null || resObj.isEmpty()) {
return ResultVOUtil.error(ResultEnum.NO_MESSAGE_HOLD_CONNECTION);
}
return ResultVOUtil.success(resObj);
}
5.前端实现
简单使用递归实现了当数据返回无效时再次发起请求
let that = this
function getData() {
if (toutStatus === statusEnum.start) {
getToutData({
linkId
}).then(res => {
if (res.code === ERROR_CODE_OK) {
that.toutData = res.data
toutStatus = statusEnum.resData
that._btnStatus()
} else {
getData()
}
})
}
}
// 递归循环调用
getData()
来源:https://blog.csdn.net/weixin_42096329/article/details/115470071


猜你喜欢
- 一.导入方式由于jdk中没有servlet对应的jar包,所以需要咱们手动引入,有两种方式:1.可以采取向lib目录导入servlet-ap
- ①概念二叉搜索树又称二叉排序树,它或者是一棵空树**,或者是具有以下性质的二叉树:若它的左子树不为空,则左子树上所有节点的值都小于根节点的值
- C++ 中String 替换指定字符串的实例详解C++的string提供了replace方法来实现字符串的替换,但是对于将字符串中某个字符串
- 小伙伴们,最近比较忙,没什么时间写,今天给大家分享的是JAVA如何导出EXCEL表格,因为最近有做这样一个功能,所以分享出来,如有不对之处,
- 最近接手了一个需求,要求实现,叮咚买菜。秒杀位置的轮播拆解通过观察发现其实还是挺简单,大致分为1、商品图片的上下轮播2、价格布局渐隐渐现在a
- 这篇文章主要介绍了Spring事务失效问题分析及解决方案,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的
- 一、@ConditionalOnClass() Spring中存在指定class对象时,注入指定配置和ConditionalOnBean()
- 平时开发的工作中, 自己组内的很多大佬经常使用Optional的用法, 自己问他们, 这个到底有什么好处呢,他们说可以很好的规避好空指针的问
- 图的实际应用在现实生活中,有许多应用场景会包含很多点以及点点之间的连接,而这些应用场景我们都可以用即将要学习的图这种数据结构去解决。地图:我
- RequestBody注解的List参数传递Controller方法参数:@RequestBody List<Long> ids
- 目录算术操作符移位操作符位操作符赋值操作符 单目操作符(类型) 强制类型转换 &n
- 需求:有些时候,我们需要连接多个数据库,但是,在方法调用前并不知道到底是调用哪个。即同时保持多个数据库的连接,在方法中根据传入的参数来确定。
- String 对象是不可改变的。每次使用 System.String 类中的方法之一时,都要在内存中创建一个新的字符串对象,这就需要为该新对
- 后台Java代码【验证码生成】/** * 随机生成6位随机验证码 */ public static String createRandomV
- 作为最基础的引用数据类型,Java 设计者为 String 提供了字符串常量池以提高其性能,那么字符串常量池的具体原理是什么,我们带着以下三
- 本文实例讲述了Java Swing实现让窗体居中显示的方法。分享给大家供大家参考,具体如下:Swing组件是AWT组建的增强组件,是功能强大
- 本文实例讲述了C#实现自定义Dictionary类。分享给大家供大家参考。具体如下:1.关于MyDictionary类本文中实现的MyDic
- 近日工程中,逐渐感觉到原来复制粘贴代码的笨重,突然想起以前有人和我说起过Git和SVN之类的版本管理工具。由于平时主要是写Java代码,所以
- 熟悉Android的朋友们都知道,不管是微博客户端还是新闻客户端,都离不开列表组件,可以说列表组件是Android数据展现方面最重要的组件,
- > 此文主要通过WinForm来制作一个休息提醒闹钟,通过设置时间间隔进行提醒,避免沉浸式的投入到工作或者学习当中,战斗的同时也要照顾