码农知识堂 - 1000bd
  •   Python
  •   PHP
  •   JS/TS
  •   JAVA
  •   C/C++
  •   C#
  •   GO
  •   Kotlin
  •   Swift
  • Netty 学习(八):新连接接入源码说明


    Netty 学习(八):新连接接入源码说明

    作者: Grey

    原文地址:

    博客园:Netty 学习(八):新连接接入源码说明

    CSDN:Netty 学习(八):新连接接入源码说明

    新连接的接入分为3个过程

    1. 检测到有新连接。

    2. 将新连接注册到 worker 线程。

    3. 注册新连接的读事件。

    检测新连接的代码在NioEventLoop中的processSelectedKey()方法中

        private void processSelectedKey(SelectionKey k, AbstractNioChannel ch) {
            ......
            final AbstractNioChannel.NioUnsafe unsafe = ch.unsafe();
            ......
                // Also check for readOps of 0 to workaround possible JDK bug which may otherwise lead
                // to a spin loop
                if ((readyOps & (SelectionKey.OP_READ | SelectionKey.OP_ACCEPT)) != 0 || readyOps == 0) {
                    unsafe.read();
                }
            .....
        }
    

    启动一个 Netty 服务端和 Netty 客户端,在unsafe.read()这一行打断点,可以得到这里的unsafe就是NioMessageUnsafe,进入NioMessageUnsafe的read()方法,

    这个方法主要做的事情就是:创建,设置并绑定NioSocketChannel。

         private final List readBuf = new ArrayList();
    
            @Override
            public void read() {
                ......
                do {
                            // 创建`NioSocketChannel`
                            int localRead = doReadMessages(readBuf);
                            ......
                        } while (continueReading(allocHandle));
              ......
                    // 设置并绑定 NioSocketChannel
                    int size = readBuf.size();
                    for (int i = 0; i < size; i ++) {
                        readPending = false;
                        pipeline.fireChannelRead(readBuf.get(i));
                    }
                    readBuf.clear();
                    allocHandle.readComplete();
                    pipeline.fireChannelReadComplete();
    
            ......
            }
        }
    
    
    

    创建NioSocketChannel调用的是doReadMessages()方法,通过Debug,可以看到doReadMessage()来自于NioServerSocketChannel中

        @Override
        protected int doReadMessages(List buf) throws Exception {
            SocketChannel ch = SocketUtils.accept(javaChannel());
    
            try {
                if (ch != null) {
                    buf.add(new NioSocketChannel(this, ch));
                    return 1;
                }
            } catch (Throwable t) {
                logger.warn("Failed to create a new channel from an accepted socket.", t);
    
                try {
                    ch.close();
                } catch (Throwable t2) {
                    logger.warn("Failed to close a socket.", t2);
                }
            }
    
            return 0;
        }
    
    

    可以看到此时调用的是 Java 底层的accept()方法,创建了一条 JDK 层面的Channel, Netty 将其封装成自定义的NioSocketChannel,并加入一个List。

    继续 Debug,进入 NioSocketChannel 的构造方法中,调用的是AbstractNioByteChannel的构造方法

        protected AbstractNioByteChannel(Channel parent, SelectableChannel ch) {
            super(parent, ch, SelectionKey.OP_READ);
        }
    

    这个方法类似在 NIO 编程中,注册 OP_READ 事件,表示 Channel 对读事件感兴趣。

    接下来是设置并绑定NioSocketChannel,处理每个NioSocketChannel,通过 Debug 可以来到AbstractUnsafe的register0()方法

    private void register0(ChannelPromise promise) {
                    // 注册Selector
                    doRegister();
                    // 执行 handler
                    pipeline.invokeHandlerAddedIfNeeded();
    
                    // 传播 ChannelRegistered事件
                    pipeline.fireChannelRegistered();
                    
                    // 注册读事件
                    if (isActive()) {
                        if (firstRegistration) {
                            pipeline.fireChannelActive();
                        } else if (config().isAutoRead()) {
                            // This channel was registered before and autoRead() is set. This means we need to begin read
                            // again so that we process inbound data.
                            //
                            // See https://github.com/netty/netty/issues/4805
                            beginRead();
                        }
                    }
    }
    

    这个方法主要完成的事情就是:

    1. 将NioSocketChannel注册到Selector上

    2. 配置自定义的Handler。

    3. 将连接注册事件传播下去,调用了每个Handler的channelRegistered方法。

    4. 注册读事件。

    完整代码见:hello-netty

    本文所有图例见:processon: Netty学习笔记

    更多内容见:Netty专栏

    参考资料#

    跟闪电侠学 Netty:Netty 即时聊天实战与底层原理

    深度解析Netty源码

  • 相关阅读:
    为什么说CDN是网站速度优化大师
    基于PHP+MySQL音乐相册网站的设计与实现
    Docker网络说明
    CRM 自动化如何改善销售和客户服务?
    浅析Java反射机制
    OJ练习第168题——课程表 III
    【Linux】命令
    Mysql的分布式事务原理理解
    windows11对编程有用的功能
    选择 Guava EventBus 还是 Spring Framework ApplicationEvent
  • 原文地址:https://www.cnblogs.com/greyzeng/p/16755179.html
    • 最新文章
    • 攻防演习之三天拿下官网站群
      数据安全治理学习——前期安全规划和安全管理体系建设
      企业安全 | 企业内一次钓鱼演练准备过程
      内网渗透测试 | Kerberos协议及其部分攻击手法
      0day的产生 | 不懂代码的"代码审计"
      安装scrcpy-client模块av模块异常,环境问题解决方案
      leetcode hot100【LeetCode 279. 完全平方数】java实现
      OpenWrt下安装Mosquitto
      AnatoMask论文汇总
      【AI日记】24.11.01 LangChain、openai api和github copilot
    • 热门文章
    • 十款代码表白小特效 一个比一个浪漫 赶紧收藏起来吧!!!
      奉劝各位学弟学妹们,该打造你的技术影响力了!
      五年了,我在 CSDN 的两个一百万。
      Java俄罗斯方块,老程序员花了一个周末,连接中学年代!
      面试官都震惊,你这网络基础可以啊!
      你真的会用百度吗?我不信 — 那些不为人知的搜索引擎语法
      心情不好的时候,用 Python 画棵樱花树送给自己吧
      通宵一晚做出来的一款类似CS的第一人称射击游戏Demo!原来做游戏也不是很难,连憨憨学妹都学会了!
      13 万字 C 语言从入门到精通保姆级教程2021 年版
      10行代码集2000张美女图,Python爬虫120例,再上征途
    Copyright © 2022 侵权请联系2656653265@qq.com    京ICP备2022015340号-1
    正则表达式工具 cron表达式工具 密码生成工具

    京公网安备 11010502049817号