本文首发于微信公众号【猿灯塔】,转载引用请说明出处

接下来的时间灯塔君持续更新Netty系列一共九篇

Netty源码解析(一):开始

当前:Netty 源码解析(二): Netty 的 Channel

Netty 源码解析(三): Netty 的 Future 和 Promise

Netty 源码解析(四): Netty 的 ChannelPipeline

Netty 源码解析(五): Netty 的线程池分析

Netty 源码解析(六): Channel 的 register 操作

Netty 源码解析(七): NioEventLoop 工作流程

Netty 源码解析(八): 回到 Channel 的 register 操作

Netty 源码解析(九): connect 过程和 bind 过程分析

今天呢!灯塔君跟大家讲:

Netty 的 Channel

这节我们来看看 NioSocketChannel 是怎么和 JDK 底层的 SocketChannel 联系在一起的,它们是一对一的关系。NioServerSocketChannel 和 ServerSocketChannel 同理,也是一对一的关系。

在 Bootstrap(客户端) 和 ServerBootstrap(服务端) 的启动过程中都会调用 channel(…) 方法:

下面,我们来看 channel(…) 方法的源码:

1// AbstractBootstrap
2public B channel(Class<? extends C> channelClass) {
3    if (channelClass == null) {
4        throw new NullPointerException("channelClass");
5    }
6    return channelFactory(new ReflectiveChannelFactory<C>(channelClass));
7}

我们可以看到,这个方法只是设置了 channelFactory 为 ReflectiveChannelFactory 的一个实例,然后我们看下这里的 ReflectiveChannelFactory 到底是什么:

newChannel() 方法是 ChannelFactory 接口中的唯一方法,工厂模式大家都很熟悉。我们可以看到,ReflectiveChannelFactory#newChannel() 方法中使用了反射调用 Channel 的无参构造方法来创建 Channel,我们只要知道,ChannelFactory 的 newChannel() 方法什么时候会被调用就可以了。

  • 对于NioSocketChannel,由于它充当客户端的功能,它的创建时机在 connect(…) 的时候;
  • 对于NioServerSocketChannel来说,它充当服务端功能,它的创建时机在绑定端口bind(…)的时候。

接下来,我们来简单追踪下充当客户端的Bootstrap中NioSocketChannel的创建过程,看看NioSocketChannel是怎么和JDK 中的SocketChannel关联在一起的:

1// Bootstrap
2public ChannelFuture connect(String inetHost, int inetPort) {
3    return connect(InetSocketAddress.createUnresolved(inetHost, inetPort));
4}

然后再往里看到这个方法:

1public ChannelFuture connect(SocketAddress remoteAddress) {
2    if (remoteAddress == null) {
3        throw new NullPointerException("remoteAddress");
4    // validate 只是校验一下各个参数是不是正确设置了
5    validate();
6    return doResolveAndConnect(remoteAddress, config.localAddress());
7}

继续:

1// 再往里就到这里了2private ChannelFuture doResolveAndConnect(final SocketAddress remoteAddress, final SocketAddress localAddress) {
3    // 我们要说的部分在这里
4    final ChannelFuture regFuture = initAndRegister();
5    final Channel channel = regFuture.channel();
6    ......
7}

然后,我们看initAndRegister()方法:

 1final ChannelFuture initAndRegister() {
2    Channel channel = null;
3    try {
4        // 前面我们说过,这里会进行 Channel 的实例化
5        channel = channelFactory.newChannel();
6        init(channel);
7    } catch (Throwable t) {
8        ...
9    }
10    ...
11    return regFuture;
12}

我们找到了channel = channelFactory.newChannel()这行代码,根据前面说的,这里会调用相应Channel的无参构造方法。

然后我们就可以去看NioSocketChannel的构造方法了:

1public NioSocketChannel() {
2    // SelectorProvider 实例用于创建 JDK 的 SocketChannel 实例
3    this(DEFAULT_SELECTOR_PROVIDER);
4}
5
6public NioSocketChannel(SelectorProvider provider) {
7    // 看这里,newSocket(provider) 方法会创建 JDK 的 SocketChannel
8    this(newSocket(provider));
9}

