Java如何使用ReentrantLock实现长轮询
作者:NichenFly 时间:2023-11-26 00:09:08
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
标签:Java,ReentrantLock,长轮询
0
投稿
猜你喜欢
java比较器comparator使用示例分享
2022-07-18 22:45:15
C#飞机打字游戏的代码示例(winform版)
2021-09-10 17:43:17
Jenkins自动构建部署项目到远程服务器上的方法步骤
2023-11-25 00:21:40
Java编程利用socket多线程访问服务器文件代码示例
2023-10-11 15:36:37
JDK源码分析之String、StringBuilder和StringBuffer
2022-01-23 00:10:20
C#字符串和Acsii码相互转换
2022-09-24 00:12:07
C#+无unsafe的非托管大数组示例详解(large unmanaged array in c# without ‘unsafe’ keyword)
2021-06-07 13:33:55
spring启动加载程序的几种方法介绍
2022-03-11 14:15:32
Android开发组件flutter的20个常用技巧示例总结
2023-06-19 17:25:23
WPF实现钟表效果
2022-11-18 15:38:05
解决Jackson解析嵌套类问题(MismatchedInputException)
2023-07-26 03:09:31
Android实现ViewPager无限循环效果(一)
2022-08-02 18:58:34
Springboot集成spring data elasticsearch过程详解
2023-04-16 00:42:16
关于后端如何解决跨域的问题说明
2023-09-19 00:59:10
使用JPA双向多对多关联关系@ManyToMany
2021-12-04 03:37:12
ehcache模糊批量移除缓存的方法
2023-01-11 12:30:37
java按指定编码写入和读取文件内容的类分享
2023-06-18 10:13:01
mybatis使用Integer类型查询可能出现的问题
2022-09-01 12:47:38
Java封装、继承、多态三大特征的理解
2023-07-16 14:26:46
Android RxJava与Retrofit结合使用详解
2021-10-19 20:10:53