消息中间件和消息队列是同一个概念的两种名称,是分布式系统中的一个重要组件。
举个例子:比如现在有一个商城系统使用的是微服务架构,其中有订单服务、库存服务、物流服务、积分服务。业务场景是如果下单了一个商品需要调用库存服务减库存、需要调用物流服务发送商品、需要调用积分服务添加响应的积分。这中间牵扯到了服务和服务之间的调用问题,为了解决这两种问题有两种方案:
消息中间件的应用场景
异步解耦
比如上文说到的例子,一个请求如果使用RPC框架同步处理的话是需要400ms的,如果是使用消息中间件异步处理这些消息的话只需要110ms。
还有一种情况是,加入此时积分服务挂掉了,那么这个订单的请求就不能完成,可能造成订单下了但是积分没有加上去的bug、要么就是订单下不成功,加入使用的是消息中间件,即使积分服务挂了,但是消息成功加入到了消息队列,这次加积分的消息就一直没有被消费掉,等到积分服务重启的时候会再去消息队列中读取消息并处理对应的积分业务。
削峰填谷
如上图所示用户的请求是根据时间段的不同有所不同的,但是服务器的处理事务的能力限于配置的问题肯定也是有限的比如上图所示的1w/qps,如果用户的请求处于1w/qps一下还好说服务器都能处理的过来,如果超过1w/qps服务器就没办法处理,此时处理不掉的请求可能就会直接丢掉,但是如果引入消息中间件,用户发过来的请求不要直接给系统让系统处理,而是添加到消息中间件,那么如果用户请求超过1w/qps之后多余的消息就会丢到MQ中,等待被处理,当用户请求降下来了此时系统再赶紧处理MQ中还没处理的消息,这样处理的结果是用户端可能慢一点收到响应但是不至于报错。所以此时消息中间件就提到了削峰填谷的作用
消息分发
如上图所示:当前有个系统有个商家端还有三个用户端,商家端发布一个商品需要在三个用户端上显示,如果此时商家修改一个商品的价格,三个系统都需要同步修改对应的价格。此时就可以使用消息中间件,商家端修改了价格只需要往MQ中丢三条消息(包含商品的最终价格),然后三个分系统分别监听,拿到消息时候在修改对应的价格(应为消息中就有对应的价格所以这里还不需要访问数据库)
常见的一些消息中间件
消息中间件对比
这里采取的是二进制文件安装(安装的是4.7.0版本)
管理控制台的安装:
入门案例(同步发送消息)
生产者
DefaultMQProducer
)Message
)//1 创建一个生产者
DefaultMQProducer producer = new DefaultMQProducer("tudou");
//2 指定NameServer的地址
producer.setNamesrvAddr("192.168.88.130:9876");
//3 启动生产者
producer.start();
for (int i = 0; i < 100; i++) {
// 4 创建消息对象
Message msg = new Message("tudou", ("hello tudou" + i).getBytes());
// 5 发送消息
SendResult result = producer.send(msg);
System.out.println(result.getSendStatus());
System.out.println("result.getMsgId() = " + result.getMsgId());
}
//6 关闭应用程序
producer.shutdown();
复制代码
控制台结果:
消费者
//1. 创建消费者对象
DefaultMQPushConsumer defaultMQPushConsumer = new DefaultMQPushConsumer("tudou");
//2. 设置NameServer地址
defaultMQPushConsumer.setNamesrvAddr("192.168.88.130:9876");
//3. 指定消费主题
defaultMQPushConsumer.subscribe("tudou","*");
//4. 冲那个位置开始消费
defaultMQPushConsumer.setConsumeFromWhere(ConsumeFromWhere.CONSUME_FROM_LAST_OFFSET);
//5. 指定一个监听器,并发消费消息
defaultMQPushConsumer.registerMessageListener(new MessageListenerConcurrently() {
@Override
public ConsumeConcurrentlyStatus consumeMessage(List list, ConsumeConcurrentlyContext consumeConcurrentlyContext) {
for (MessageExt msg:list) {
System.out.println("msgId =" + msg.getMsgId() + "----- msgTopic =" + msg.getTopic() + "----- msgBody =" + new String(msg.getBody()));
}
//这里需要返回消费成功告诉消息中间件否则该条消息还会推送给你
return ConsumeConcurrentlyStatus.CONSUME_SUCCESS;
}
});
//6. 启动消费者
defaultMQPushConsumer.start();
复制代码
控制台效果:
发送消息
下面就是发送消息的两种方式
同步消息
这里的同步消息并不是说等到消息被消费后的同步,而是消息丢到消息中间件后的结果同步。只有等到上一个消息返回状态之后才能继续发送下一条消息。 上文中说到的例子就是一个同步发送消息的案例
如下图所示:
如上图所示send方法会返回一个SendResult
对象则是同步消息
异步消息
发送消息不需要等到上一条消息发送完成,但是也会给到一个发送状态的回调
producer.send(msg,new SendCallback(){
@Override
public void onSuccess(SendResult sendResult) {
//消息发送成功
}
@Override
public void onException(Throwable throwable) {
//发送消息异常
}
});
复制代码
一次性发送消息
生产者只需要将消息丢出去就行,不会受到发送消息后的状态,一般用于日志发送等,有点像UDP,只管把你跑出去剩下的我不管producer.sendOneway(msg);
生产者组
同一类Producer的集合,这类Producer发送同一类消息且发送逻辑一致。如果发送的是事务消息且原始生产者在发送之后崩溃,则Broker服务器会联系同一生产者组的其他生产者实例以提交或回溯消费。
消息封装(Tag、Key)
Tag:用来给消息进行标记, 可以通过Tag对消息进行分类,
Key: 可以设置消息的一个唯一ID, 用于区分每个消息的标志, 业务ID 并且在管理控制台 , 可以通过Key进行消息的查询跟踪把不同类型的消息交给不同的消费者进行消费
接收消息
DefaultLitePullConsumer defaultLitePullConsumer = new DefaultLitePullConsumer("tudou1");
defaultLitePullConsumer.setNamesrvAddr("192.168.88.130:9876");
//3. 指定消费主题
defaultLitePullConsumer.subscribe("tudou","*");
//4. 冲那个位置开始消费
defaultLitePullConsumer.setConsumeFromWhere(ConsumeFromWhere.CONSUME_FROM_LAST_OFFSET);
defaultLitePullConsumer.start();
while (true){
List msgs = defaultLitePullConsumer.poll();
for (MessageExt msg : msgs) {
System.out.println("接收到的消息:" + new String(msg.getBody()));
}
}
复制代码
DefaultMQPushConsumer defaultMQPushConsumer = new DefaultMQPushConsumer("tudou");
consumer.setMessageModel(MessageModel.CLUSTERING);
consumer.setMessageModel(MessageModel.BROADCASTING);
defaultLitePullConsumer.setConsumeFromWhere(ConsumeFromWhere.CONSUME_FROM_LAST_OFFSET);
consumer.setConsumeTimestamp("20200612083300");
从指定的时间开始消费, 如果不指定, 则默认从半个小时前的数据开始消费顺序消息
顺序消息指的是消费者能够按照消息发送的消息一次消费。实现原理就是指定队列
如果消息队列只有一个的情况下其实默认就是顺序的,应为在默认的情况下消息发送会采取Round Robin轮询方式把消息发送到不同的queue(分区队列);而消费消息的时候从多个queue上拉取消息,这种情况发送和消费是不能保证顺序。但是如果控制发送的顺序消息只依次发送到同一个queue中,消费的时候只从这个queue上依次拉取,则就保证了顺序。当发送和消费参与的queue只有一个,则是全局有序;如果多个queue参与,则为分区有序,即相对每个queue,消息都是有序的。所以这里就衍生除了两种概念:
MessageQueue messageQueue = new MessageQueue("queue_1","xzkjdeMac-mini.local",1);
SendResult result = producer.send(msg,messageQueue);
复制代码
和平时发送消息不同的是需要添加一个指定队列 可以发现只有queueId=1的队列才有消息延时消息
通过设置messageDelayLevel的等级来实现消息的延迟,默认的等级有18个等级1~18分别对应着1s 5s 10s 30s 1m 2m 3m 4m 5m 6m 7m 8m 9m 10m 20m 30m 1h 2h,如下图所示:
这里需要注意的是RocketMQ不能随意设置延时时间,但是可以修改配置增加等级。
大概的原理是:延迟的消息不会立马放到消息队列中而是会放到一个Topic叫SCHEDULE_TOPIC_XXXX,然后延迟的等级对应这队列如下图所示:
时间到了之后再将消息转到对应的正式队列
对应代码如下:
// 4 创建消息对象
Message msg = new Message("delay_1","testTag", ("hello tudou" + i).getBytes());
msg.setDelayTimeLevel(6);
// 5 发送消息
SendResult result = producer.send(msg);
复制代码
过滤消息
Tag消息过滤defaultMQPushConsumer.subscribe("delay_1","TestA || TestB");
如下图所示只消费了TestA和TestB标签的消息
SQL消息过滤
RocketMQ只定义了一些基本语法来支持这个特性。你也可以很容易地扩展它。
常量支持类型为:
只有使用push模式的消费者才能用使用SQL92标准的sql语句,接口如下:public void subscribe(finalString topic, final MessageSelector messageSelector)
一般使用SQL消息过滤之前是需要给消息赋值一些属性的然后用这些属性进行判断过滤 但是需要注意的是默认情况下是不支持属性过滤的需要通过配置参数开启在配置文件中配置对应的参数(conf/broker.conf):enablePropertyFilter=true如下图所示:
不配置会直接报错 这里需要注意的是如果换了配置启动命令中需要指定配置文件如下:nohup bin/mqbroker -c conf/broker.conf -n 127.0.0.1:9876 &
否则走的还是默认的配置 最后可以跑一下代码测试:
生产者:
//1 创建一个生产者
DefaultMQProducer producer = new DefaultMQProducer("tudou1");
//2 指定NameServer的地址
producer.setNamesrvAddr("127.0.0.1:9876");
//3 启动生产者
producer.start();
for (int i = 0; i < 5; i++) {
// 4 创建消息对象
Message msg = new Message("delay_1","TestB", ("hello tudou" + i).getBytes());
int score = 100 - i;
msg.putUserProperty("score",String.valueOf(score));
msg.putUserProperty("name","测试同学-"+i);
// 5 发送消息
SendResult result = producer.send(msg);
System.out.println(result.getSendStatus());
System.out.println("result.getMsgId() = " + result.getMsgId());
}
//6 关闭应用程序
producer.shutdown();
复制代码
消费者:
//1. 创建消费者对象
DefaultMQPushConsumer defaultMQPushConsumer = new DefaultMQPushConsumer("tudou_1");
//2. 设置NameServer地址
defaultMQPushConsumer.setNamesrvAddr("127.0.0.1:9876");
//3. 指定消费主题
defaultMQPushConsumer.subscribe("delay_1", MessageSelector.bySql("score > 98"));
//4. 冲那个位置开始消费
defaultMQPushConsumer.setConsumeFromWhere(ConsumeFromWhere.CONSUME_FROM_LAST_OFFSET);
//5. 指定一个监听器,并发消费消息
defaultMQPushConsumer.registerMessageListener(new MessageListenerConcurrently() {
@Override
public ConsumeConcurrentlyStatus consumeMessage(List list, ConsumeConcurrentlyContext consumeConcurrentlyContext) {
for (MessageExt msg:list) {
System.out.println("name =" + msg.getUserProperty("name") + "----- score =" + msg.getUserProperty("score") + "----- msgBody =" + new String(msg.getBody()));
}
return ConsumeConcurrentlyStatus.CONSUME_SUCCESS;
}
});
//6. 启动消费者
defaultMQPushConsumer.start();
复制代码
消费结果:
只消费了score大于98的消息
批量消息
批量消息每次发送消息不能超过4MB
添加依赖
org.apache.rocketmq
rocketmq-spring-boot-starter
2.2.2
复制代码
生产者简单示例代码
@RestController
public class TestController {
@Resource
private RocketMQTemplate rocketMQTemplate;
@GetMapping("/send")
public String test(String msg){
SendResult result = rocketMQTemplate.syncSend("test",msg);
return JSON.toJSONString(result);
}
}
复制代码
执行结果如下:
消费者简单示例代码
@Component
@RocketMQMessageListener(
consumerGroup = "test2",
topic = "test")
public class Custom implements RocketMQListener {
@Override
public void onMessage(String s) {
System.out.println("接收到的消息 ---- " + s);
}
}
复制代码
这里需要注意的是消费者组只能有一个否则报错
执行结果:
其他
其他的还有发送同步消息、异步消息、一次性消息这都比较简单我们可以看到明确的方法名如下图所示:
这里就不过多的演示了也就是简单的方法调用
rocketMQTemplate.convertAndSend("01-boot-hello:TagB",message,map);
SendResult result = rocketMQTemplate.syncSend("test:Tag_A",msg);
//在消息的头信息中设置key
Message> message = MessageBuilder.withPayload(msg).setHeader(MessageConst.PROPERTY_KEYS, "testKey").build();
rocketMQTemplate.send("test_key", message);
复制代码
Map map=new HashMap<>();
//用户自定义属性
map.put("age", 18);
map.put("name", "tudou");
rocketMQTemplate.convertAndSend("test_tag:TagB",message,map);
复制代码
也可以像上文设置key一样直接setHeader同样的key也可以使用这种方式设置rocketMQTemplate.getProducer()
获取该对象Message message1 = MessageBuilder.withPayload(msg).build();
技术架构
上图是官网提供的一个架构图,从上图大概可以看出RocketMQ主要是由以下几个部分组成的:(上文也有提及)
Producer:消息发布的角色,支持分布式集群方式部署。Producer通过MQ的负载均衡模块选择相应的Broker集群队列进行消息投递,投递的过程支持快速失败并且低延迟。
Consumer:消息消费的角色,支持分布式集群方式部署。支持以push推,pull拉两种模式对消息进行消费。同时也支持集群方式和广播方式的消费,它提供实时消息订阅机制,可以满足大多数用户的需求。
NameServer:NameServer是一个非常简单的Topic路由注册中心,其角色类似Dubbo中的zookeeper,支持Broker的动态注册与发现。主要包括两个功能:Broker管理,NameServer接受Broker集群的注册信息并且保存下来作为路由信息的基本数据。然后提供心跳检测机制,检查Broker是否还存活;路由信息管理,每个NameServer将保存关于Broker集群的整个路由信息和用于客户端查询的队列信息。然后Producer和Conumser通过NameServer就可以知道整个Broker集群的路由信息,从而进行消息的投递和消费。NameServer通常也是集群的方式部署,各实例间相互不进行信息通讯。Broker是向每一台NameServer注册自己的路由信息,所以每一个NameServer实例上面都保存一份完整的路由信息。当某个NameServer因某种原因下线了,Broker仍然可以向其它NameServer同步其路由信息,Producer,Consumer仍然可以动态感知Broker的路由的信息。
BrokerServer:Broker主要负责消息的存储、投递和查询以及服务高可用保证,为了实现这些功能,Broker包含了以下几个重要子模块。
部署架构
RocketMQ 网络部署特点
NameServer是一个几乎无状态节点,可集群部署,节点之间无任何信息同步。(应为broker的心跳机制会想每一个NameServer发送信息,所以NameServer直接是不需要再同步消息的)
Broker部署相对复杂,Broker分为Master与Slave,一个Master可以对应多个Slave,但是一个Slave只能对应一个Master,Master与Slave 的对应关系通过指定相同的BrokerName,不同的BrokerId 来定义,BrokerId为0表示Master,非0表示Slave。Master也可以部署多个。每个Broker与NameServer集群中的所有节点建立长连接,定时注册Topic信息到所有NameServer。 注意:当前RocketMQ版本在部署架构上支持一Master多Slave,但只有BrokerId=1的从服务器才会参与消息的读负载。
Producer与NameServer集群中的其中一个节点(随机选择)建立长连接,定期从NameServer获取Topic路由信息,并向提供Topic 服务的Master建立长连接,且定时向Master发送心跳。Producer完全无状态,可集群部署。
Consumer与NameServer集群中的其中一个节点(随机选择)建立长连接,定期从NameServer获取Topic路由信息,并向提供Topic服务的Master、Slave建立长连接,且定时向Master、Slave发送心跳。Consumer既可以从Master订阅消息,也可以从Slave订阅消息,消费者在向Master拉取消息时,Master服务器会根据拉取偏移量与最大偏移量的距离(判断是否读老消息,产生读I/O),以及从服务器是否可读等因素建议下一次是从Master还是Slave拉取。
结合部署架构图,描述集群工作流程:
启动NameServer,NameServer起来后监听端口,等待Broker、Producer、Consumer连上来,相当于一个路由控制中心。
Broker启动,跟所有的NameServer保持长连接,定时发送心跳包。心跳包中包含当前Broker信息(IP+端口等)以及存储所有Topic信息。注册成功后,NameServer集群中就有Topic跟Broker的映射关系。
收发消息前,先创建Topic,创建Topic时需要指定该Topic要存储在哪些Broker上,也可以在发送消息时自动创建Topic。
Producer发送消息,启动时先跟NameServer集群中的其中一台建立长连接,并从NameServer中获取当前发送的Topic存在哪些Broker上,轮询从队列列表中选择一个队列,然后与队列所在的Broker建立长连接从而向Broker发消息。
Consumer跟Producer类似,跟其中一台NameServer建立长连接,获取当前订阅Topic存在哪些Broker上,然后直接跟Broker建立连接通道,开始消费消息。
消息存储
消息存储流程
消息存储架构图中主要有下面三个跟消息存储相关的文件构成。
(1) CommitLog:消息主体以及元数据的存储主体,存储Producer端写入的消息主体内容,消息内容不是定长的。单个文件大小默认1G ,文件名长度为20位,左边补零,剩余为起始偏移量,比如00000000000000000000代表了第一个文件,起始偏移量为0,文件大小为1G=1073741824;当第一个文件写满了,第二个文件为00000000001073741824,起始偏移量为1073741824,以此类推。消息主要是顺序写入日志文件,当文件满了,写入下一个文件;(写入的消息会一直按照文件的顺序写入)。具体存储的位置是可以从配置信息中看到的如下图所示
(2) ConsumeQueue:消息消费队列,引入的目的主要是提高消息消费的性能,由于RocketMQ是基于主题topic的订阅模式,消息消费是针对主题进行的,如果要遍历commitlog文件中根据topic检索消息是非常低效的。Consumer即可根据ConsumeQueue来查找待消费的消息。其中,ConsumeQueue(逻辑消费队列)作为消费消息的索引,保存了指定Topic下的队列消息在CommitLog中的起始物理偏移量offset,消息大小size和消息Tag的HashCode值(如果使用tags过滤的话会使用到,在CommonQueue存储这个信息也比较好理解,就是过滤掉的信息就不需要再冲CommonL了og中查找读取)。consumequeue文件可以看成是基于topic的commitlog索引文件,故consumequeue文件夹的组织方式如下:topic/queue/file三层组织结构,具体存储路径为:$HOME/store/consumequeue/{topic}/{queueId}/{fileName}。同样consumequeue文件采取定长设计,每一个条目共20个字节,分别为8字节的commitlog物理偏移量、4字节的消息长度、8字节tag hashcode,单个文件由30W个条目组成,可以像数组一样随机访问每一个条目,每个ConsumeQueue文件大小约5.72M;
(3) IndexFile:IndexFile(索引文件)提供了一种可以通过key或时间区间来查询消息的方法。Index文件的存储位置是:HOME\store\indexHOME \store\indexHOME\store\index{fileName},文件名fileName是以创建时的时间戳命名的,固定的单个IndexFile文件大小约为400M,一个IndexFile可以保存 2000W个索引,IndexFile的底层存储设计为在文件系统中实现HashMap结构,故rocketmq的索引文件其底层实现为hash索引。
在上面的RocketMQ的消息存储整体架构图中可以看出,RocketMQ采用的是混合型的存储结构,即为Broker单个实例下所有的队列共用一个日志数据文件(即为CommitLog)来存储。RocketMQ的混合型存储结构(多个Topic的消息实体内容都存储于一个CommitLog中)针对Producer和Consumer分别采用了数据和索引部分相分离的存储结构,Producer发送消息至Broker端,然后Broker端使用同步或者异步的方式对消息刷盘持久化,保存至CommitLog中。只要消息被刷盘持久化至磁盘文件CommitLog中,那么Producer发送的消息就不会丢失。正因为如此,Consumer也就肯定有机会去消费这条消息。当无法拉取到消息后,可以等下一次消息拉取,同时服务端也支持长轮询模式,如果一个消息拉取请求未拉取到消息,Broker允许等待30s的时间,只要这段时间内有新消息到达,将直接返回给消费端。这里,RocketMQ的具体做法是,使用Broker端的后台服务线程—ReputMessageService不停地分发请求并异步构建ConsumeQueue(逻辑消费队列)和IndexFile(索引文件)数据。
页缓存与内存映射
页缓存-PageCache是OS对文件的缓存,用于加速对文件的读写。一般来说,程序对文件进行顺序读写的速度几乎接近于内存的读写速度,主要原因就是由于OS使用PageCache机制对读写访问操作进行了性能优化,将一部分的内存用作PageCache。对于数据的写入,OS会先写入至Cache内,随后通过异步的方式由pdflush内核线程将Cache内的数据刷盘至物理磁盘上。对于数据的读取,如果一次读取文件时出现未命中PageCache的情况,OS从物理磁盘上访问读取文件的同时,会顺序对其他相邻块的数据文件进行预读取。
如上图所示,此时需要读取9这个数据第一次用户态是回去访问页缓存如果此时有数据则直接命中,如果也缓存中没有数据则新开一页大小是4kb,然后此时也缓存在向硬盘拉取数据此时会将相邻的一些数据全部拉取过来存到也缓存中然后再将用户需要的数据返回。
写入的时候首先还是需要先判断页缓存中是否存在如果存在则修改否则新建一页添加数据对于被修改的页缓存,内核会定时把这些页缓存刷新到文件中。
所以为什么上文说到的使用顺序存储就是应为页缓存能极大的提交系统IO的速度
在RocketMQ中,ConsumeQueue逻辑消费队列存储的数据较少,并且是顺序读取,在page cache机制的预读取作用下,Consume Queue文件的读性能几乎接近读内存,即使在有消息堆积情况下也不会影响性能。而对于CommitLog消息存储的日志数据文件来说,读取消息内容时候会产生较多的随机访问读取,严重影响性能。如果选择合适的系统IO调度算法,比如设置调度算法为“Deadline”(此时块存储采用SSD的话),随机读的性能也会有所提升。
另外,RocketMQ主要通过MappedByteBuffer对文件进行读写操作。其中,利用了NIO中的FileChannel模型将磁盘上的物理文件直接映射到用户态的内存地址中(这种Mmap的方式减少了传统IO将磁盘文件数据在操作系统内核地址空间的缓冲区和用户应用程序地址空间的缓冲区之间来回进行拷贝的性能开销),将对文件的操作转化为直接对内存地址进行操作,从而极大地提高了文件的读写效率(正因为需要使用内存映射机制,故RocketMQ的文件存储都使用定长结构来存储,方便一次将整个文件映射至内存)。
Mmap补充:mmap是一种内存映射文件的方法,即将一个文件或者其它对象映射到进程的地址空间,实现文件磁盘地址和进程虚拟地址空间中一段虚拟地址的一一对映关系,正常情况下一个进程需要修改或者读取一个文件的数据是需要先从页缓存中读取过来就是需要内核态拷贝到用户态或者需要冲用户态拷贝到内核态,如果有了映射关系则禁止直接通过虚拟内存中拿到数据在内存中的地址可以直接通过指针读取和修改(当然无果内存中没有数据则会冲磁盘读取过来之后在进行读写)
消息刷盘
消息发送到Broker之后的存储流程如下图所示
先走到内存,然后为了数据的安全做持久化存储所以就在存到本地文件。 这里就牵扯到了同步刷盘和异步输盘:
负载均衡
Producer负载均衡
其实就是通过轮询发送到不同的队列上,让每个队列的消息数很平均
Consumer负载均衡
应为队列中的消息都是轮询插入的所以只需要消费者集群监听相同的队列就能实现负载均衡,这里需要注意的是队列数量要大于消费者服务的数量否则会出现某个消费服务没有消息可以消费
源码下载地址:github.com/apache/rock…
NameServer启动流程
//定时任务查询broker状态
this.scanExecutorService.scheduleAtFixedRate(NamesrvController.this.routeInfoManager::scanNotActiveBroker,
5, this.namesrvConfig.getScanNotActiveBrokerInterval(), TimeUnit.MILLISECONDS);
//定时加载kv配置信息
this.scheduledExecutorService.scheduleAtFixedRate(NamesrvController.this.kvConfigManager::printAllPeriodically,
1, 10, TimeUnit.MINUTES);
复制代码
这里需要注意的是namesrv检测broker状态的机制并不是向broker发送消息,而是利用broker查看namesrv状态的机制来实现的,broker查看namesrv状态的机制是发送消息而且会发送当前时间,而namesrv会把每次broker过来的消息保存下来启动一个定时器每个几秒查看一下最新的broker时间和当前时间差值大于120秒则namesrv则会任务broker服务不可用Broker启动流程