日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区

您的位置:首頁技術文章
文章詳情頁

Java實現心跳機制的方法

瀏覽:14日期:2022-08-29 14:42:27

一、心跳機制簡介

在分布式系統中,分布在不同主機上的節點需要檢測其他節點的狀態,如服務器節點需要檢測從節點是否失效。為了檢測對方節點的有效性,每隔固定時間就發送一個固定信息給對方,對方回復一個固定信息,如果長時間沒有收到對方的回復,則斷開與對方的連接。

發包方既可以是服務端,也可以是客戶端,這要看具體實現。因為是每隔固定時間發送一次,類似心跳,所以發送的固定信息稱為心跳包。心跳包一般為比較小的包,可根據具體實現。心跳包主要應用于長連接的保持與短線鏈接。

一般而言,應該客戶端主動向服務器發送心跳包,因為服務器向客戶端發送心跳包會影響服務器的性能。

二、心跳機制實現方式

心跳機制有兩種實現方式,一種基于TCP自帶的心跳包,TCP的SO_KEEPALIVE選項可以,系統默認的默認跳幀頻率為2小時,超過2小時后,本地的TCP 實現會發送一個數據包給遠程的 Socket. 如果遠程Socket 沒有發回響應, TCP實現就會持續嘗試 11 分鐘, 直到接收到響應為止。 否則就會自動斷開Socket連接。但TCP自帶的心跳包無法檢測比較敏感地知道對方的狀態,默認2小時的空閑時間,對于大多數的應用而言太長了。可以手工開啟KeepAlive功能并設置合理的KeepAlive參數。

另一種在應用層自己進行實現,基本步驟如下:

Client使用定時器,不斷發送心跳;Server收到心跳后,回復一個包;Server為每個Client啟動超時定時器,如果在指定時間內沒有收到Client的心跳包,則Client失效。

三、Java實現心跳機制

這里基于Java實現的簡單RPC框架實現心跳機制。Java實現代碼如下所示:

心跳客戶端類:

public class HeartbeatClient implements Runnable { private String serverIP = '127.0.0.1'; private int serverPort = 8089; private String nodeID = UUID.randomUUID().toString(); private boolean isRunning = true; // 最近的心跳時間 private long lastHeartbeat; // 心跳間隔時間 private long heartBeatInterval = 10 * 1000; public void run() { try { while (isRunning) { HeartbeatHandler handler = RPClient.getRemoteProxyObj(HeartbeatHandler.class, new InetSocketAddress(serverIP, serverPort)); long startTime = System.currentTimeMillis(); // 是否達到發送心跳的周期時間 if (startTime - lastHeartbeat > heartBeatInterval) { System.out.println('send a heart beat'); lastHeartbeat = startTime; HeartbeatEntity entity = new HeartbeatEntity(); entity.setTime(startTime); entity.setNodeID(nodeID); // 向服務器發送心跳,并返回需要執行的命令 Cmder cmds = handler.sendHeartBeat(entity); if (!processCommand(cmds)) continue; } } } catch (Exception e) { e.printStackTrace(); } } private boolean processCommand(Cmder cmds) { // ... return true; } }

心跳包實體類:

public class HeartbeatEntity implements Serializable { private long time; private String nodeID; private String error; private Map<String, Object> info = new HashMap<String, Object>(); public String getNodeID() { return nodeID; } public void setNodeID(String nodeID) { this.nodeID = nodeID; } public String getError() { return error; } public void setError(String error) { this.error = error; } public Map<String, Object> getInfo() { return info; } public void setInfo(Map<String, Object> info) { this.info = info; } public long getTime() { return time; } public void setTime(long time) { this.time = time; }}

服務器接受心跳包返回的命令對象類:

public class Cmder implements Serializable { private String nodeID; private String error; private Map<String, Object> info = new HashMap<String, Object>(); public String getNodeID() { return nodeID; } public void setNodeID(String nodeID) { this.nodeID = nodeID; } public String getError() { return error; } public void setError(String error) { this.error = error; } public Map<String, Object> getInfo() { return info; } public void setInfo(Map<String, Object> info) { this.info = info; }}

RPC服務注冊中心:

public class ServiceCenter { private ExecutorService executor = Executors.newFixedThreadPool(20); private final ConcurrentHashMap<String, Class> serviceRegistry = new ConcurrentHashMap<String, Class>(); private AtomicBoolean isRunning = new AtomicBoolean(true); // 服務器監聽端口 private int port = 8089; // 心跳監聽器 HeartbeatLinstener linstener; // 單例模式 private static class SingleHolder { private static final ServiceCenter INSTANCE = new ServiceCenter(); } private ServiceCenter() { } public static ServiceCenter getInstance() { return SingleHolder.INSTANCE; } public void register(Class serviceInterface, Class impl) { System.out.println('regeist service ' + serviceInterface.getName()); serviceRegistry.put(serviceInterface.getName(), impl); } public void start() throws IOException { ServerSocket server = new ServerSocket(); server.bind(new InetSocketAddress(port)); System.out.println('start server'); linstener = HeartbeatLinstener.getInstance(); System.out.println('start listen heart beat'); try { while (true) { // 1.監聽客戶端的TCP連接,接到TCP連接后將其封裝成task,由線程池執行 executor.execute(new ServiceTask(server.accept())); } } finally { server.close(); } } public void stop() { isRunning.set(false); executor.shutdown(); } public boolean isRunning() { return isRunning.get(); } public int getPort() { return port; } public void settPort(int port) { this.port = port; } public ConcurrentHashMap<String, Class> getServiceRegistry() { return serviceRegistry; } private class ServiceTask implements Runnable { Socket clent = null; public ServiceTask(Socket client) { this.clent = client; } public void run() { ObjectInputStream input = null; ObjectOutputStream output = null; try { // 2.將客戶端發送的碼流反序列化成對象,反射調用服務實現者,獲取執行結果 input = new ObjectInputStream(clent.getInputStream()); String serviceName = input.readUTF(); String methodName = input.readUTF(); Class<?>[] parameterTypes = (Class<?>[]) input.readObject(); Object[] arguments = (Object[]) input.readObject(); Class serviceClass = serviceRegistry.get(serviceName); if (serviceClass == null) { throw new ClassNotFoundException(serviceName + ' not found'); } Method method = serviceClass.getMethod(methodName, parameterTypes); Object result = method.invoke(serviceClass.newInstance(), arguments); // 3.將執行結果反序列化,通過socket發送給客戶端 output = new ObjectOutputStream(clent.getOutputStream()); output.writeObject(result); } catch (Exception e) { e.printStackTrace(); } finally { if (output != null) { try { output.close(); } catch (IOException e) { e.printStackTrace(); } } if (input != null) { try { input.close(); } catch (IOException e) { e.printStackTrace(); } } if (clent != null) { try { clent.close(); } catch (IOException e) { e.printStackTrace(); } } } } }}

心跳監聽類:

package com.cang.heartbeat; import java.io.IOException;import java.io.ObjectInputStream;import java.io.ObjectOutputStream;import java.lang.reflect.Method;import java.net.InetSocketAddress;import java.net.ServerSocket;import java.net.Socket;import java.util.Iterator;import java.util.Map;import java.util.concurrent.ConcurrentHashMap;import java.util.concurrent.ExecutorService;import java.util.concurrent.Executors;import java.util.concurrent.atomic.AtomicBoolean; /** * 心跳監聽保存信息 * * @author cang * @create_time 2016-09-28 11:40 */public class HeartbeatLinstener { private ExecutorService executor = Executors.newFixedThreadPool(20); private final ConcurrentHashMap<String, Object> nodes = new ConcurrentHashMap<String, Object>(); private final ConcurrentHashMap<String, Long> nodeStatus = new ConcurrentHashMap<String, Long>(); private long timeout = 10 * 1000; // 服務器監聽端口 private int port = 8089; // 單例模式 private static class SingleHolder { private static final HeartbeatLinstener INSTANCE = new HeartbeatLinstener(); } private HeartbeatLinstener() { } public static HeartbeatLinstener getInstance() { return SingleHolder.INSTANCE; } public ConcurrentHashMap<String, Object> getNodes() { return nodes; } public void registerNode(String nodeId, Object nodeInfo) { nodes.put(nodeId, nodeInfo); nodeStatus.put(nodeId, System.currentTimeMillis()); } public void removeNode(String nodeID) { if (nodes.containsKey(nodeID)) { nodes.remove(nodeID); } } // 檢測節點是否有效 public boolean checkNodeValid(String key) { if (!nodes.containsKey(key) || !nodeStatus.containsKey(key)) return false; if ((System.currentTimeMillis() - nodeStatus.get(key)) > timeout) return false; return true; } // 刪除所有失效節點 public void removeInValidNode() { Iterator<Map.Entry<String, Long>> it = nodeStatus.entrySet().iterator(); while (it.hasNext()) { Map.Entry<String, Long> e = it.next(); if ((System.currentTimeMillis() - nodeStatus.get(e.getKey())) > timeout) { nodes.remove(e.getKey()); } } } }

心跳處理類接口:

public interface HeartbeatHandler { public Cmder sendHeartBeat(HeartbeatEntity info);}

心跳處理實現類:

public class HeartbeatHandlerImpl implements HeartbeatHandler { public Cmder sendHeartBeat(HeartbeatEntity info) { HeartbeatLinstener linstener = HeartbeatLinstener.getInstance(); // 添加節點 if (!linstener.checkNodeValid(info.getNodeID())) { linstener.registerNode(info.getNodeID(), info); } // 其他操作 Cmder cmder = new Cmder(); cmder.setNodeID(info.getNodeID()); // ... System.out.println('current all the nodes: '); Map<String, Object> nodes = linstener.getNodes(); for (Map.Entry e : nodes.entrySet()) { System.out.println(e.getKey() + ' : ' + e.getValue()); } System.out.println('hadle a heartbeat'); return cmder; }}

測試類:

public class HeartbeatTest { public static void main(String[] args) { new Thread(new Runnable() { public void run() { try { ServiceCenter serviceServer = ServiceCenter.getInstance(); serviceServer.register(HeartbeatHandler.class, HeartbeatHandlerImpl.class); serviceServer.start(); } catch (IOException e) { e.printStackTrace(); } } }).start(); Thread client1 = new Thread(new HeartbeatClient()); client1.start(); Thread client2 = new Thread(new HeartbeatClient()); client2.start(); }}

四、總結

上面的代碼還有很多不足的地方,希望有空能進行改善:

配置為硬編碼; 命令類Cmder沒有實際實現,返回的Cmder對象沒有實際進行處理;

其他小問題就暫時不管了,希望以后能重寫上面的代碼。

以上就是Java實現心跳機制的方法的詳細內容,更多關于Java實現心跳機制的資料請關注好吧啦網其它相關文章!

標簽: Java
相關文章:
日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区
悠悠资源网久久精品| 国产精品麻豆成人av电影艾秋 | 91免费精品国偷自产在线在线| 国产欧美一级| 国产一区二区三区亚洲| 久久久精品五月天| 视频一区二区三区在线| 欧美一区免费| 电影亚洲精品噜噜在线观看| 日韩中文字幕不卡| 91欧美精品| sm捆绑调教国产免费网站在线观看| 婷婷激情图片久久| 欧美日韩一区二区三区不卡视频| 97国产成人高清在线观看| 国产亚洲网站| 老司机精品视频网| 午夜国产精品视频| 国产精品网址| 美女亚洲一区| 国产精品一区二区三区美女 | 视频一区日韩精品| 高清一区二区三区| 日韩一区二区久久| 国产精品v一区二区三区| 激情婷婷综合| 美女精品视频在线| 美女被久久久| 精品精品99| 一区二区国产精品| 日韩国产欧美一区二区| 日韩高清电影免费| 国产精品99一区二区| 国产精品伦一区二区| 美女久久网站| 成人三级高清视频在线看| 免费一级片91| 亚洲精品国产嫩草在线观看| 亚洲不卡视频| 色婷婷精品视频| 国产日产精品_国产精品毛片| 国产精品88久久久久久| 精品视频在线你懂得| 免费日本视频一区| 99精品视频在线| 国产福利资源一区| 中文字幕av一区二区三区四区| 欧美日韩国产v| 久久av超碰| 亚洲欧洲美洲国产香蕉| 日本韩国欧美超级黄在线观看| 欧美自拍一区| 久久av在线| 欧洲精品一区二区三区| 国产精品一区毛片| 亚洲欧美在线综合| 国产亚洲在线观看| 91看片一区| 精品久久亚洲| 日本欧美一区二区| 香蕉久久久久久久av网站| 色天使综合视频| 老司机精品视频网| 日本91福利区| 国产探花在线精品| 国产亚洲精品精品国产亚洲综合| 91精品国产调教在线观看| 91嫩草精品| 影视先锋久久| 国产精品一区2区3区| 亚洲精品1区2区| 精品国产乱码久久久久久1区2匹| 一本色道久久精品| 国产精品久久久久久久久久10秀| 亚洲一区国产| 国产精品久久久久久久久妇女| 日韩综合一区二区| 91精品成人| 国产精品99在线观看| 日韩精品亚洲专区在线观看| 精品捆绑调教一区二区三区| 国产精品国产三级在线观看| 亚洲a在线视频| 免费在线成人| 蜜桃视频在线观看一区二区| 91精品精品| 精品五月天堂| 国产精品香蕉| 婷婷成人av| 六月天综合网| 99久久激情| 欧美a级一区二区| 日韩综合小视频| 久久av在线| 一区在线免费观看| 一本大道色婷婷在线| 久久尤物视频| 国产精品丝袜在线播放| 亚洲精选91| re久久精品视频| 日韩精品一区二区三区免费观看| 精品国产鲁一鲁****| 国产精品一区二区精品| 日韩高清一区| 亚洲色图综合| 免费成人性网站| 国产精品日韩| 国产精品毛片在线| 悠悠资源网久久精品| 欧美日韩激情| 人人精品亚洲| 久久久久久久久99精品大| 国产精品久久久久久久久久10秀 | 日韩一区二区三区精品| 国产亚洲高清视频| 1024精品一区二区三区| 女生影院久久| 日韩欧美自拍| 欧美一级鲁丝片| 日韩欧美自拍| 999精品一区| 久久亚洲国产| 欧美日韩国产一区二区三区不卡| 电影天堂国产精品| 999国产精品999久久久久久| 亚洲国内欧美| 好看的亚洲午夜视频在线| 一区久久精品| 免费观看在线色综合| 亚洲乱亚洲高清| 日韩精品免费观看视频| 7m精品国产导航在线| 欧美精品影院| 久久久免费人体| 久久久久久久欧美精品| 亚洲精品少妇| 国产亚洲字幕| 国产精品jk白丝蜜臀av小说| 国产精品久久久久77777丨| 国产精品v日韩精品v欧美精品网站| 免费在线亚洲| 久久蜜桃av| 美女被久久久| 国产人成精品一区二区三| 久久精品国产99国产| 日韩一区电影| 黄色在线一区| 日本色综合中文字幕| 国产日韩三级| 日本久久精品| 91精品国产调教在线观看 | 久久精品凹凸全集| 久久天堂影院| 久久三级福利| 免费不卡在线观看| 欧美日本精品| 日韩中文影院| 亚洲免费网址| 青青国产精品| 四虎成人av| 亚洲精品2区| 免费人成黄页网站在线一区二区| 欧美欧美黄在线二区| 久久精品国产一区二区| 久久久久久黄| 在线国产精品一区| 欧美日韩中文| 久久久久国产精品一区三寸| 中文字幕一区日韩精品| 另类小说一区二区三区| 香蕉精品久久| 91精品国产自产观看在线| 97精品国产福利一区二区三区| 亚洲一级特黄| 欧美一区自拍| 久久久久久久久久久9不雅视频| 亚洲欧美日本日韩| 国产亚洲欧美日韩在线观看一区二区| 日韩欧美国产精品综合嫩v| 日本欧美在线看| 高清av一区| 一区二区三区四区日韩| 日产午夜精品一线二线三线| 久久福利毛片| 色婷婷综合网| 日日夜夜免费精品| 蜜桃视频在线网站| 亚洲最大av| 天堂√中文最新版在线| 男女性色大片免费观看一区二区| 久久香蕉网站| 中文一区二区| 国产成人免费| 亚洲精品系列| 欧美特黄一级大片| 国产伦精品一区二区三区千人斩| 免费精品国产的网站免费观看| 国产精品九九| 亚洲黄色影院| 国产成人精品一区二区三区在线|