我们可以看到,在调用 newSocket(provider) 的时候,会创建JDK NIO的一个 SocketChannel实例:

1private static SocketChannel newSocket(SelectorProvider provider) {
2    try {
3        // 创建 SocketChannel 实例
4        return provider.openSocketChannel();
5    } catch (IOException e) {
6        throw new ChannelException("Failed to open a socket.", e);
7    }
8}

NioServerSocketChannel同理,也非常简单,从ServerBootstrap#bind(...)方法一路点进去就清楚了。

所以我们知道了,NioSocketChannel 在实例化过程中,会先实例化JDK底层的 SocketChannel,NioServerSocketChannel 也一样,会先实例化 ServerSocketChannel 实例:

说到这里,我们顺便再继续往里看一下NioSocketChannel的构造方法:

1public NioSocketChannel(SelectorProvider provider) {
2    this(newSocket(provider));
3}

刚才我们看到这里newSocket(provider)创建了底层的SocketChannel 实例我们继续往下看构造方法:

1public NioSocketChannel(Channel parent, SocketChannel socket) {
2    super(parent, socket);
3    config = new NioSocketChannelConfig(this, socket.socket());
4}

上面有两行代码,第二行代码很简单,实例化了内部的 NioSocketChannelConfig 实例,它用于保存channel的配置信息,这里没有我们现在需要关心的内容,直接跳过。

第一行调用父类构造器,除了设置属性外,还设置了SocketChannel的非阻塞模式:

 1protected AbstractNioByteChannel(Channel parent, SelectableChannel ch) {
2    // 毫无疑问,客户端关心的是 OP_READ 事件,等待读取服务端返回数据
3    super(parent, ch, SelectionKey.OP_READ);
4}
5
6// 然后是到这里
7protected AbstractNioChannel(Channel parent, SelectableChannel ch, int readInterestOp) {
8    super(parent);
9    this.ch = ch;
10    // 我们看到这里只是保存了 SelectionKey.OP_READ 这个信息,在后面的时候会用到
11    this.readInterestOp = readInterestOp;
12    try {
13        // ******设置 channel 的非阻塞模式******
14        ch.configureBlocking(false);
15    } catch (IOException e) {
16        ......
17    }
18}

NioServerSocketChannel的构造方法类似,也设置了非阻塞,然后设置服务端关心的SelectionKey.OP_ACCEPT事件:

1public NioServerSocketChannel(ServerSocketChannel channel) {
2    // 对于服务端来说,关心的是 SelectionKey.OP_ACCEPT 事件,等待客户端连接
3    super(null, channel, SelectionKey.OP_ACCEPT);
4    config = new NioServerSocketChannelConfig(this, javaChannel().socket());
5}

这节关于Channel的内容我们先介绍这么多,主要就是实例化了JDK层的SocketChannel或ServerSocketChannel,然后设置了非阻塞模式,我们后面再继续深入下去。

365天干货不断,可以微信搜索「 猿灯塔」第一时间阅读,回复【资料】【面试】【简历】有我准备的一线大厂面试资料和简历模板

