`
chenHZ123
  • 浏览: 4099 次
  • 性别: Icon_minigender_1
  • 来自: 北京
社区版块
存档分类
最新评论

java NIO (4)

 
阅读更多

在上一篇文章中介绍了关于缓冲区的一些细节内容,现在终于可以进入NIO中最有意思的部分非阻塞I/O。通常在进行同步I/O操作时,如果读取数据,代码会阻塞直至有 可供读取的数据。同样,写入调用将会阻塞直至数据能够写入。传统的Server/Client模式会基于TPR(Thread per Request),服务器会为每个客户端请求建立一个线程,由该线程单独负责处理一个客户请求。这种模式带来的一个问题就是线程数量的剧增,大量的线程会增大服务器的开销。大多数的实现为了避免这个问题,都采用了线程池模型,并设置线程池线程的最大数量,这由带来了新的问题,如果线程池中有200个线程,而有200个用户都在进行大文件下载,会导致第201个用户的请求无法及时处理,即便第201个用户只想请求一个几KB大小的页面。传统的 Server/Client模式如下图所示:

 

NIO中非阻塞I/O采用了基于Reactor模式的工作方式,I/O调用不会被阻塞,相反是注册感兴趣的特定I/O事件,如可读数据到达,新的套接字连接等等,在发生特定事件时,系统再通知我们。NIO中实现非阻塞I/O的核心对象就是Selector,Selector就是注册各种I/O事件地 方,而且当那些事件发生时,就是这个对象告诉我们所发生的事件,如下图所示:

从图中可以看出,当有读或写等任何注册的事件发生时,可以从Selector中获得相应的SelectionKey,同时从 SelectionKey中可以找到发生的事件和该事件所发生的具体的SelectableChannel,以获得客户端发送过来的数据。关于 SelectableChannel的可以参考Java NIO使用及原理分析(一)

 

使用NIO中非阻塞I/O编写服务器处理程序,大体上可以分为下面三个步骤:

1. 向Selector对象注册感兴趣的事件 
2. 从Selector中获取感兴趣的事件 
3. 根据不同的事件进行相应的处理

接下来我们用一个简单的示例来说明整个过程。首先是向Selector对象注册感兴趣的事件:

  1. /* 
  2.  * 注册事件 
  3.  * */  
  4. protected Selector getSelector() throws IOException {  
  5.     // 创建Selector对象  
  6.     Selector sel = Selector.open();  
  7.       
  8.     // 创建可选择通道,并配置为非阻塞模式  
  9.     ServerSocketChannel server = ServerSocketChannel.open();  
  10.     server.configureBlocking(false);  
  11.       
  12.     // 绑定通道到指定端口  
  13.     ServerSocket socket = server.socket();  
  14.     InetSocketAddress address = new InetSocketAddress(port);  
  15.     socket.bind(address);  
  16.       
  17.     // 向Selector中注册感兴趣的事件  
  18.     server.register(sel, SelectionKey.OP_ACCEPT);   
  19.     return sel;  
  20. }  

创建了ServerSocketChannel对象,并调用configureBlocking()方法,配置为非阻塞模式,接下来的三行代码把该通道绑定到指定端口,最后向Selector中注册事件,此处指定的是参数是OP_ACCEPT,即指定我们想要监听accept事件,也就是新的连接发 生时所产生的事件,对于ServerSocketChannel通道来说,我们唯一可以指定的参数就是OP_ACCEPT。

从Selector中获取感兴趣的事件,即开始监听,进入内部循环:

  1. /* 
  2.  * 开始监听 
  3.  * */   
  4. public void listen() {   
  5.     System.out.println("listen on " + port);  
  6.     try {   
  7.         while(true) {   
  8.             // 该调用会阻塞,直到至少有一个事件发生  
  9.             selector.select();   
  10.             Set<SelectionKey> keys = selector.selectedKeys();  
  11.             Iterator<SelectionKey> iter = keys.iterator();  
  12.             while (iter.hasNext()) {   
  13.                 SelectionKey key = (SelectionKey) iter.next();   
  14.                 iter.remove();   
  15.                 process(key);   
  16.             }   
  17.         }   
  18.     } catch (IOException e) {   
  19.         e.printStackTrace();  
  20.     }   
  21. }  

在非阻塞I/O中,内部循环模式基本都是遵循这种方式。首先调用select()方法,该方法会阻塞,直到至少有一个事件发生,然后再使用selectedKeys()方法获取发生事件的SelectionKey,再使用迭代器进行循环。

最后一步就是根据不同的事件,编写相应的处理代码:

 

  1. /* 
  2.  * 根据不同的事件做处理 
  3.  * */  
  4. protected void process(SelectionKey key) throws IOException{  
  5.     // 接收请求  
  6.     if (key.isAcceptable()) {  
  7.         ServerSocketChannel server = (ServerSocketChannel) key.channel();  
  8.         SocketChannel channel = server.accept();  
  9.         channel.configureBlocking(false);  
  10.         channel.register(selector, SelectionKey.OP_READ);  
  11.     }  
  12.     // 读信息  
  13.     else if (key.isReadable()) {  
  14.         SocketChannel channel = (SocketChannel) key.channel();   
  15.         int count = channel.read(buffer);   
  16.         if (count > 0) {   
  17.             buffer.flip();   
  18.             CharBuffer charBuffer = decoder.decode(buffer);   
  19.             name = charBuffer.toString();   
  20.             SelectionKey sKey = channel.register(selector, SelectionKey.OP_WRITE);   
  21.             sKey.attach(name);   
  22.         } else {   
  23.             channel.close();   
  24.         }   
  25.         buffer.clear();   
  26.     }  
  27.     // 写事件  
  28.     else if (key.isWritable()) {  
  29.         SocketChannel channel = (SocketChannel) key.channel();   
  30.         String name = (String) key.attachment();   
  31.           
  32.         ByteBuffer block = encoder.encode(CharBuffer.wrap("Hello " + name));   
  33.         if(block != null)  
  34.         {  
  35.             channel.write(block);  
  36.         }  
  37.         else  
  38.         {  
  39.             channel.close();  
  40.         }  
  41.   
  42.      }  
  43. }  

 

此处分别判断是接受请求、读数据还是写事件,分别作不同的处理。

到这里关于Java NIO使用及原理分析的四篇文章就全部完成了。Java NIO提供了通道、缓冲区、选择器这样一组抽象概念,极大的简化了我们编写高性能并发型服务器程序,后面有机会我会继续谈谈使用Java NIO的一些想法。

分享到:
评论

相关推荐

    java NIO和java并发编程的书籍

    java NIO和java并发编程的书籍java NIO和java并发编程的书籍java NIO和java并发编程的书籍java NIO和java并发编程的书籍java NIO和java并发编程的书籍java NIO和java并发编程的书籍java NIO和java并发编程的书籍java...

    JavaNIO chm帮助文档

    Java NIO系列教程(一) Java NIO 概述 Java NIO系列教程(二) Channel Java NIO系列教程(三) Buffer Java NIO系列教程(四) Scatter/Gather Java NIO系列教程(五) 通道之间的数据传输 Java NIO系列教程(六)...

    Java NIO英文高清原版

    Java NIO英文高清原版

    java nio 包读取超大数据文件

    Java nio 超大数据文件 超大数据文件Java nio 超大数据文件 超大数据文件Java nio 超大数据文件 超大数据文件Java nio 超大数据文件 超大数据文件Java nio 超大数据文件 超大数据文件Java nio 超大数据文件 超大数据...

    java NIO.zip

    java NIO.zip

    Java NIO 中文 Java NIO 中文 Java NIO 中文文档

    Java NIO 深入探讨了 1.4 版的 I/O 新特性,并告诉您如何使用这些特性来极大地提升您所写的 Java 代码的执行效率。这本小册子就程序员所面临的有代表性的 I/O 问题作了详尽阐述,并讲解了 如何才能充分利用新的 I/O ...

    java NIO 中文版

    讲解了 JavaIO 与 JAVA NIO区别,JAVA NIO设计理念,以及JDK中java NIO中语法的使用

    java NIO 视频教程

    Java NIO(New IO)是一个可以替代标准Java IO API的IO API(从Java 1.4开始),Java NIO提供了与标准IO不同的IO工作方式。 Java NIO: Channels and Buffers(通道和缓冲区) 标准的IO基于字节流和字符流进行操作的,...

    java nio 实现socket

    java nio 实现socketjava nio 实现socketjava nio 实现socketjava nio 实现socketjava nio 实现socket

    java nio中文版

    java NIO是 java New IO 的简称,在 jdk1.4 里提供的新 api 。 Sun 官方标榜的特性如下: – 为所有的原始类型提供 (Buffer) 缓存支持。 – 字符集编码解码解决方案。 – Channel :一个新的原始 I/O 抽象。 – 支持...

    Java Nio selector例程

    java侧起server(NioUdpServer1.java),基于Java Nio的selector 阻塞等候,一个android app(NioUdpClient1文件夹)和一个java程序(UI.java)作为两个client分别向该server发数据,server收到后分别打印收到的消息...

    java NIO技巧及原理

    java NIO技巧及原理解析,java IO原理,NIO框架分析,性能比较

    java nio 读文件

    java nio 读文件,java nio 读文件

    java基于NIO实现Reactor模型源码.zip

    java基于NIO实现Reactor模型源码java基于NIO实现Reactor模型源码java基于NIO实现Reactor模型源码java基于NIO实现Reactor模型源码java基于NIO实现Reactor模型源码java基于NIO实现Reactor模型源码java基于NIO实现...

    JAVA NIO 学习资料

    JAVA NIO学习资料JAVA NIO学习资料

    Java NIO测试示例

    Java NIO测试示例

    基于Java NIO实现五子棋游戏.zip

    基于Java NIO实现五子棋游戏.zip基于Java NIO实现五子棋游戏.zip 基于Java NIO实现五子棋游戏.zip基于Java NIO实现五子棋游戏.zip 基于Java NIO实现五子棋游戏.zip基于Java NIO实现五子棋游戏.zip 基于Java NIO实现...

    Java NIO.pdf

    java nio编程 非阻塞模式的通信 电子书 带目录标签

    Java NIO 英文文字版

    Many serious Java programmers, especially enterprise Java programmers, consider the new I/O API--called NIO for New Input/Output--the most important feature in the 1.4 version of the Java 2 Standard ...

    java nio入门学习,两个pdf

    java nio入门学习,两个pdfjava nio入门学习,两个pdf

Global site tag (gtag.js) - Google Analytics