nio通讯过程

初步了解nio,但是一直很难理解。网上看到一篇博文,感觉讲的非常好。

nio通讯过程情景模拟

>1.学校(ServerSocketChannel)

>2.学校教务处(Selector)

>3.老师(ServerSocket)

>4.学生(SocketChannel)

>5.员工号/学生号(SelectionKey)

学校相当于我们的网络应用程序,一旦开学,学校就不能轻易停止,要一直运行到学期结束

启动学校就要:

// 新建NIO通道
ServerSocketChannel serverSocketChannel = ServerSocketChannel.open();
// 使通道为非阻塞
serverSocketChannel.configureBlocking(false);  

老师:相当于服务端的socket,一个老师对应多个学生,多个学生向老师请教,

老师会一一做出回答,学校要正常运行,当不了老师,所以在开学之前,必须

先聘请专业的老师来学校任教

serverSocketChannel.socket().bind(bindAddress, SOCKET_BACKLOG); // 新建socket通道的端口  

学校教务处:老师有了,但是需要有部门对老师和学生做统一的管理,如果

你去一个学校找一个人,实在是找不到了,你可以告诉教务处,那个人是学生,

还是老师,是老师的话,员工编号老师姓名是什么;是学生的话,学生号,

学生姓名是什么。教务处就会告诉你,他在那里。

// 将NIO通道绑定到选择器,绑定后分配的主键为key
SelectionKey key = serverSocketChannel.register(selector, SelectionKey.OP_ACCEPT);  
serverSocketChannel注册了选择器后,学校下面的老师ServerSocket也就加入了员工册 。

所以老师的编号就是key 

学生(相当于client):学校,老师,教务处都有了,现在可以招生了。 

如果有学生来报名

```

    while(true){//除非学期结束,否则一直等待学生

    int num = selector.select();//获取通道内是否有选择器的关心事件, 意思是有多少学生报告

    if(num<1){continue; }

    Set selectedKeys = selector.selectedKeys();//获取通道内关心事件的集合 ,这里的集合就是老师和学生的编号集合,如果key是学生的,那就是老学生来问问题,如果key是老师的,那就是招生办的老师带着一个新生来注册

    Iterator it = selectedKeys.iterator();

    while (it.hasNext()) {//遍历每个key (学生key和老师key)

    .......

    }

    .....

    }

```

既然有学生来报告,那就有两种可能,

一种是招生老师带新生来注册包名,

一种是老生来问问题。

上面的while(it.hasNext())体可以这么写

```

while (it.hasNext()) {//遍历每个事件
    try {
        SelectionKey key = (SelectionKey) it.next(); //先得到这个学生的编号key
        //判断是新生报道还是老生问问题
        if ((key.readyOps() & SelectionKey.OP_ACCEPT) == SelectionKey.OP_ACCEPT) {
            //这是招生老师的Key说明是新生注册,先找到招生老师,再由招生老师找到新生,就可以给新生注册学号了
            ServerSocketChannel serverChanel = (ServerSocketChannel) key.channel(); //通过key把学校和老师找到了
            //从serverSocketChannel中创建出与客户端的连接socketChannel 有了老师才有学生,不可能我教计算机的,来一个想学李小龙的都让他报名
            SocketChannel sc = serverChanel.accept(); //学生报名成功
            sc.configureBlocking(false);
            // 把新连接注册到选择器,新生被接收后给注册个新学号
            SelectionKey newKey = sc.register(selector, SelectionKey.OP_READ); //注册学号成功,并分配学生的权限
            it.remove(); //新生注册任务完成了,呵呵
            System.out.println("Got connection from " + sc);
        } else if ((key.readyOps() & SelectionKey.OP_READ) == SelectionKey.OP_READ) { //读客户端数据的事件,此时有客户端发数据过来,客户端事件 这是老学生来问问题了。
            // 读取数据 ,接受学生的问题
            SocketChannel sc = (SocketChannel) key.channel(); //通过学号知道是谁问的问题
            //下面接受问题
            int bytesEchoed = 0;
            while ((bytesEchoed = sc.read(echoBuffer)) > 0) {
                System.out.println("bytesEchoed:" + bytesEchoed);
            }
            echoBuffer.flip();
            System.out.println("limet:" + echoBuffer.limit());
            byte[] content = new byte[echoBuffer.limit()];
            echoBuffer.get(content);
            String result = new String(content);
            doPost(result, sc); //相应老师会去做回答的,细节自己去写吧
            echoBuffer.clear();
            it.remove(); //任务完成,记得上面也是一样,要remove掉,否则下一次又来一次任务,就死循环了
        }
    } catch (Exception e) {
    }
}

```

ssc.register( selector, SelectionKey.OP_ACCEPT );
这个方法是把ssc注册绑定到选择器selector 这样下次你想找ssc或者判断一个对象是不是ssc就可以通过selector来查找,查找是通过判断ssc的key得到的。
至于第二个参数SelectionKey.OP_ACCEPT 你可以理解成ssc的key类型或者操作权限
如果 ssc是学校老师,那么绑定成功后 老师就拥有了OP_ACCEPT的权限或者说他的key类型是SelectionKey.OP_ACCEPT
Accept是接受的意思,这是不是很像socket编程里的 accept()方法呢? 是的,没错,我们正是通过这个参数给了老师招生和带学生来注册的权限。

而学生呢?
他拥有的权限为SelectionKey.OP_READ 表示有收发读取消息的权限,即问问题的权限,因此他不能帮别的学生注册。

所以你回到上面仔细看看while结构体里面做了判断如下:

if ((key.readyOps() & SelectionKey.OP_ACCEPT) == SelectionKey.OP_ACCEPT) {...} 很明显,拥有Accept权限的人只可能是老师,那老师有什么事会找教务处? 那肯定就是他是招生办的,招到一个学生来报名来注册了。
然后,马上给这个新连上来的客户端分配了一个key
SelectionKey newKey = sc.register( selector,
SelectionKey.OP_READ ); 看,这里只给他OP_READ,而不是Accept哦

另一个if
else
if((key.readyOps() & SelectionKey.OP_READ)== SelectionKey.OP_READ){

//很明显,这是这学生,因为所有带OP_READ的人都是前面由招生办老师带过来注册过的。

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值