BIO、NIO、AIO 介绍和适用场景分析

IO的方式一般分为几种,同步阻塞的BIO、同步非阻塞的NIO、异步非阻塞的AIO。java

1、同步阻塞的BIOweb

在JDK1.4以前,咱们创建网络链接的时候采用BIO模式,须要先在服务端启动一个serverSocket,而后在客户端启动socket来对服务端进行通讯,默认状况下服务端须要对每一个请求创建一堆线程等待请求,而客户端发送请求后,先咨询服务端是否有线程相应,若是没有则会一直等待或者遭到拒绝请求,若是有的话,客户端线程会等待请求结束后才继续执行。编程

2、同步非阻塞的NIO后端

NIO自己是基于事件驱动思想来完成的,其主要想解决的是BIO的大并发问题,在使用同步I/O的网络应用中,若是要同时处理多个客户端请求,或是在客户端要同时和多个服务器进行通信,就必须使用多线程来处理。也就是说,将每个客户端请求分配给一个线程来单独处理。这样作虽然能够达到咱们的要求,但同时又带来另一个问题。因为每建立一个线程,就要为这个线程分配必定的内存空间,并且操做系统自己对线程的总数有必定的限制。若是客户端的请求过多,服务端程序可能会由于不堪重负而拒绝客户端的请求,甚至服务器可能会所以而瘫痪。服务器

NIO基于Reactor,当socket有流可读或可写入socket时,操做系统会相应的通知引用程序进行处理,应用再将流读取到缓冲区或写入操做系统。也就是说,这个时候,已经不是一个链接就要对应一个处理线程了,而是有效的请求,对应一个线程,当链接没有数据时,是没有工做线程来处理的。微信

BIO和NIO一个比较重要的不一样,是咱们使用BIO的时候每每会引入多线程,每一个链接一个单独的线程;而NIO则是使用单线程或者使用少许线程,每一个链接公用一个线程。网络

NIO的最重要的地方是当一个链接建立后,不须要对应一个线程,这个链接会被注册到多路复用器上,因此全部的链接只须要一个线程就能搞定,当这个线程中多路复用器进行轮询的时候,发现链接上有请求的话,才开启一个线程进行处理,也就是一个请求一个线程模式。多线程

在NIO的处理方式中,当一个请求来的时候,开启线程进行处理,可能会等待后端应用的资源(JDBC链接等),其实这个线程就被阻塞了,若是并发上来,还会有BIO同样的问题。架构

HTTP/1.1出现后,有了Http长链接,这样除了超时和指明特定关闭的http header外,这个连接是一直打开的状态的,这样在NIO处理中能够进一步的进化,在后端资源中能够实现资源池或者队列,当请求来的话,开启的线程把请求和请求数据传送给后端资源池或者队列里面就返回,而且在全局的地方保持住这个现场(哪一个链接的哪一个请求等),这样前面的线程仍是能够去接受其余的请求,然后端的应用的处理只须要执行队列里面的就能够了,这样请求处理和后端应用是异步的.当后端处理完,到全局地方获得现场,产生响应,这个就实现了异步处理。并发

3、异步非阻塞AIO

与NIO不一样,当进行读写操做时,只需直接调用API的read或write方法便可。这两种方法均为异步的,对于读操做而言,当有流可读取时,操做系统会将可读的流传入read方法的缓冲区,并通知应用程序;对于写操做而言,当操做系统将write方法传递的流写入完毕时,操做系统主动通知应用程序。便可以理解为, read/write方法都是异步的,完成后会主动调用回调函数。   在JDK1.7中,这部份内容成为AIO,

主要在java.nio.channels包下增长了下面四个异步通道:

  • AsynchronousSocketChannel
  • AsynchronousServerSocketChannel
  • AsynchronousFileChannel
  • AsynchronousDatagramChannel

其中的read/write方法,会返回一个带回调函数的对象,当执行完读取/写入操做后,直接调用回调函数。  

BIO是一个链接一个线程。

NIO是一个请求一个线程。

AIO是一个有效请求一个线程。

