【Java.NIO】Java NIO中的阻塞 V.S. 非阻塞

参考: 广义同步&异步 阻塞&非阻塞 及 网络IO中的同步&异步,阻塞&非阻塞


从JDK1.4版本开始,引入了非阻塞的通信机制

服务器程序接受客户连接,客户程序建立与服务器的连接,以及服务器程序和客户端程序收发数据的操作都可以按非阻塞的方式进行。服务器程序只需要创建一个线程,就能完成同时与多个客户通信的任务。


线程阻塞

线程在运行中会因为某些原因而阻塞,所有处于阻塞状态的线程的共同特征是:

放弃CPU,暂停运行,只有等到导致阻塞的原因消除,才能恢复运行;或者被其他线程中断,该线程会退出阻塞状态,并且抛出InterruptedException。


线程阻塞的原因

导致线程阻塞的原因主要有以下几方面:

  • 线程执行了Thread.sleep(int n)方法,即线程放弃CPU,睡眠n毫秒,然后恢复运行
  • 线程要执行一段同步代码,由于无法获得相关的同步锁,进入阻塞状态,等待获得了同步锁,才能恢复运行
  • 线程执行一个对象的wait()方法,进入阻塞状态,只有等到其他线程执行了该对象的notify()或notifyAll()方法,才可能将其唤醒
  • 线程执行I/O操作或进行远程通信时,会因为等待相关的资源而进入阻塞状态

在进行远程通信时,在客户程序中,线程在以下情况可能进入阻塞状态:

  • 请求与服务器建立连接时,即当线程执行Socket的带参数的构造方法,或执行Socket的connect()方法时,会进入阻塞状态,直到连接成功,此线程才从Socket的构造方法或connect()方法返回
  • 线程从Socket的输入流读入数据时,如果没有足够的数据,就会进入阻塞状态,直到读到了足够的数据,或者到达了输入流的末尾,或者出现了异常,才从输入流的read()方法返回或异常中断。输入流中有多少数据才算足够,这要看线程执行的read()方法的类型:
    • int read():只要输入流中要一个字节,就算足够
    • int read(byte[] buff): 只要输入流中的字节数与参数buff数组的长度相同,就算足够
    • String readLine():只要输入流中有一行字符串,就算足够
  • 线程向Socket输出流中写一批数据时,可能会进入阻塞状态,等到输出了所有的数据,或者出现异常,才从输出流的write()方法返回或异常中断
  • 调用Socket的setSoLinger()方法设置了关闭Socket的延迟事件,那么当线程执行Socket的close()方法时,会进入阻塞状态,直到底层Socket发送完所有剩余数据,或者超过了方法设置的延迟时间,才从close()方法返回

在服务器程序中,线程在一下情况下可能会进入阻塞状态:

  • 线程执行ServerSocket的accept()方法,等待客户的连接,直到收到了客户连接,才从accept()方法返回
  • 线程从Socket的输入流读入数据,如果输入流没有足够的数据,就会进入阻塞状态
  • 线程向Socket的输出流写一批数据时,可能会进入阻塞状态,等到输出了所有的数据,或者出现异常,才从输出流的write()方法返回或异常中断


这种可能出现阻塞的输入和输出操作被称为阻塞I/O。与此对照,如果执行输入和输出操作时,不会发生阻塞,则称为非阻塞I/O。


服务器程序用多线程处理阻塞通信的局限

Java虚拟机会为每个线程分配独立的堆栈空间,工作线程数目越多,系统开销就越大,而且增加了Java虚拟机调度线程的负担,增减了线程之间同步的复杂性,提高了线程死锁的可能性

工作线程的时间大多浪费在阻塞I/O操作上,Java虚拟机需要频繁的转让CPU的使用权,使进入阻塞状态的线程放弃CPU,在把CPU分配给处于可运行状态的线程。

保持适当的工作线程,会提高服务器的并发性能,但是当工作线程的数目达到某个极限,超出了系统的负荷,反而会降低并发性能,使得多数客户无法快速得到服务器的响应。


非阻塞通信

所谓非阻塞,就是指当线程执行这些方法时,如果操作还没有就绪,就立即返回,而不会一直等到操作就绪。

例如:当线程接受客户连接时,如果没有客户连接,就立即返回;或者当线程从输入流中读数据时,如果输入流中还没有数据,就立即返回;或者如果输入流中还没有足够的数据,那么就读取现有的数据,然后返回。


while (一直等待,直到有接受连接就绪事件,或读就绪事件,或写就绪事件发生时){  // 阻塞
    if (有客户连接)
        接收客户的连接;                                             // 非阻塞
    if (某个Socket的输入流中有可读数据)
        从输入流中读数据;                                           // 非阻塞
    if (某个Socket的输出流可以写数据)
        向输出流写数据;                                             // 非阻塞
}


注意:上面while循环条件中的操作还是按照阻塞方式进行的,如果未发生任何事件,就会进入阻塞状态,直到接受连接就绪,读就绪事件或写就绪事件中至少有一个事件发生时,才会执行while循环体中的操作。在while循环中,一般会包含在特定条件下退出循环的操作。







  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值