Netty 源码解析(二):Netty 的 Channel的更多相关文章

  1. Netty 源码解析(六): Channel 的 register 操作

    原创申明:本文由公众号[猿灯塔]原创,转载请说明出处标注 今天是猿灯塔“365篇原创计划”第六篇. 接下来的时间灯塔君持续更新Netty系列一共九篇   Netty 源码解析(一 ):开始 Netty ...

  2. Netty 源码解析(八): 回到 Channel 的 register 操作

    原创申明:本文由公众号[猿灯塔]原创,转载请说明出处标注 今天是猿灯塔“365篇原创计划”第八篇. 接下来的时间灯塔君持续更新Netty系列一共九篇 Netty 源码解析(一): 开始 Netty 源 ...

  3. Netty 源码解析(三): Netty 的 Future 和 Promise

    今天是猿灯塔“365篇原创计划”第三篇. 接下来的时间灯塔君持续更新Netty系列一共九篇 Netty 源码解析(一): 开始 Netty 源码解析(二): Netty 的 Channel 当前:Ne ...

  4. Netty 源码解析(九): connect 过程和 bind 过程分析

    原创申明:本文由公众号[猿灯塔]原创,转载请说明出处标注 今天是猿灯塔“365篇原创计划”第九篇. 接下来的时间灯塔君持续更新Netty系列一共九篇 Netty 源码解析(一): 开始 Netty 源 ...

  5. Netty 源码解析(七): NioEventLoop 工作流程

    原创申明:本文由公众号[猿灯塔]原创,转载请说明出处标注 今天是猿灯塔“365篇原创计划”第七篇. 接下来的时间灯塔君持续更新Netty系列一共九篇 Netty 源码解析(一): 开始 Netty 源 ...

  6. Netty 源码解析(五): Netty 的线程池分析

    今天是猿灯塔“365篇原创计划”第五篇. 接下来的时间灯塔君持续更新Netty系列一共九篇 Netty 源码解析(一): 开始 Netty 源码解析(二): Netty 的 Channel Netty ...

  7. Netty 源码解析(四): Netty 的 ChannelPipeline

    今天是猿灯塔“365篇原创计划”第四篇. 接下来的时间灯塔君持续更新Netty系列一共九篇 Netty 源码解析(一): 开始 Netty 源码解析(二): Netty 的 Channel Netty ...

  8. Netty 源码(二)NioEventLoop 之 Channel 注册

    Netty 源码(二)NioEventLoop 之 Channel 注册 Netty 系列目录(https://www.cnblogs.com/binarylei/p/10117436.html) 一 ...

  9. Netty源码解析—客户端启动

    Netty源码解析-客户端启动 Bootstrap示例 public final class EchoClient { static final boolean SSL = System.getPro ...

随机推荐

  1. Java实现 Leetcode 169 求众数

    public static int majorityElement(int[] nums) { int num = nums[0], count = 1; for(int i=1;i<nums. ...

  2. 使用liunx系统自带的工具sar监控指定接口速率

    1.路由器双出口部署,接口可以实现负载分担,在接口负载比例设置为1:2之后,管理员反馈流量有些异常,内网tracert -d 外网域名或者IP,都走一条链路. 2.底层尝试使用sar命令监控两个接口的 ...

  3. akka-typed(4) - EventSourcedBehavior in action

    前面提到过,akka-typed中较重要的改变是加入了EventSourcedBehavior.也就是说增加了一种专门负责EventSource模式的actor, 最终和其它种类的actor一道可以完 ...

  4. 调优 | Apache Hudi应用调优指南

    通过Spark作业将数据写入Hudi时,Spark应用的调优技巧也适用于此.如果要提高性能或可靠性,请牢记以下几点. 输入并行性:Hudi对输入进行分区默认并发度为1500,以确保每个Spark分区都 ...

  5. MIT6.S081/6.828 实验1:Lab Unix Utilities

    Mit6.828/6.S081 fall 2019的Lab1是Unix utilities,主要内容为利用xv6的系统调用实现sleep.pingpong.primes.find和xargs等工具.本 ...

  6. android在service中stopself遇到的问题

    在service的oncreate中直接调用stopservice停止自己,依然会执行onstartcommand方法后,最后才调用ondestory方法

  7. JVM中堆的介绍

    一.堆的概述 一个JVM实例只有一个堆内存,堆也是Java内存管理的核心区域,堆在JVM启动的时候创建,其空间大小也被创建,是JVM中最大的一块内存空间,所有线程共享Java堆,物理上不连续的逻辑上连 ...

  8. ELK的踩坑之旅

    前言 设计思路如下 有3台机器 2台做elasticsearch的主副节点 1台做kibana和elasticsearch_head 由于机器匮乏我还在这台机器上部署了logstash和nginx服务 ...

  9. 做一个有价值的seoer

    什么是SEO就不用在解释了吧,一般来看这篇文章的都是老鸟了,基础的术语解释也啥意义. 从早期的cms建站,到现在的批量镜像站群等操作,似乎搜索引擎现在都下了重手,前几年还好用的方式现在已经不再是那么理 ...

  10. ping外网:unknown host www.baidu.comc排查

    ping 百度出现:(ping www.baidu.com) "ping: unknown host www.baidu.com"    1.ping 网关确定是否连接上路由器,并 ...