Consumer启动后,会请求broker,broker接收请求网络
RemotingCommand 消息传输过程中对数据内容的封装结构,主要属性
code:请求操作码,应答方根据不同的请求码进行不同的业务处理
language:请求方实现的语言
version:版本
opaque:相当于requestId,在同一个连接上的不同请求标识码,与响应消息中相对应
flag:区分普通RPC还是oneWay RPC标识
remark:传输自定义文本信息
extFields:请求自定义扩展信息
customHeader:自定义头,不进行序列化
body:消息主体的二进制字节数据内容
NettyRemotingServer 服务端Netty start如下处理 ServerBootstrap

serverBootstrap.childHandler(new ChannelInitializer<SocketChannel>() {
     @Override
     public void initChannel(SocketChannel ch) throws Exception {
         ch.pipeline()
             .addLast(defaultEventExecutorGroup, HANDSHAKE_HANDLER_NAME, handshakeHandler)
             .addLast(defaultEventExecutorGroup,
                 encoder,
                 new NettyDecoder(),
                 //心跳
                 new IdleStateHandler(0, 0, nettyServerConfig.getServerChannelMaxIdleTimeSeconds()),
                 connectionManageHandler,//连接管理
                 serverHandler//真正干活的是这个线程池,有连接来的时候boss处理,channel有事件发生的时候 selecter将事件读出来,然后交给EventExecutor来处理
             );
     }
 });NettyRemotingClient 客户端Netty start如下处理 Bootstrap
bootstrap.handler(new ChannelInitializer<SocketChannel>() {
     @Override
     public void initChannel(SocketChannel ch) throws Exception {
         ChannelPipeline pipeline = ch.pipeline();
         if (nettyClientConfig.isUseTLS()) {
             if (null != sslContext) {
                 pipeline.addFirst(defaultEventExecutorGroup, "sslHandler", sslContext.newHandler(ch.alloc()));
                 log.info("Prepend SSL handler");
             } else {
                 log.warn("Connections are insecure as SSLContext is null!");
             }
         }
         pipeline.addLast(
             //使用这个线程组来处理processor
             defaultEventExecutorGroup,
             new NettyEncoder(),
             new NettyDecoder(),
             //心跳
             new IdleStateHandler(0, 0, nettyClientConfig.getClientChannelMaxIdleTimeSeconds()),
             new NettyConnectManageHandler(),//链接管理
             new NettyClientHandler());//处理消息的接收
     }
 });

NettyEncoder和NettyDecoder 是对RemotingCommand的编码解码,解析消息头,解析出消息内容,粘包拆包

BrokerController的构造方法创建PullMessageProcessor 通过remotingServer注册PULL_MESSAGE 消息拉取请求
processRequest负责对consumer的消费请求进行处理
1、创建响应头RemotingCommand
2、读取消息响应头
3、解码拉取消息的请求头拓展字段
4、权限、订阅关系、topic是否存在、队列有效性、消费者组是否存在、广播模式检查
5、DefaultMessageStore.getMessage
  入参:
  group:消费者组名
  topic:消息主题
  queueId:消息队列ID
  offset:拉取的消费队列偏移量
  maxMsgNums:一次拉取消息条数,默认32条
  messageFilter:消息过滤器
  1、获取commitLog文件中的最大偏移量
  2、根据topic、queueId获取消息消费队列ConsumeQueue
  3、获取此消费队列最小偏移量和最大偏移量
  4、根据需要拉取消息的偏移量与队列最大最小偏移量对比
    1、maxOffset为0表示队列中没有消息
    2、offset < minOffset 需要的offset比队列中的最新offset小,设置下次拉取最小的offset,
    3、offset == maxOffset 表示超出一个 返回状态:OFFSET_OVERFLOW_ONE,offset 保持不变
    4、offset > maxOffset 表示超出,OFFSET_OVERFLOW_BADLY,计算下一次拉取拉取的开始偏移量
    5、offset 大于minOffset 并小于maxOffset,正常情况
  5、从consuequeue 中从当前 offset 到当前 consueque 中最大可读消息内存
  6、获取最大过滤消息字节数,max(16000, maxMsgNums * 20) 最低16000 因为有消息过滤机制,可能不满足
    指定拉取的消息数,尽量满足返回这么多条消息
  7、循环拉取消息
    1、bufferConsumeQueue消费队列中获取commitLog的偏移量、消息总长度、tag hashcode
    2、如果拉取到的消息偏移量小于下一个要拉取的物理偏移量,跳过该条消息
    3、检查offsetPy,拉取的偏移量是否在磁盘上,maxOffsetPy-offsetPy > memory 的话,memory = 物理内存 * 这个比例
       说明 offsetPy 这个偏移量的消息已经从内存中置换到磁盘中了
    4、判断本次拉取任务是否完成
    5、执行消息过滤
    6、从commitLog文件中根据偏移量和消息大小读取消息 commitLog.getMessage
    7、将读取到的消息结果添加到结果集
    8、设置下次拉取任务开始nextBeginOffset offset + (i / ConsumeQueue.CQ_STORE_UNIT_SIZE)
    9、如果是超过了常驻内存设置下次拉取从从服务器拉取
6、根据 response.getCode() 分别作出不同处理
  1、SUCCESS 更新broker统计信息
    isTransferMsgByHeap:是否在heap内存汇总直接转换,间获取到的byteBuffer在heap内存汇总转换为字节数组
    普通IO进行转换,将bytebuffer转成字节数组,设置到response中去,使用堆内存来处理的
    通过channel写入fileRegion
  2、PULL_NOT_FOUND
    1、brokerAllowSuspend:构建消息拉取是的拉取标记,默认true
    2、是否支持长轮训,不支持设置1000ms作为下次拉取消息的等待时间
    3、创建pullRequest 提交给ullRequestHoldService 线程去调度,触发消息拉取
    4、设置response = null,此次调用不会向客户端输出任何字节,客户端网络请求客户端的读事件不会触发,不会触发对响应结果的处理,处于等待状态
7、判断它这次拉取消息请求里面带没带着消费offset,如果带着的话,就找到ConsumerOffset组件,然后更新一下消费offset
   消息消费者除了定时任务5更新一下消费进度,还可以通过拉取消息的时候带着消费offset,进行消费进度的更新