先来个例子理解一下概念,以银行取款为例: 

  • 同步 : 本身亲自出马持银行卡到银行取钱(使用同步IO时,Java本身处理IO读写);
  • 异步 : 委托一小弟拿银行卡到银行取钱,而后给你(使用异步IO时,Java将IO读写委托给OS处理,须要将数据缓冲区地址和大小传给OS(银行卡和密码),OS须要支持异步IO操做API);
  • 阻塞 : ATM排队取款,你只能等待(使用阻塞IO时,Java调用会一直阻塞到读写完成才返回);
  • 非阻塞 : 柜台取款,取个号,而后坐在椅子上作其它事,等号广播会通知你办理,没到号你就不能去,你能够不断问大堂经理排到了没有,大堂经理若是说还没到你就不能去(使用非阻塞IO时,若是不能读写Java调用会立刻返回,当IO事件分发器会通知可读写时再继续进行读写,不断循环直到读写完成)

4、java对BIO、NIO、AIO的支持

java BIO:同步并阻塞

在此种方式下,用户进程在发起一个IO操做之后,必须等待IO操做的完成,只有当真正完成了IO操做之后,用户进程才能运行。JAVA传统的IO模型属于此种方式!

服务器实现模式为一个链接一个线程,即客户端有链接请求时服务器端须要启动一个线程进行处理,若是这个链接不作任何事情会形成没必要要的线程开销,固然能够经过线程池机制改善。

java NIO:同步非阻塞

在此种方式下,用户进程发起一个IO操做之后边可返回作其它事情,可是用户进程须要时不时的询问IO操做是否就绪,这就要求用户进程不停的去询问,从而引入没必要要的CPU资源浪费。其中目前JAVA的NIO就属于同步非阻塞IO。

服务器实现模式为一个请求一个线程,即客户端发送的链接请求都会注册到多路复用器上,多路复用器轮询到链接有I/O请求时才启动一个线程进行处理。

java AIO:异步非阻塞

在此种模式下,用户进程只须要发起一个IO操做而后当即返回,等IO操做真正的完成之后,应用程序会获得IO操做完成的通知,此时用户进程只须要对数据进行处理就行了,不须要进行实际的IO读写操做,由于真正的IO读取或者写入操做已经由内核完成了。目前Java中尚未支持此种IO模型。 

服务器实现模式为一个有效请求一个线程,客户端的I/O请求都是有OS先完成了再通知服务器应用去启动线程进行处理。

5、BIO、NIO、AIO适用场景分析

BIO方式适用于链接数目比较少且固定的架构,这种方式对服务器资源要求比较高,并发局限于应用中,JDK1.4之前的惟一选择,程序只管简单易理解。

NIO方式适用于链接数目多且比较短的架构,好比聊天服务器,并发局限于应用中,编程比较复杂,JDK1.4开始支持。

AIO方式适用于链接数目多且链接比较长的架构,好比相册服务器,充分调用OS参与并发操做,编程比较复杂,JDK1.7开始支持。

6、 Tomcat( BIO )和Jetty(NIO)

Tomcat和Jetty是目前全球范围内最著名的两款开源的webserver/servlet容器。

相同点:

一、Tomcat和Jetty都是一种servlet引擎,他们都支持标准的servlet规范和JavaEE的规范。

不一样点:

一、架构比较

(1)Jetty的架构比Tomcat简单。

(2)Jetty的架构是基于handler来实现的,主要的扩展功能均可以使用handler来实现,扩展简单。

(3)Tomcat的架构是基于容器设计的,进行扩展是须要了解Tomcat的总体设计结构,不易扩展。

二、性能比较

(1)Jetty和Tomcat性能方面差别不大。

(2)Jetty能够同时处理大量链接并且能够长时间保持链接,适合web聊天应用等等。

(3)Jetty的架构简单,所以做为服务器,Jetty能够按需加载组件,减小没必要要的组件,减小了服务器内部开销,从而提升服务器性能。

(4)Jetty默认采用NIO结束处理I/O请求上更占优点,在处理静态资源时,性能较高。

三、Tomcat适合处理少数很是繁忙的连接,Tomcat的整体性能更高。Tomcat默认采用BIO处理I/O请求,在处理静态资源时,性能较差。

四、其余比较

(1)Jetty的应用更加快速,修改简单,对新的servlet规范的支持较好。

(2)Tomcat目前应用比较普遍,对javaEE和servlet的支持更加全面,不少性能会直接集成进来。

 

更多精彩博客,请关注“素小暖”微信公众号

相关文章
相关标签/搜索