通過前面 Apache Mina 入門 (二)—— 異步通信機制
我們可以實現一個長連接的客戶端。但會發現一個問題,就是當網絡、服務器、應用程序出現問題而導致連接斷開後,我們的客戶端不能自動重連服務器。導致客戶端程序癱瘓,不能使用。這個時候,通過增加一個監聽器,就能實現重連。
在我們實際生產環境中,斷線的原因可能更復雜:網絡不穩定、延時、服務器負載高、服務器或者應用程序的發送或者接收緩衝區滿等等問題都可能導致數據傳輸過程出現類似於斷線的情況,這個時候,光檢測Session關閉是遠遠不夠的,這個時候就需要一種重連機制,比如讀寫空閒超過30秒,就進行重連。對於數據不間斷、實時性高、數據量大的應用場景,更是實用。
具體實現代碼如下:
/**
* 添加重連監聽
*
* @author liuc
* @date 2017-12-20
*
*/
public class ClientReconnectTest {
public static IoSession session = null;
public static NioSocketConnector connector = null;
public static void main(String[] args) {
connector = new NioSocketConnector();
connector.setConnectTimeoutMillis(30000); // 設置連接超時
connector.getSessionConfig().setReceiveBufferSize(10240); // 設置接收緩衝區的大小
connector.getSessionConfig().setSendBufferSize(10240);// 設置輸出緩衝區的大小
connector.getFilterChain().addLast("logger", new LoggingFilter());
connector.getFilterChain().addLast("codec",
new ProtocolCodecFilter(new ByteArrayCodecFactory()));// 設置編碼過濾器
connector.setHandler(new ClientHandler());// 設置事件處理器
connector.setDefaultRemoteAddress(new InetSocketAddress("127.0.0.1",
8888));// 設置默認訪問地址
// 添加重連監聽---實現自動重連
connector.addListener(new IoListener() {
@Override
public void sessionDestroyed(IoSession arg0) throws Exception {
//重連10次
for (int i= 0 ;i <= 10; i++) {
try {
Thread.sleep(3000);
ConnectFuture future = connector.connect();
future.awaitUninterruptibly();// 等待連接創建成功
session = future.getSession();// 獲取會話
if (session.isConnected()) {
System.out.println("斷線重連["
+ connector.getDefaultRemoteAddress()
.getHostName()
+ ":"
+ connector.getDefaultRemoteAddress()
.getPort() + "]成功");
break;
}
} catch (Exception ex) {
System.out.println("重連服務器登錄失敗,3秒再連接一次:" + ex.getMessage());
}
}
}
});
//確保連接成功,連接n次,
for (int i= 0 ;i <= 5; i++) {
try {
ConnectFuture future = connector.connect();
future.awaitUninterruptibly(); // 等待連接創建成功
session = future.getSession(); // 獲取會話
if(session.isConnected()){
System.out.println("連接服務端"
+ connector.getDefaultRemoteAddress()
.getHostName()
+ ":"
+ connector.getDefaultRemoteAddress()
.getPort()
+ "[成功]"
+ ",,時間:"
+ new SimpleDateFormat("yyyy-MM-dd HH:mm:ss")
.format(new Date()));
break;
}
} catch (RuntimeIoException e) {
System.out.println(
"連接服務端"
+ connector.getDefaultRemoteAddress()
.getHostName()
+ ":"
+ connector.getDefaultRemoteAddress()
.getPort()
+ "失敗"
+ ",,時間:"
+ new SimpleDateFormat("yyyy-MM-dd HH:mm:ss")
.format(new Date())
+ ", 連接SOCKET服務異常,請檢查SOCKET端口、IP是否正確,MSG服務是否啓動,異常內容:"
+ e.getMessage());
try {
Thread.sleep(5000);// 連接失敗後,重連間隔5s
} catch (InterruptedException e1) {
e1.printStackTrace();
}
}
}
BaseMessageForServer message = new BaseMessageForServer();
String content = "hello world!";
CRC32 crc = new CRC32();
try {
crc.update(content.getBytes("GBK"));
} catch (UnsupportedEncodingException e) {
e.printStackTrace();
}
message.setFuncid(5);
message.setPacketIdCode(10000);
message.setContent(content);
message.setCheckCode(crc.getValue());
message.setLength(content.getBytes().length);
session.write(message);
}
}
參考文章:http://blog.csdn.net/chwshuang/article/details/51023647