Java NIO原理图文分析及代码实现

前言:

最近在分析hadoop的RPC(Remote Procedure Call Protocol ,远程过程调用协议,它是一种通过网络从远程计算机程序上请求服务,而不需要了解底层网络技术的协议。可以参考:http://baike.baidu.com/view/32726.htm )机制时,发现hadoop的RPC机制的实现主要用到了两个技术:动态代理(动态代理可以参考博客:http://weixiaolu.iteye.com/blog/1477774 )和java NIO。为了能够正确地分析hadoop的RPC源码,我觉得很有必要先研究一下java NIO的原理和具体实现。

这篇博客我主要从两个方向来分析java NIO

目录:

一.java NIO 和阻塞I/O的区别
     1. 阻塞I/O通信模型
     2. java NIO原理及通信模型
二.java NIO服务端和客户端代码实现

具体分析:

一.java NIO 和阻塞I/O的区别

1. 阻塞I/O通信模型

假如现在你对阻塞I/O已有了一定了解,我们知道阻塞I/O在调用InputStream.read()方法时是阻塞的,它会一直等到数据到来时(或超时)才会返回;同样,在调用ServerSocket.accept()方法时,也会一直阻塞到有客户端连接才会返回,每个客户端连接过来后,服务端都会启动一个线程去处理该客户端的请求。阻塞I/O的通信模型示意图如下:

如果你细细分析,一定会发现阻塞I/O存在一些缺点。根据阻塞I/O通信模型,我总结了它的两点缺点:

1. 当客户端多时,会创建大量的处理线程。且每个线程都要占用栈空间和一些CPU时间

2. 阻塞可能带来频繁的上下文切换,且大部分上下文切换可能是无意义的。

在这种情况下非阻塞式I/O就有了它的应用前景。

2. java NIO原理及通信模型

Java NIO是在jdk1.4开始使用的,它既可以说成“新I/O”,也可以说成非阻塞式I/O。下面是java NIO的工作原理:

1. 由一个专门的线程来处理所有的 IO 事件,并负责分发。
2. 事件驱动机制:事件到的时候触发,而不是同步的去监视事件。
3. 线程通讯:线程之间通过 wait,notify 等方式通讯。保证每次上下文切换都是有意义的。减少无谓的线程切换。

阅读过一些资料之后,下面贴出我理解的java NIO的工作原理图:

(注:每个线程的处理流程大概都是读取数据、解码、计算处理、编码、发送响应。)

Java NIO的服务端只需启动一个专门的线程来处理所有的 IO 事件,这种通信模型是怎么实现的呢?呵呵,我们一起来探究它的奥秘吧。java NIO采用了双向通道(channel)进行数据传输,而不是单向的流(stream),在通道上可以注册我们感兴趣的事件。一共有以下四种事件:

事件名 对应值
服务端接收客户端连接事件 SelectionKey.OP_ACCEPT(16)
客户端连接服务端事件 SelectionKey.OP_CONNECT(8)
读事件 SelectionKey.OP_READ(1)
写事件 SelectionKey.OP_WRITE(4)

服务端和客户端各自维护一个管理通道的对象,我们称之为selector,该对象能检测一个或多个通道 (channel) 上的事件。我们以服务端为例,如果服务端的selector上注册了读事件,某时刻客户端给服务端发送了一些数据,阻塞I/O这时会调用read()方法阻塞地读取数据,而NIO的服务端会在selector中添加一个读事件。服务端的处理线程会轮询地访问selector,如果访问selector时发现有感兴趣的事件到达,则处理这些事件,如果没有感兴趣的事件到达,则处理线程会一直阻塞直到感兴趣的事件到达为止。下面是我理解的java NIO的通信模型示意图:

 二.java NIO服务端和客户端代码实现

为了更好地理解java NIO,下面贴出服务端和客户端的简单代码实现。

服务端:

package cn.nio; 

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.ServerSocketChannel;
import java.nio.channels.SocketChannel;
import java.util.Iterator; 

/**
 * NIO服务端
 * @author 小路
 */
public class NIOServer {
 //通道管理器
 private Selector selector; 

 /**
  * 获得一个ServerSocket通道,并对该通道做一些初始化的工作
  * @param port 绑定的端口号
  * @throws IOException
  */
 public void initServer(int port) throws IOException {
  // 获得一个ServerSocket通道
  ServerSocketChannel serverChannel = ServerSocketChannel.open();
  // 设置通道为非阻塞
  serverChannel.configureBlocking(false);
  // 将该通道对应的ServerSocket绑定到port端口
  serverChannel.socket().bind(new InetSocketAddress(port));
  // 获得一个通道管理器
  this.selector = Selector.open();
  //将通道管理器和该通道绑定,并为该通道注册SelectionKey.OP_ACCEPT事件,注册该事件后,
  //当该事件到达时,selector.select()会返回,如果该事件没到达selector.select()会一直阻塞。
  serverChannel.register(selector, SelectionKey.OP_ACCEPT);
 } 

 /**
  * 采用轮询的方式监听selector上是否有需要处理的事件,如果有,则进行处理
  * @throws IOException
  */
 @SuppressWarnings("unchecked")
 public void listen() throws IOException {
  System.out.println("服务端启动成功!");
  // 轮询访问selector
  while (true) {
   //当注册的事件到达时,方法返回;否则,该方法会一直阻塞
   selector.select();
   // 获得selector中选中的项的迭代器,选中的项为注册的事件
   Iterator ite = this.selector.selectedKeys().iterator();
   while (ite.hasNext()) {
    SelectionKey key = (SelectionKey) ite.next();
    // 删除已选的key,以防重复处理
    ite.remove();
    // 客户端请求连接事件
    if (key.isAcceptable()) {
     ServerSocketChannel server = (ServerSocketChannel) key
       .channel();
     // 获得和客户端连接的通道
     SocketChannel channel = server.accept();
     // 设置成非阻塞
     channel.configureBlocking(false); 

     //在这里可以给客户端发送信息哦
     channel.write(ByteBuffer.wrap(new String("向客户端发送了一条信息").getBytes()));
     //在和客户端连接成功之后,为了可以接收到客户端的信息,需要给通道设置读的权限。
     channel.register(this.selector, SelectionKey.OP_READ); 

     // 获得了可读的事件
    } else if (key.isReadable()) {
      read(key);
    } 

   } 

  }
 }
 /**
  * 处理读取客户端发来的信息 的事件
  * @param key
  * @throws IOException
  */
 public void read(SelectionKey key) throws IOException{
  // 服务器可读取消息:得到事件发生的Socket通道
  SocketChannel channel = (SocketChannel) key.channel();
  // 创建读取的缓冲区
  ByteBuffer buffer = ByteBuffer.allocate(10);
  channel.read(buffer);
  byte[] data = buffer.array();
  String msg = new String(data).trim();
  System.out.println("服务端收到信息:"+msg);
  ByteBuffer outBuffer = ByteBuffer.wrap(msg.getBytes());
  channel.write(outBuffer);// 将消息回送给客户端
 } 

 /**
  * 启动服务端测试
  * @throws IOException
  */
 public static void main(String[] args) throws IOException {
  NIOServer server = new NIOServer();
  server.initServer(8000);
  server.listen();
 } 

}

客户端:

package cn.nio; 

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.util.Iterator; 

/**
 * NIO客户端
 * @author 小路
 */
public class NIOClient {
 //通道管理器
 private Selector selector; 

 /**
  * 获得一个Socket通道,并对该通道做一些初始化的工作
  * @param ip 连接的服务器的ip
  * @param port 连接的服务器的端口号
  * @throws IOException
  */
 public void initClient(String ip,int port) throws IOException {
  // 获得一个Socket通道
  SocketChannel channel = SocketChannel.open();
  // 设置通道为非阻塞
  channel.configureBlocking(false);
  // 获得一个通道管理器
  this.selector = Selector.open(); 

  // 客户端连接服务器,其实方法执行并没有实现连接,需要在listen()方法中调
  //用channel.finishConnect();才能完成连接
  channel.connect(new InetSocketAddress(ip,port));
  //将通道管理器和该通道绑定,并为该通道注册SelectionKey.OP_CONNECT事件。
  channel.register(selector, SelectionKey.OP_CONNECT);
 } 

 /**
  * 采用轮询的方式监听selector上是否有需要处理的事件,如果有,则进行处理
  * @throws IOException
  */
 @SuppressWarnings("unchecked")
 public void listen() throws IOException {
  // 轮询访问selector
  while (true) {
   selector.select();
   // 获得selector中选中的项的迭代器
   Iterator ite = this.selector.selectedKeys().iterator();
   while (ite.hasNext()) {
    SelectionKey key = (SelectionKey) ite.next();
    // 删除已选的key,以防重复处理
    ite.remove();
    // 连接事件发生
    if (key.isConnectable()) {
     SocketChannel channel = (SocketChannel) key
       .channel();
     // 如果正在连接,则完成连接
     if(channel.isConnectionPending()){
      channel.finishConnect(); 

     }
     // 设置成非阻塞
     channel.configureBlocking(false); 

     //在这里可以给服务端发送信息哦
     channel.write(ByteBuffer.wrap(new String("向服务端发送了一条信息").getBytes()));
     //在和服务端连接成功之后,为了可以接收到服务端的信息,需要给通道设置读的权限。
     channel.register(this.selector, SelectionKey.OP_READ); 

     // 获得了可读的事件
    } else if (key.isReadable()) {
      read(key);
    } 

   } 

  }
 }
 /**
  * 处理读取服务端发来的信息 的事件
  * @param key
  * @throws IOException
  */
 public void read(SelectionKey key) throws IOException{
  //和服务端的read方法一样
 } 

 /**
  * 启动客户端测试
  * @throws IOException
  */
 public static void main(String[] args) throws IOException {
  NIOClient client = new NIOClient();
  client.initClient("localhost",8000);
  client.listen();
 } 

}

小结:

终于把动态代理和java NIO分析完了,呵呵,下面就要分析hadoop的RPC机制源码了,博客地址:http://weixiaolu.iteye.com/blog/1504898 。不过如果对java NIO的理解存在异议的,欢迎一起讨论。
如需转载,请注明出处:http://weixiaolu.iteye.com/blog/1479656

(0)

相关推荐

  • Java Socket编程实例(四)- NIO TCP实践

    一.回传协议接口和TCP方式实现: 1.接口: import java.nio.channels.SelectionKey; import java.io.IOException; public interface EchoProtocol { void handleAccept(SelectionKey key) throws IOException; void handleRead(SelectionKey key) throws IOException; void handleWrite(

  • Java NIO和IO的区别

    下表总结了Java NIO和IO之间的主要差别,我会更详细地描述表中每部分的差异. 复制代码 代码如下: IO                NIO面向流            面向缓冲阻塞IO            非阻塞IO无                选择器 面向流与面向缓冲 Java NIO和IO之间第一个最大的区别是,IO是面向流的,NIO是面向缓冲区的. Java IO面向流意味着每次从流中读一个或多个字节,直至读取所有字节,它们没有被缓存在任何地方.此外,它不能前后移动流中的数

  • Java NIO工作原理的全面分析

    ◆  输入/输出:概念性描述I/O 简介I/O ? 或者输入/输出 ? 指的是计算机与外部世界或者一个程序与计算机的其余部分的之间的接口.它对于任何计算机系统都非常关键,因而所有 I/O 的主体实际上是内置在操作系统中的.单独的程序一般是让系统为它们完成大部分的工作.在 Java 编程中,直到最近一直使用 流 的方式完成 I/O.所有 I/O 都被视为单个的字节的移动,通过一个称为 Stream 的对象一次移动一个字节.流 I/O 用于与外部世界接触.它也在内部使用,用于将对象转换为字节,然后再

  • Java 高并发八:NIO和AIO详解

    IO感觉上和多线程并没有多大关系,但是NIO改变了线程在应用层面使用的方式,也解决了一些实际的困难.而AIO是异步IO和前面的系列也有点关系.在此,为了学习和记录,也写一篇文章来介绍NIO和AIO. 1. 什么是NIO NIO是New I/O的简称,与旧式的基于流的I/O方法相对,从名字看,它表示新的一套Java I/O标 准.它是在Java 1.4中被纳入到JDK中的,并具有以下特性: NIO是基于块(Block)的,它以块为基本单位处理数据 (硬盘上存储的单位也是按Block来存储,这样性能

  • java NIO 详解

    Java NIO提供了与标准IO不同的IO工作方式: Channels and Buffers(通道和缓冲区):标准的IO基于字节流和字符流进行操作的,而NIO是基于通道(Channel)和缓冲区(Buffer)进行操作,数据总是从通道读取到缓冲区中,或者从缓冲区写入到通道中. Asynchronous IO(异步IO):Java NIO可以让你异步的使用IO,例如:当线程从通道读取数据到缓冲区时,线程还是可以进行其他事情.当数据被写入到缓冲区时,线程可以继续处理它.从缓冲区写入通道也类似. S

  • Java 对象序列化 NIO NIO2详细介绍及解析

    Java 对象序列化 NIO NIO2详细介绍及解析 概要: 对象序列化 对象序列化机制允许把内存中的Java对象转换成与平台无关的二进制流,从而可以保存到磁盘或者进行网络传输,其它程序获得这个二进制流后可以将其恢复成原来的Java对象. 序列化机制可以使对象可以脱离程序的运行而对立存在 序列化的含义和意义 序列化 序列化机制可以使对象可以脱离程序的运行而对立存在 序列化(Serialize)指将一个java对象写入IO流中,与此对应的是,对象的反序列化(Deserialize)则指从IO流中恢

  • 支撑Java NIO与NodeJS的底层技术

    支撑Java NIO 与 NodeJS的底层技术 众所周知在近几个版本的Java中增加了一些对Java NIO.NIO2的支持,与此同时NodeJS技术栈中最为人称道的优势之一就是其高性能IO,那么我们今天要讨论的话题就是支撑这些技术的底层技术. 开始之前先要提出的一个问题是: 为什么NodeJS和Java NIO2没有在更早的时间出现? 答案:个人认为是底层的支撑技术还不成熟. 那么,底层技术指的是什么呢?对的,我想很多人已经猜到,是操作系统技术.本文提出的两个概念Java NIO2和Node

  • Java的NIO与IO的详解及对比

    Java的NIO与IO的区别 NIO是JDK1.4引入的异步IO,NIO核心部分就是三点: Channel Buffer Selector NIO与IO对比 NIO与IO的区别,总体上来说体现在三个方面: IO 基于流(Stream oriented), 而 NIO 基于 Buffer (Buffer oriented) IO 操作是阻塞的, 而 NIO 操作是非阻塞的 IO 没有 selector 概念, 而 NIO 有 selector 概念. 基于 Stream 与基于 Buffer 传统

  • java使用nio2拷贝文件的示例

    这个程序只是为了更方便的进行拷贝文件(夹)而创造.1.可以不用新建文件夹,就像windows的复制粘贴一样简单.2.有简单的出错重连机制3.不需要重复拷贝,差异化复制文件.4.拷贝文件夹的时候可以不用复制全路径,只关注需要拷贝的文件夹.5.程序做了简单的必要检查,效率也不算低.6.使用的是7的nio2的新API. 复制代码 代码如下: import java.io.IOException;import java.nio.file.FileVisitResult;import java.nio.f

  • Java Socket编程实例(五)- NIO UDP实践

    一.回传协议接口和UDP方式实现: 1.接口: import java.nio.channels.SelectionKey; import java.io.IOException; public interface EchoProtocol { void handleAccept(SelectionKey key) throws IOException; void handleRead(SelectionKey key) throws IOException; void handleWrite(

随机推荐