您好,登錄后才能下訂單哦!
這篇文章主要介紹“Java中如何使用ReentrantLock實現長輪詢”,在日常操作中,相信很多人在Java中如何使用ReentrantLock實現長輪詢問題上存在疑惑,小編查閱了各式資料,整理出簡單好用的操作方法,希望對大家解答”Java中如何使用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;
}
當異步數據返回,使用上一步的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);
}
}
}
執行回調操作
public void distribute(Message message, ChannelHandlerContext ctx) {
MessageType messageType = message.getMessageType();
switch (messageType) {
case TOUT_DATA_RESPONSE:
// 數據響應
toutService.callback(message);
break;
}
}
調用時,判斷返回的值是否為空,如果為空,與前端約定,當返回該狀態值時,應再次發起相同請求
/**
* 獲取探測數據(使用長輪詢實現)
* @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);
}
簡單使用遞歸實現了當數據返回無效時再次發起請求
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()
到此,關于“Java中如何使用ReentrantLock實現長輪詢”的學習就結束了,希望能夠解決大家的疑惑。理論與實踐的搭配能更好的幫助大家學習,快去試試吧!若想繼續學習更多相關知識,請繼續關注億速云網站,小編會繼續努力為大家帶來更多實用的文章!
免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。