`
落叶换新叶
  • 浏览: 25632 次
  • 性别: Icon_minigender_1
社区版块
存档分类
最新评论

java socket 心跳链接

阅读更多

1、client客户端

public class Client {

	/**
	 * 处理服务端发回的对象,可实现该接口。
	 */
	public static interface ObjectAction{
		void doAction(Object obj,Client client);
	}
	
	public static final class DefaultObjectAction implements ObjectAction{
		public void doAction(Object obj,Client client) {
			System.out.println("处理:\t"+obj.toString());
		}
	}
	
	public static void main(String[] args) throws UnknownHostException, IOException {
		String serverIp = "127.0.0.1";
		int port = 65432;
		Client client = new Client(serverIp,port);
		client.start();
	}
	
	private String serverIp;
	private int port;
	private Socket socket;
	private boolean running=false; //连接状态
	
	private long lastSendTime; //最后一次发送数据的时间
	
	//用于保存接收消息对象类型及该类型消息处理的对象
	private ConcurrentHashMap<Class, ObjectAction> actionMapping = new ConcurrentHashMap<Class,ObjectAction>();
	
	public Client(String serverIp, int port) {
		this.serverIp=serverIp;
		this.port=port;
	}
	
	public void start() throws UnknownHostException, IOException {
		if(running)return;
		socket = new Socket(serverIp,port);
		System.out.println("本地端口:"+socket.getLocalPort());
		lastSendTime=System.currentTimeMillis();
		running=true;
		new Thread(new KeepAliveWatchDog()).start();  //保持长连接的线程,每隔2秒项服务器发一个一个保持连接的心跳消息
		new Thread(new ReceiveWatchDog()).start();    //接受消息的线程,处理消息
	}
	
	public void stop(){
		if(running)running=false;
	}
	
	/**
	 * 添加接收对象的处理对象。
	 * @param cls 待处理的对象,其所属的类。
	 * @param action 处理过程对象。
	 */
	public void addActionMap(Class<Object> cls,ObjectAction action){
		actionMapping.put(cls, action);
	}

	public void sendObject(Object obj) throws IOException {
		ObjectOutputStream oos = new ObjectOutputStream(socket.getOutputStream());
		oos.writeObject(obj);
		System.out.println("发送:\t"+obj);
		oos.flush();
	}
	
	class KeepAliveWatchDog implements Runnable{
		long checkDelay = 10;
		long keepAliveDelay = 2000;
		public void run() {
			while(running){
				if(System.currentTimeMillis()-lastSendTime>keepAliveDelay){
					try {
						Client.this.sendObject(new KeepAlive());
					} catch (IOException e) {
						e.printStackTrace();
						Client.this.stop();
					}
					lastSendTime = System.currentTimeMillis();
				}else{
					try {
						Thread.sleep(checkDelay);
					} catch (InterruptedException e) {
						e.printStackTrace();
						Client.this.stop();
					}
				}
			}
		}
	}
	
	class ReceiveWatchDog implements Runnable{
		public void run() {
			while(running){
				try {
					InputStream in = socket.getInputStream();
					if(in.available()>0){
						ObjectInputStream ois = new ObjectInputStream(in);
						Object obj = ois.readObject();
						System.out.println("接收:\t"+obj);
						ObjectAction oa = actionMapping.get(obj.getClass());
						oa = oa==null?new DefaultObjectAction():oa;
						oa.doAction(obj, Client.this);
					}else{
						Thread.sleep(10);
					}
				} catch (Exception e) {
					e.printStackTrace();
					Client.this.stop();
				} 
			}
		}
	}
}

 2、Server

public class Server {

	/**
	 * 要处理客户端发来的对象,并返回一个对象,可实现该接口。
	 */
	public interface ObjectAction{
		Object doAction(Object rev, Server server);
	}
	
	public static final class DefaultObjectAction implements ObjectAction{
		public Object doAction(Object rev,Server server) {
			System.out.println("处理并返回:"+rev);
			return rev;
		}
	}
	
	public static void main(String[] args) {
		int port = 65432;
		Server server = new Server(port);
		server.start();
	}
	
	private int port;
	private volatile boolean running=false;
	private long receiveTimeDelay=3000;
	private ConcurrentHashMap<Class, ObjectAction> actionMapping = new ConcurrentHashMap<Class,ObjectAction>();
	private Thread connWatchDog;
	
	public Server(int port) {
		this.port = port;
	}

	public void start(){
		if(running)return;
		running=true;
		connWatchDog = new Thread(new ConnWatchDog());
		connWatchDog.start();
	}
	
	@SuppressWarnings("deprecation")
	public void stop(){
		if(running)running=false;
		if(connWatchDog!=null)connWatchDog.stop();
	}
	
	public void addActionMap(Class<Object> cls,ObjectAction action){
		actionMapping.put(cls, action);
	}
	
	class ConnWatchDog implements Runnable{
		public void run(){
			try {
				ServerSocket ss = new ServerSocket(port,5);
				while(running){
					Socket s = ss.accept();
					new Thread(new SocketAction(s)).start();
				}
			} catch (IOException e) {
				e.printStackTrace();
				Server.this.stop();
			}
			
		}
	}
	
	class SocketAction implements Runnable{
		Socket s;
		boolean run=true;
		long lastReceiveTime = System.currentTimeMillis();
		public SocketAction(Socket s) {
			this.s = s;
		}
		public void run() {
			while(running && run){
				if(System.currentTimeMillis()-lastReceiveTime>receiveTimeDelay){
					overThis();
				}else{
					try {
						InputStream in = s.getInputStream();
						if(in.available()>0){
							ObjectInputStream ois = new ObjectInputStream(in);
							Object obj = ois.readObject();
							lastReceiveTime = System.currentTimeMillis();
							System.out.println("接收:\t"+obj);
							ObjectAction oa = actionMapping.get(obj.getClass());
							oa = oa==null?new DefaultObjectAction():oa;
							Object out = oa.doAction(obj,Server.this);
							if(out!=null){
								ObjectOutputStream oos = new ObjectOutputStream(s.getOutputStream());
								oos.writeObject(out);
								oos.flush();
							}
						}else{
							Thread.sleep(10);
						}
					} catch (Exception e) {
						e.printStackTrace();
						overThis();
					} 
				}
			}
		}
		
		private void overThis() {
			if(run)run=false;
			if(s!=null){
				try {
					s.close();
				} catch (IOException e) {
					e.printStackTrace();
				}
			}
			System.out.println("关闭:"+s.getRemoteSocketAddress());
		}
		
	}
}

 3、KeepAlive 保持心跳的类

public class KeepAlive implements Serializable {

	private static final long serialVersionUID = -2813120366138988480L;

	/* 覆盖该方法,仅用于测试使用。
	 * @see java.lang.Object#toString()
	 */
	@Override
	public String toString() {
		return new SimpleDateFormat("yyyy-MM-dd HH:mm:ss").format(new Date())+"\t维持连接包";
	}
}

 

 

 

 

 

0
0
分享到:
评论

相关推荐

    Socket长连接+心跳包+发送读取

    2. **Socket对象**:在Java等语言中,Socket类代表一个网络连接,ServerSocket类用于监听客户端连接请求。 3. **心跳机制**:心跳包的设计和实现,包括心跳间隔时间、心跳包内容、超时策略、重试机制等。心跳包可以...

    java Socket 长链接拔网线检测

    综上所述,Java Socket长链接拔网线检测涉及超时检测、心跳机制以及TCP Keepalive等多种方法。在实际应用中,结合具体场景选择合适的检测策略,并设计合理的异常处理和恢复机制,可以有效地保证长连接的稳定性和可靠...

    Android的socket长连接(心跳检测)

    - 在Android中,我们可以使用`java.net.Socket`类来创建和管理Socket连接。 2. **Android中创建Socket连接** - 创建Socket实例:`Socket socket = new Socket("服务器IP", 服务器端口);` - 数据传输:通过`...

    Java实现Socket长连接和短连接

    实现Java Socket长连接的关键在于管理好连接状态和心跳机制: - 连接管理:维持一个连接池,为每次请求分配或复用已有的连接。 - 心跳机制:定期发送心跳包,检测连接是否有效,防止因网络问题导致的“假死”连接。 ...

    基于java socket的聊天-单聊

    Java Socket编程是网络通信的基础,尤其在构建基于TCP/IP协议的客户端-服务器应用程序时,它扮演着关键角色。本教程将深入探讨如何使用Java Socket实现一个简单的单聊系统,包括客户端(ChatSocketClient)和服务器...

    java_sx.rar_java socket _java 通讯_socket

    Java Socket是Java编程语言中用于实现网络通信的核心API,它基于TCP/IP协议栈,提供了可靠的双向、连接导向的数据传输服务。在Java中,Socket和ServerSocket类是进行网络通信的基础,它们分别代表客户端和服务器端的...

    Java实现心跳机制的方法

    心跳包是一种小包,主要应用于长连接的保持与短线链接。 二、心跳机制实现方式 心跳机制有两种实现方式,一种基于TCP自带的心跳包,TCP的SO_KEEPALIVE选项可以实现心跳机制。系统默认的默认跳帧频率为2小时,超过2...

    Java 长链接示例

    1. **Socket编程**:Java中的`java.net.Socket`和`java.net.ServerSocket`类是进行TCP通信的基础。服务器端创建`ServerSocket`监听特定端口,客户端创建`Socket`连接服务器。 2. **BufferedReader和PrintWriter**:...

    Java的socket长连接实例

    在Java编程中,Socket通信是网络编程的基础,它允许两台计算机通过TCP/IP协议进行双向通信。本实例将深入探讨如何实现一个Java的Socket长连接。长连接是指在客户端和服务端建立连接后,保持连接状态,多次进行数据...

    java建立TCP长链接

    在Java中,我们通常使用`java.net.Socket`类和`java.net.ServerSocket`类来实现TCP通信。以下是建立TCP长连接的基本步骤: 1. **服务器端:** - 创建一个`ServerSocket`实例,指定监听的端口号。 - 调用`accept()...

    Socket服务器端断开后重新打开,客户端将自动完成链接

    在Java等编程语言中,我们可以使用Socket类来创建服务器端(ServerSocket)和客户端(Socket)。服务器端通常会创建一个ServerSocket,并绑定到特定的IP地址和端口号,然后调用其accept()方法开始监听连接请求。每当...

    socke 服务端多链接+心跳包

    本文将深入探讨"socke 服务端多链接+心跳包"这一主题,解析如何在服务端处理多个客户端连接,并维持这些连接的活跃状态。 首先,让我们了解什么是Socket。Socket,即套接字,是网络通信中的一个概念,它提供了进程...

    有关socket的编程代码(简单的聊天程序).rar_Socket编程 聊天_java socket 聊天

    Socket编程是计算机网络通信中的重要概念,特别是在Java中,它为应用程序提供了低级的、面向连接的、基于字节流的通信方式。本压缩包包含的"有关socket的编程代码(简单的聊天程序)"是一个基础的Java聊天程序示例,...

    基于java的Socket 聊天通信演示代码.zip

    在Java中,Socket是Java的`java.net`包提供的类,它允许两个网络应用程序之间建立全双工(双向)的通信链接。Socket通信通常涉及到TCP/IP协议,这是一种面向连接的、可靠的传输协议,确保数据的正确传输。 `Chat...

    flex与java采用socket方式通信

    9. **博客链接**:提到的博客链接(https://xiegangthrille.iteye.com/blog/660219)可能包含了作者对于Flex和Java Socket通信的详细实践和经验分享,可以参考学习。 通过上述知识点,我们可以了解到如何在Flex和...

    Android应用源码安卓与PC的Socket通信项目java版

    本项目"Android应用源码安卓与PC的Socket通信项目java版"聚焦于如何使用Java在Android设备上实现与PC之间的Socket通信。 首先,让我们深入理解Socket的基本概念。Socket是网络上的进程间通信(IPC)的一种方式,...

    用java实现cmpp协议

    在Java中实现CMPP协议,首先需要使用`java.net.Socket`类创建TCP连接到中国移动的CMPP服务器。TCP是一种面向连接的、可靠的传输层协议,确保数据的有序无损传输。 3. **CMPP报文格式** CMPP报文由报文头和报文体...

    如何判断SOCKET通信中,客户端在10s内未发送数据或者未收到服务器发送的数据,自动断开.doc

    ### 如何判断SOCKET通信中,客户端在10秒内未发送数据或未收到服务器发送的数据,自动断开 在本文档中,我们将探讨一种机制,即如何在SOCKET长连接通信中判断客户端在一定时间内(例如10秒)是否收到来自服务器的...

    OICQ(Java版)

    OICQ的核心是网络通信,Java中的`java.net`包提供了丰富的网络编程接口,如Socket和ServerSocket,用于建立客户端和服务器之间的连接。开发者可以通过这些接口实现数据的发送和接收,构建TCP或UDP通信。 **2. 即时...

    cmpp2 java实现及网关模拟器

    CMPP2(China Mobile Short Message Peer-to-Peer ...通过参考提供的链接(http://blog.csdn.net/wu_jia123/article/details/50039953)中的教程,开发者可以更深入地了解如何在Java环境中搭建和使用CMPP2系统。

Global site tag (gtag.js) - Google Analytics