`

JavaSocket学习---NIO实现非阻塞的通信

阅读更多

示例代码见附件:很遗憾,太懒,代码内容还是我分的包,如果你要运行的话,还需要自己下一点小的功夫,改变一下包路径。

 

 

Server端代码:

package com.henushang.socket.chapter4nio;

import java.io.IOException;
import java.net.InetSocketAddress;
import java.nio.ByteBuffer;
import java.nio.CharBuffer;
import java.nio.channels.SelectionKey;
import java.nio.channels.Selector;
import java.nio.channels.ServerSocketChannel;
import java.nio.channels.SocketChannel;
import java.nio.charset.Charset;
import java.util.Iterator;
import java.util.Set;

public class EchoServerNoBlock {

	private int port = 8000;
	private ServerSocketChannel serverSocketChannel;
	Selector selector;

	public EchoServerNoBlock() throws Exception {
		selector = Selector.open();
		serverSocketChannel = ServerSocketChannel.open();
		serverSocketChannel.socket().setReuseAddress(true);
		serverSocketChannel.configureBlocking(false);// 设置通信为非阻塞模式,没有这一配置,则无法实现非阻塞
		serverSocketChannel.socket().bind(new InetSocketAddress(port));
		System.out.println("等待连接...");
	}

	public void service() throws Exception {
		serverSocketChannel.register(selector, SelectionKey.OP_ACCEPT);
		while (selector.select() > 0) {
			Set<SelectionKey> readyKeys = selector.selectedKeys();
			Iterator<SelectionKey> iterator = readyKeys.iterator();
			SelectionKey key = null;
			while (iterator.hasNext()) {
				try {
					key = iterator.next();
					iterator.remove();
					if (key.isAcceptable()) {
						System.out.println("accept");
						ServerSocketChannel ssc = (ServerSocketChannel) key
								.channel();
						SocketChannel sc = ssc.accept();
						System.out.println("接收到链接"
								+ sc.socket().getInetAddress() + ":"
								+ sc.socket().getPort());
						sc.configureBlocking(false);
						ByteBuffer buffer = ByteBuffer.allocate(1024);
						sc.register(selector, SelectionKey.OP_READ
								| SelectionKey.OP_WRITE, buffer);
					}
					if (key.isReadable()) {
						System.out.println("read");
						receive(key);
					}
					if (key.isWritable()) {
						send(key);
					}
				} catch (Exception e) {
					e.printStackTrace();
					if (key != null) {
						key.cancel();
						key.channel().close();
					}
				}
			}
		}
	}

	/**
	 * 发送消息处理方法
	 * @param key
	 * @throws Exception
	 * @author henushang
	 */
	public void send(SelectionKey key) throws Exception {
		ByteBuffer buffer = (ByteBuffer) key.attachment();
		SocketChannel socketChannel = (SocketChannel) key.channel();
		buffer.flip();
		String data = decode(buffer);
		if (data.indexOf("\r\n") == -1) {
			return;
		}
		String outputData = data.substring(0, data.indexOf("\n") + 1);
		System.out.println(outputData);
		ByteBuffer outputBuffer = encode("henushang:" + outputData);
		while (outputBuffer.hasRemaining()) {
			socketChannel.write(outputBuffer);
		}
		ByteBuffer temp = encode(outputData);
		buffer.position(temp.limit());
		buffer.compact();

		if (outputData.equals("byt\r\n")) {
			key.cancel();
			socketChannel.close();
			System.out.println("关闭连接");
		}

	}

	/**
	 * 接受链接处理方法
	 * @param key
	 * @throws IOException
	 * @author henushang
	 */
	public void receive(SelectionKey key) throws IOException {
		SocketChannel sc = (SocketChannel) key.channel();
		ByteBuffer buffer = (ByteBuffer) key.attachment();
		ByteBuffer readBuffer = ByteBuffer.allocate(32);
		sc.read(readBuffer);
		readBuffer.flip();

		buffer.limit(readBuffer.capacity());
		buffer.put(readBuffer);
	}

	static Charset charset = Charset.forName("UTF-8");

	/**
	 * 解码
	 * @param byteBuffer
	 * @return
	 * @author henushang
	 */
	public static String decode(ByteBuffer byteBuffer) {
		CharBuffer charBuffer = charset.decode(byteBuffer);
		return charBuffer.toString();
	}

	/**
	 * 编码
	 * @param msg
	 * @return
	 * @author henushang
	 */
	public static ByteBuffer encode(String msg) {
		return charset.encode(msg);
	}

}

 

客户端代码:

package com.henushang.socket.chapter4nio;

import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStreamReader;
import java.io.PrintWriter;
import java.net.InetAddress;
import java.net.InetSocketAddress;
import java.nio.channels.SocketChannel;

import com.henushang.socket.util.SocketUtils;

public class EchoClient {
	
	private SocketChannel socketChannel;
	private int port = 8000;
	
	public EchoClient() throws Exception {
		socketChannel = SocketChannel.open();
		InetAddress inetAddress = InetAddress.getLocalHost();
		InetSocketAddress inetSocketAddress = new InetSocketAddress(inetAddress, port);
		socketChannel.connect(inetSocketAddress);
		System.out.println("准备连接服务器");
	}
	
	public static void main(String[] args) throws Exception {
		new EchoClient().talk();
	}
	
	public void talk() {
		try {
			BufferedReader reader = SocketUtils.getReader(socketChannel.socket());
			PrintWriter pw = SocketUtils.getWriter(socketChannel.socket());
			BufferedReader localreaderReader = new BufferedReader(new InputStreamReader(System.in));
			String msg = null;
			while ((msg = localreaderReader.readLine()) != null) {
				System.out.println(msg);
				pw.println(msg);
				pw.flush();
				System.out.println(reader.readLine());
				if ("bye".equals(msg)) {
					break;
				}
			}
		} catch (IOException e) {
			e.printStackTrace();
		}finally{
			SocketUtils.close(socketChannel);
		}
		
	}
}

 

工具类代码:

package com.henushang.socket.util;

import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStream;
import java.io.InputStreamReader;
import java.io.OutputStream;
import java.io.PrintWriter;
import java.net.Socket;
import java.nio.channels.SocketChannel;

public class SocketUtils {
	
	public static PrintWriter getWriter(Socket socket) throws IOException {
		OutputStream os = socket.getOutputStream();
		return new PrintWriter(os);
	}
	
	public static PrintWriter getWriter(SocketChannel socketChannel) throws IOException {
		return getWriter(socketChannel.socket());
	}
	
	public static BufferedReader getReader(Socket socket) throws IOException{
		InputStream is = socket.getInputStream();
		return new BufferedReader(new InputStreamReader(is, "UTF-8"));
	}
	
	public static BufferedReader getReader(SocketChannel socketChannel) throws IOException{
		return getReader(socketChannel.socket());
	}
	
	public static void close(Socket socket) {
		try {
			if (socket != null) {
				socket.close();
			}
		} catch (Exception e) {
			e.printStackTrace();
		}
	}

	public static void close(SocketChannel socketChannel) {
		try {
			if (socketChannel != null) {
				socketChannel.close();
			}
		} catch (Exception e) {
			e.printStackTrace();
		}
	}
	
	public static String echo(String msg) {
		return "echo:" + msg;
	}
}

 

 

这种方式是一个线程在不停的循环检测监听到的事件,然后分别作出相应的处理。可是这种方式只适合那种处理过程时间短的情况,如果某一个操作处理的事件太长的话,则会让其他的事件一直处于等待状态,比如发送或者接收一个很大的文件,这样显然是很不合适的。

 

使用NIO的NoBlocking的操作方式固然减少了系统使用多线程建立连接的开销,但是也并不是完全适合任何情况的。

所以,在几种方式中没有最好的,只有最合适的。

 

而做程序也是一样的,没有那个语言是最好的,只有最合适的语言。

 

 

0
0
分享到:
评论

相关推荐

    javasocket学习资料

    Java NIO(New IO)库提供了非阻塞的Socket选项,可以提高程序的效率和响应性。 8. **TCP特性**:TCP是一种面向连接的、可靠的传输协议,它保证了数据的有序和无损传输。这意味着每个数据包都会被确认,如果数据...

    NioSocket,包括server端和client端

    NioSocket是一个基于Java NIO(非阻塞I/O)技术实现的网络通信框架,它包含服务器端(Server)和客户端(Client)两部分。在Java编程中,NIO(New Input/Output)提供了一种不同于传统IO模型的I/O操作方式,其核心...

    java socket,javasocket教程

    2. **NIO(非阻塞I/O)** - Java NIO库(New I/O)提供了一种新的I/O模型,可以在单线程中高效处理多个连接,适合大量并发的服务器。 3. **SSL/TLS安全通信** - Java Socket支持SSL/TLS协议,通过`SSLSocket`和`...

    JAVA电子白板即时通信程序

    1. **网络编程**:使用Java的Socket编程或者更高层次的Java NIO(非阻塞I/O)来实现客户端和服务器之间的通信。 2. **协议设计**:定义一套通信协议,包括消息格式、命令编码等,确保数据在不同客户端之间正确传输...

    JAVAsocket接口调试程序

    - **NIO**:使用Java的非阻塞I/O(New IO)API,提高并发性能。 - **线程池**:使用线程池管理客户端连接,提高资源利用率。 通过以上对Java Socket接口调试程序的分析,开发者可以更好地理解和掌握网络通信的...

    javaSocket共16页.pdf.zip

    Java Socket编程也支持异步操作,即非阻塞I/O。这通常通过NIO(New IO)库实现,包括`Selector`、`Channel`和`Buffer`等组件。NIO允许程序同时处理多个连接,提高了服务器的并发性能。 除此之外,Java Socket编程还...

    JAVA Socket 经典教程

    - **NIO(非阻塞I/O)**:Java NIO库提供了非阻塞的Socket编程模型,适用于高并发场景,如`Selector`和`Channel`。 7. **实践应用** - **聊天应用**:基于Socket的简单聊天室,客户端发送消息到服务器,服务器...

    Java做的socket聊天系统

    针对大量用户时,可以考虑使用NIO(非阻塞I/O)来提高服务器的处理能力,或者采用负载均衡策略分散服务器压力。 以上就是关于“Java做的socket聊天系统”的主要知识点,涵盖了Socket通信的基础、TCP协议、服务器和...

    java Socket 好处

    2. **阻塞与非阻塞模式**:Socket默认采用阻塞模式,可以通过NIO(非阻塞I/O)来提高性能,减少线程的阻塞时间。 3. **套接字选项**:Socket类提供了`setSoTimeout()`、`setKeepAlive()`等方法,可以设置超时、保持...

    Java_TCP_IP_Socket编程(原书第2版)

    9. **NIO(非阻塞I/O)**:Java NIO(New I/O)提供了一种非阻塞的I/O模型,通过Selector和Channel等组件,可以高效地处理大量并发连接,尤其适用于高并发的服务器应用。 10. **套接字选项与网络参数**:Java的...

    java网络编程 socket

    - NIO(非阻塞I/O):Java的非阻塞I/O模型可以提高高并发场景下的性能,如使用Selector监控多个Socket的状态。 - SSL/TLS:对于需要加密传输的场景,可以使用Java的SSL套接字提供安全的网络通信。 通过阅读《Java...

    JavaSocket编程(二)Java面向连接的类.pdf

    最后,Java NIO(New IO)库提供了`java.nio.channels.SocketChannel`和`java.nio.channels.ServerSocketChannel`,它们提供了非阻塞I/O和选择器,适用于高并发的网络应用。不过,这超出了基本的`Socket`编程范围,...

    JAVA Socket教程

    Java Socket教程是学习Java网络编程的重要部分,它允许两台机器通过网络进行通信。...随着对Socket编程的深入,你还可以探索更复杂的应用场景,如SSL/TLS加密通信、套接字选项配置、NIO(非阻塞I/O)等高级主题。

    三步学会Java Socket编程.doc

    在实际应用中,可能会涉及到多线程处理多个客户端连接,或者使用更高级的I/O模型如NIO(非阻塞I/O)来提高性能。此外,异常处理是必不可少的,确保在网络通信过程中能正确处理各种可能出现的问题。 通过以上步骤,...

    javasocket.rar_网络编程_Java_

    Java Socket编程也支持异步通信,通过使用Selector和NIO(非阻塞I/O)类,可以同时处理多个客户端连接。此外,Socket编程还可以结合多线程技术,让服务器同时服务多个客户端。 在实际应用中,Java Socket常用于构建...

    Java Socket实现文件传输示例代码

    Java Socket是Java网络编程中的基础组件,用于实现两个应用程序...此外,对于大型文件,可能会使用缓冲区池和NIO(非阻塞I/O)来提高效率。在实际开发中,还需要考虑如何优雅地关闭资源,避免内存泄漏和资源耗尽问题。

    java socket长连接客户端服务端(标准实例)

    1. 异步处理:在实际应用中,服务端可能需要同时处理多个客户端连接,可以使用多线程或者Java NIO来实现非阻塞I/O。 2. 错误处理:确保对IOException和其他可能的异常进行妥善处理,避免程序中断。 3. 安全性:考虑...

Global site tag (gtag.js) - Google Analytics