NIO同步非阻塞模型学习使用

别拿奋斗当借口,吃好才能继续走

NIO同步非阻塞模型

  • Java NIO(New IO)也有人称之为 java non-blocking IO是从Java 1.4版本开始引入的一个新的IO API。NIO可以理解为非阻塞IO,传统的IO的read和write只能阻塞执行,线程在读写IO期间不能干其他事情,比如调用socket.read()时,如果服务器一直没有数据传输过来,线程就一直阻塞,而NIO中可以配置socket为非阻塞模式。
  • NIO 相关类都被放在 java.nio 包及子包下,并且对原 java.io 包中的很多类进行改写。
  • NIO 有三大核心部分:Channel( 通道) ,Buffer( 缓冲区), Selector( 选择器)

NIO 和 BIO 的比较

在这里插入图片描述

  1. BIO 以流的方式处理数据,而NIO以块的方式处理数据,块I/O的效率比流I/O高很多
  2. BIO是阻塞的,NIO则是非阻塞的
  3. BIO基于字节流和字符流进行操作,而NIO基于channel(通道)和Buffer(缓冲区)进行操作,数据总是从通道读取到缓冲区中,或者从缓冲区写入通道中。Selector(选择器)用于监听多个通道的事件(比如连接请求、事件到达等),因此使用单个线程就可以监听多个客户端通道。

NIO 三大核心

NIO三大核心:Channel通道、Buffer缓冲区、Selector选择器

Buffer缓冲区

缓冲区本质上是一块可以写入数据的内存块,底层是一个数组,这块内存被包装为NIO Buffer对象,并提供了一组方法,用来方便的访问该块内存。相比较直接对数组的操作,Buffer API更加容易操作和管理。

Channel(通道)

Java NIO的通道类似流,但又有些不同:既可以从通道中读取数据,又可以写数据到通道。但流的(input或output)读写通常是单向的,通道可以非阻塞读取和写入通道,通道可以支持读取和写入缓冲区,也支持异步读写。每个channel都会对应一个buffer

Selector选择器

Selector是一个java NIo组件,可以能够检查一个或多个NIO通道,并确定哪些通道已经准备好进行读取或写入。Selector会根据不同的事件,在各个通道上切换,这样,一个单独的线程可以管理多个channel,从而管理多个网络连接,提高效率。一个线程对应一个Selector,一个Selector对应多个channel(连接)

在这里插入图片描述

NIO的核心在于:通道和缓冲区。通道表示打开IO设备(例如:文件、套接字)的连接。若需要使用NIO系统,需要获取用于连接IO设备的通道以及用于容纳数据的缓冲区。然后操作缓冲区,对数据进行处理。简而言之,channel负责传输,Buffer负责存取数据

NIO非阻塞式网络通信入门案例

需求:服务端接收客户端的连接请求,并接收多个客户端发送过来的事件。

代码案例

服务端:

import java.io.IOException;
import java.net.InetSocketAddress;
import java.nio.ByteBuffer;
import java.nio.channels.*;
import java.util.Iterator;


/**
* @PackageName: com.netty.nio.buffer.demo2
* @author: youjp
* @create: 2021-01-04 09:59
* @description: TODO NIO模式下的服务端
* @Version: 1.0
*/
public class Server {


    public static void main(String[] args) throws IOException {
        //1,获取通道
        ServerSocketChannel channel = ServerSocketChannel.open();
        //2.切换到非阻塞模式
        channel.configureBlocking(false);
        //3.绑定连接,绑定8888端口
        channel.bind(new InetSocketAddress(8888));
        //4.获取选择器
        Selector selector = Selector.open();
        //5.讲通道注册到获取的选择器上,并指定监听事件,这里刚开始为连接
        channel.register(selector, SelectionKey.OP_ACCEPT);
        System.out.println("-----服务端准备就绪-----");
        //6.轮询选择器获取已经"准备就绪的事件"
        while (selector.select() > 0) {


            //7.获取当前选择器中所有注册的“选择键(已就绪的监听事件)”
            Iterator<SelectionKey> it = selector.selectedKeys().iterator();
            while (it.hasNext()) {
                //8.获取准备“就绪”事件
                SelectionKey sk = it.next();
                //判断具体事件
                if (sk.isAcceptable()) {
                    //若为接收就绪,获取客户端连接
                    SocketChannel sChannel = channel.accept();
                    sChannel.configureBlocking(false);//设置非阻塞模式
                    sChannel.register(selector, SelectionKey.OP_READ);//以读事件,将该通道注册到选择器上,
                    System.out.println("--通道准备就绪------"+sChannel.getRemoteAddress()+"------");
                } else if (sk.isReadable()) { //读准备就绪


                    //获取当前选择器上“读就绪”状态的通道
                    SocketChannel ch = (SocketChannel) sk.channel();
                    //使用缓冲区读取数据
                    ByteBuffer buffer = ByteBuffer.allocate(1024);
                    int len = 0;
                    while ((len = ch.read(buffer)) > 0) {
                        buffer.flip();//切换为读取模式
                        System.out.println(new String(buffer.array(), 0, len));
                        buffer.clear();
                    }


                }


                //取消当前选择键
                it.remove();


            }
        }
    }
}

客户端

import java.io.IOException;
import java.net.InetSocketAddress;
import java.nio.ByteBuffer;
import java.nio.channels.SocketChannel;
import java.text.SimpleDateFormat;
import java.util.Scanner;


/**
* @PackageName: com.netty.nio.buffer.demo2
* @author: youjp
* @create: 2021-01-04 09:59
* @description: NIo模式下的客户端
* @Version: 1.0
*/
public class Client {


    public static void main(String[] args) throws IOException {
        //1.获取通道
        SocketChannel sh=SocketChannel.open(new InetSocketAddress("127.0.0.1",8888));
        //2.设置为非阻塞模式
        sh.configureBlocking(false);
        //3.设置缓冲区
        ByteBuffer buffer= ByteBuffer.allocate(1024);


        //4.写入数据发给服务端
        Scanner scan=new Scanner(System.in);
        while (scan.hasNext()){
            String str=scan.nextLine();
            buffer.put((new SimpleDateFormat("yyyy/MM/dd HH:mm:ss").format(System.currentTimeMillis())
                    + "\n" + str).getBytes());
            buffer.flip();
            sh.write(buffer);
            buffer.clear();




        }
        //5.关闭通道
        sh.close();
    }
}

有兴趣的老爷,可以关注我的公众号【一起收破烂】,回复【006】获取2021最新java面试资料以及简历模型120套哦~
在这里插入图片描述

评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

收破烂的小熊猫~

你的鼓励将是我创造最大的东西~

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值