JAVA BIO与NIO、AIO的区别(这个容易理解)

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

一、BIO

     在JDK1.4出来之前,我们建立网络连接的时候采用BIO模式,需要先在服务端启动一个ServerSocket,然后在客户端启动Socket来对服务端进行通信,默认情况下服务端需要对每个请求建立一堆线程等待请求,而客户端发送请求后,先咨询服务端是否有线程响应,如果没有则会一直等待或者遭到拒绝请求,如果有的话,客户端线程会等待请求结束后才继续执行。

二、NIO

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

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

   BIO与NIO一个比较重要的不同,是我们使用BIO的时候往往会引入多线程,每个连接一个单独的线程;而NIO则是使用单线程或者只使用少量的多线程,多个连接共用一个线程。

      NIO的最重要的地方是当一个连接创建后,不需要对应一个线程,这个连接会被注册到多路复用器上面,所以所有的连接只需要一个线程就可以搞定,当这个线程中的多路复用器进行轮询的时候,发现连接上有请求的话,才开启一个线程进行处理,也就是一个请求一个线程模式。

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

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

三、AIO

     与NIO不同,当进行读写操作时,只须直接调用API的read或write方法即可。这两种方法均为异步的,对于读操作而言,当有流可读取时,操作系统会将可读的流传入read方法的缓冲区,并通知应用程序;对于写操作而言,当操作系统将write方法传递的流写入完毕时,操作系统主动通知应用程序。  即可以理解为,read/write方法都是异步的,完成后会主动调用回调函数。  在JDK1.7中,这部分内容被称作NIO.2,主要在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事件分发器会通知可读写时再继续进行读写,不断循环直到读写完成)

Java对BIO、NIO、AIO的支持:

  • Java BIO : 同步并阻塞,服务器实现模式为一个连接一个线程,即客户端有连接请求时服务器端就需要启动一个线程进行处理,如果这个连接不做任何事情会造成不必要的线程开销,当然可以通过线程池机制改善。

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

  • Java AIO(NIO.2) : 异步非阻塞,服务器实现模式为一个有效请求一个线程,客户端的I/O请求都是由OS先完成了再通知服务器应用去启动线程进行处理,

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

  • BIO方式适用于连接数目比较小且固定的架构,这种方式对服务器资源要求比较高,并发局限于应用中,JDK1.4以前的唯一选择,但程序直观简单易理解。

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

  • AIO方式使用于连接数目多且连接比较长(重操作)的架构,比如相册服务器,充分调用OS参与并发操作,编程比较复杂,JDK7开始支持。

另外,I/O属于底层操作,需要操作系统支持,并发也需要操作系统的支持,所以性能方面不同操作系统差异会比较明显。

在高性能的I/O设计中,有两个比较著名的模式Reactor和Proactor模式,其中Reactor模式用于同步I/O,而Proactor运用于异步I/O操作。

一般来说I/O模型可以分为:同步阻塞,同步非阻塞,异步阻塞,异步非阻塞IO

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

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

异步阻塞IO:此种方式下是指应用发起一个IO操作以后,不等待内核IO操作的完成,等内核完成IO操作以后会通知应用程序,这其实就是同步和异步最关键的区别,同步必须等待或者主动的去询问IO是否完成,那么为什么说是阻塞的呢?因为此时是通过select系统调用来完成的,而select函数本身的实现方式是阻塞的,而采用select函数有个好处就是它可以同时监听多个文件句柄,从而提高系统的并发性!

异步非阻塞IO:在此种模式下,用户进程只需要发起一个IO操作然后立即返回,等IO操作真正的完成以后,应用程序会得到IO操作完成的通知,此时用户进程只需要对数据进行处理就好了,不需要进行实际的IO读写操作,因为真正的IO读取或者写入操作已经由内核完成了。Java AIO属于这种异步非阻塞模型。  

  • 118
    点赞
  • 596
    收藏
    觉得还不错? 一键收藏
  • 25
    评论
### 回答1: Java中的BIONIOAIO是三种不同的I/O模型。 BIO(Blocking I/O)是阻塞式I/O模型,即在进行I/O操作时,线程会一直阻塞直到操作完成。这种模型适用于连接数较少的情况,但在高并发环境下会导致线程阻塞,资源浪费。 NIO(Non-blocking I/O)是非阻塞式I/O模型,即在进行I/O操作时,线程不会一直阻塞,而是可以继续执行其他任务。这种模型适用于连接数较多的情况,可以提高系统的并发性能。 AIO(Asynchronous I/O)是异步I/O模型,即在进行I/O操作时,线程不需要等待操作完成,而是可以继续执行其他任务。这种模型适用于I/O操作时间较长的情况,可以提高系统的吞吐量。 总之,BIO适用于连接数较少的情况,NIO适用于连接数较多的情况,而AIO适用于I/O操作时间较长的情况。 ### 回答2: Java中的“BIO”、“NIO”和“AIO”是三种不同的I/O模型。其中BIO(Blocking I/O),也被称为传统I/O,是一种基于流(Stream)的I/O模型;NIO(Non-blocking I/O)是一种面向缓冲区(Buffer)的I/O模型;AIO(Asynchronous I/O)是一种最新的异步I/O模型。 BIO模型的 I/O操作是同步阻塞的,在进行网络访问时,需要建立连接、使用流(Stream)等函数操作,由于数据读取操作会进行阻塞等待,对于高并发请求存在着阻塞等待的情况,这将导致网络服务变得非常缓慢。 NIO模型采用信息缓冲区(Buffer)实现数据的传输,在进行网络访问时,数据是被存储在缓存区中,可以进行异步操作,也可以进行同步操作,但是NIO模型并不能彻底解决并发问题,如果同一时间依然有多个网络请求的时候,服务器依然很难处理这些请求。 AIO是异步I/O,采用了事件处理机制,当一个异步通道被打开时,系统会开辟一个线程专门为了处理I/O操作,会在异步I/O操作完成时通知线程进行后面的操作,最后将结果返回给调用者。AIO作为一种异步模型,可以处理大量并发请求,不会在I/O的阻塞等待耗费大量的CPU时间,非常适用于高并发系统的应用设计。 总的来说,Java BIO模型是同步阻塞模型,NIO模型是同步非阻塞模型,AIO模型是异步非阻塞模型,它们各有优缺点,在选择时需要根据实际应用情况和需求进行选择。 ### 回答3: Java BioNIOAIOJava编程语言中主要用于网络编程的三种I/O模型。 Java Bio (Blocking I/O)是阻塞式I/O,通过Socket等流式数据传输方式与外部设备进行数据交换。这种模型的主要特点是阻塞式读写模式,也就是一旦读写函数被调用,在读写操作完成之前会阻塞程序执行,如果要读写多个连接,就需要多线程或多进程才能处理。 Java NIO(Non-blocking I/O)是非阻塞式I/O,通过Selector等多路复用机制实现并发读写。这种模型通过使用少量的线程处理客户端请求,可以更好地利用计算机的资源,降低了线程切换的开销,从而提升了程序的吞吐量和并发性能。 Java AIO(Asynchronous I/O)是异步I/O,通过异步通过轮询机制等方式进行I/O操作。这种模型通过事件通知方式实现异步操作,不需要线程或者回调函数来处理I/O事件。因为异步I/O采用了异步非阻塞I/O读写模式,所以在性能、扩展性等方面有很大的优势。 总的来说,Java Bio适用于连接数较少、并发性能要求不高的应用场景;Java NIO适用于连接数高、并发性能要求较高的应用场景,例如服务器应用;Java AIO适用于并发性能要求极高、I/O操作非常频繁的应用场景。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 25
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值