Java网络编程——使用NIO实现非阻塞Socket通信
来源:互联网 发布:淘宝ps切片工具怎么用 编辑:程序博客网 时间:2024/05/22 01:35
除了普通的Socket与ServerSocket实现的阻塞式通信外,java提供了非阻塞式通信的NIO API。先看一下NIO的实现原理。
从图中可以看出,服务器上所有Channel(包括ServerSocketChannel和SocketChannel)都需要向Selector注册,而该Selector则负责监视这些Socket的IO状态,当其中任意一个或者多个Channel具有可用的IO操作时,该Selector的select()方法将会返回大于0的整数,该整数值就表示该Selector上有多少个Channel具有可用的IO操作,并提供了selectedKeys()方法来返回这些Channel对应的SelectionKey集合。正是通过Selector,使得服务器端只需要不断地调用Selector实例的select()方法即可知道当前所有Channel是否有需要处理的IO操作。
看个demo
NClient.java
import java.io.IOException;import java.net.InetSocketAddress;import java.nio.ByteBuffer;import java.nio.channels.SelectionKey;import java.nio.channels.Selector;import java.nio.channels.SocketChannel;import java.nio.charset.Charset;import java.util.Scanner;public class NClient {//定义检测SocketChannel的Selector对象private Selector selector=null;//定义处理编码和解码的字符集private Charset charset=Charset.forName("UTF-8");//客户端SocketChannelprivate SocketChannel sc=null;public void init() throws IOException{selector=Selector.open();InetSocketAddress isa=new InetSocketAddress("127.0.0.1",30000);//调用open静态方法创建连接到指定主机的SocketChannelsc=SocketChannel.open(isa);//设置该sc以非阻塞方式工作sc.configureBlocking(false);//将Socketchannel对象注册到指定Selectorsc.register(selector, SelectionKey.OP_READ);//启动读取服务器端数据的线程new ClientThread().start();//创建键盘输入流Scanner scan=new Scanner(System.in);while(scan.hasNextLine()){//读取键盘输入String line=scan.nextLine();//将键盘输入的内容输出到SocketChannel中sc.write(charset.encode(line));}}//定义读取服务器数据的线程private class ClientThread extends Thread{public void run(){try{while(selector.select()>0){//遍历每个有可用IO操作Channel对应的SelectionKeyfor(SelectionKey sk:selector.selectedKeys()){//删除正在处理的SelectionKeyselector.selectedKeys().remove(sk);//如果该SelectionKey对应的Channel中有可读的数据if(sk.isReadable()){//使用NIO读取channel中的数据SocketChannel sc=(SocketChannel) sk.channel();ByteBuffer buff=ByteBuffer.allocate(1024);String content="";while(sc.read(buff)>0){//sc.read(buff);buff.flip();content+=charset.decode(buff);}//打印输出读取的内容System.out.println("聊天信息"+content);//为下一次读取做准备sk.interestOps(SelectionKey.OP_READ);}}}}catch(IOException ex){ex.printStackTrace();}}}public static void main(String[]args) throws IOException{new NClient().init();}}
NServer.java
import java.io.IOException;import java.net.InetSocketAddress;import java.nio.ByteBuffer;import java.nio.channels.Channel;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;public class NServer {//用于检测所有Channel状态的Selectorprivate Selector selector=null;//定义实现编码、解码的字符集对象private Charset charset=Charset.forName("UTF-8");public void init() throws IOException{selector=Selector.open();//通过open方法来打开一个未绑定的ServerSocketChannel实例ServerSocketChannel server=ServerSocketChannel.open();InetSocketAddress isa=new InetSocketAddress("127.0.0.1",30000);//将该ServerSocketChannel绑定到指定ip地址server.socket().bind(isa);//设置ServerSocket以非阻塞方式工作server.configureBlocking(false);//将server注册到指定Selector对象server.register(selector, SelectionKey.OP_ACCEPT);while(selector.select()>0){//依次处理selector上的每个已选择的SelectionKeyfor(SelectionKey sk:selector.selectedKeys()){//从selector上的已选择Key集中删除正在处理的SelectionKeyselector.selectedKeys().remove(sk);//如果sk对应的通信包含客户端的连接请求if(sk.isAcceptable()){//调用accept方法接受连接,产生服务器端对应的SocketChannelSocketChannel sc=server.accept();//设置采用非阻塞模式sc.configureBlocking(false);sc.register(selector, SelectionKey.OP_READ);//将sk对应的Channel设置成准备接受其他请求sk.interestOps(SelectionKey.OP_ACCEPT);}//如果sk对应的通道有数据需要读取if(sk.isReadable()){//获取该SelectionKey对应的Channel,该Channel中有可读的数据SocketChannel sc=(SocketChannel) sk.channel();//定义准备之星读取数据的ByteBufferByteBuffer buff=ByteBuffer.allocate(1024);String content="";//开始读取数据try{while(sc.read(buff)>0){buff.flip();content+=charset.decode(buff);}//打印从该sk对应的Channel里读到的数据System.out.println("=========="+content);//将sk对应的Channel设置成准备下一次读取sk.interestOps(SelectionKey.OP_READ);//如果捕捉到该sk对应的channel出现异常,即表明该channel对应的client出现了//异常,所以从selector中取消sk的注册}catch(IOException e){//从Selector中删除指定的SelectionKeysk.cancel();if(sk.channel()!=null){sk.channel().close();}}//如果content的长度大于0,即聊天信息不为空if(content.length()>0){//遍历该selector里注册的所有SelectKeyfor(SelectionKey key:selector.keys()){//选取该key对应的ChannelChannel targetChannel=key.channel();//如果该channel是SocketChannel对象if(targetChannel instanceof SocketChannel){//将独到的内容写入该Channel中SocketChannel dest=(SocketChannel) targetChannel;dest.write(charset.encode(content));}}}}}}}public static void main(String[]args) throws IOException{new NServer().init();}}
通过java提供的NIO实现非阻塞Socket通信,大大提高了网络服务器的性能。
- Java网络编程——使用NIO实现非阻塞Socket通信
- Java网络编程——使用NIO实现非阻塞Socket通信
- Java网络编程——使用NIO实现非阻塞Socket通信
- 使用NIO实现非阻塞Socket通信
- 使用NIO实现非阻塞Socket通信
- Thinking in Java--使用NIO实现非阻塞Socket通信
- java NIO 实现非阻塞socket通信
- Java网络编程之(三): TCP协议使用NIO实现非阻塞Soket通信
- Java网络编程之(三): TCP协议使用NIO实现非阻塞Soket通信
- 使用NIO实现非阻塞Socket通信原理
- java nio实现非阻塞Socket通信实例
- JAVA 网络编程(2) SOCKET 非阻塞NIO示例
- NIO Socket实现非阻塞通信示例
- NIO实现非阻塞Socket通信
- Java NIO 非阻塞socket通信案例
- 使用NIO实现非阻塞式的网络通信
- 基于NIO实现非阻塞Socket编程
- 基于NIO实现非阻塞Socket编程
- Win32 Desktop Apps (Visual C++)
- uva 11464 - Even Parity(暴力枚举)
- 摩根斯丹利(大摩)面试经历
- 有关T-SQL的10个好习惯
- poj 1502 MPI Maelstrom
- Java网络编程——使用NIO实现非阻塞Socket通信
- 将一个整数转换为若干连续整数的和
- java web开发学习-7 LCDS
- Android FrameBuffer 中绘图的简单例子
- Android中的“再按一次返回键退出程序”实现
- java web开发学习-8 BlazeDS
- C++ 打印各种图形
- 社交的良好方式
- java web开发学习-9 JavaScript