RabbbitMQ基本使用及其五种工作模型

这篇具有很好参考价值的文章主要介绍了RabbbitMQ基本使用及其五种工作模型。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

初识MQ

同步通讯和异步通讯

        什么是同步通讯呢?举个例子,你认识了一个小姐姐,聊的很火热,于是你们慢慢开始打电话,视频聊天,这种方式就成为同步通讯,那什么是一部通讯呢,同样的,你认识了多个小姐姐,你和他们进行文字聊天,这时候你一个人可以和多个人聊天,这就是异步通讯。我们之前进行服务间调用时使用的RestTemplaste,Feign就是同步调用。

同步调用的优缺点

        优点:时效性强,可以立即得到回复,就像你打视频一样

        缺点:假如你一个项目中存在很多业务,相互之间进行调用,如果你增加了新的需求,此时因为原本代码是同步调用,代码耦合度很高,于是乎修改代码变得十分繁琐,并且一个业务可能会调用很多服务,只有上一个服务调用完了,才能到下一个服务,等待的过程中就造成了资源浪费,性能下降,如果当前调用的服务失败,还可能会导致级联失败,服务雪崩。

RabbbitMQ基本使用及其五种工作模型,java,开发语言,RabbitMQ,MQ,五种工作模式


异步调用的优缺点

优点:

1,代码耦合度低,因为异步调用是采取事件驱动来实现,当请求进来之后到达Broker之后Broker通知各自微服务去执行,服务间不在需要相互调用。

2,吞吐量提升,因为异步调用不像同步调用那样每个服务需要等待上游完成调用。

3,故障隔离,服务之间相互不进行调用,即使你挂了也跟我没关系。

4,流量削峰,假如同时又大量请求,但是你的服务处理请求能力是有限的,于是Broker会净请求先拦截,看服务又能力处理多少请求,就拿多少请求,不会一次性全部发布订阅。

缺点:

1,对Broker的依赖十分高,对他的可靠性,安全性,吞吐能力要求很高,万一他挂了......

2,服务之间相互调用关系不清晰,业务没有明显的流程线,代码出问题不容易排查。

RabbbitMQ基本使用及其五种工作模型,java,开发语言,RabbitMQ,MQ,五种工作模式

所以,需要根据场景来选择同步还是异步,一般大多数需要同步。

什么是MQ?

MQ(MessageQueue),中文是消息队列,也就是存放消息的队列,也就是时间驱动中的Broker.

常见的MQ对比:

RabbbitMQ基本使用及其五种工作模型,java,开发语言,RabbitMQ,MQ,五种工作模式

我们这里选择RabbitMQ

RabbitMQ安装

RabbitMQ是基于Erlang语言开发的开源消息中间件,因此它具备强大的并发处理能力

官网地址:RabbitMQ: One broker to queue them all | RabbitMQhttps://www.rabbitmq.com/

这里我们只用Docker来安装RabbitMQ(快捷方便)

1,拉取RabbitMQ的镜像

docker pull rabbitmq:3-management 

2,运行RabbitMQ容器

docker run \
 -e RABBITMQ_DEFAULT_USER=你的账户名\
 -e RABBITMQ_DEFAULT_PASS=你的密码\
 --name mq \  
 --hostname mq1 \
 -p 15672:15672 \
 -p 5672:5672 \
 -d \
 rabbitmq:3-management

注意:账户名和密码是自己定义的,15672是RabbitMQ的管理端端口,5672是RabbitMQ通讯的端口。

3,在浏览器输入IP地址:15672,输入帐号名和密码登录

RabbbitMQ基本使用及其五种工作模型,java,开发语言,RabbitMQ,MQ,五种工作模式

至此,安装成功

我们可以看到界面有好多目录,具体作用如下

channel:用来操作mq的工具

exchange:路由消息到队列中

queue:缓存消息

virtual host:虚拟主机,是对queue,exchange等资源的逻辑分组

MQ的整体结构:

消息发送者将消息发送到交换机,交换机将其路由到队列,消费者从队列中取走消息

RabbbitMQ基本使用及其五种工作模型,java,开发语言,RabbitMQ,MQ,五种工作模式

MQ中常见消息模型

大致可以分为两类

第一类:基本消息队列(BasicQueue),工作消息队列(WorkQueue),他们都是最基本的消息队列

RabbbitMQ基本使用及其五种工作模型,java,开发语言,RabbitMQ,MQ,五种工作模式RabbbitMQ基本使用及其五种工作模型,java,开发语言,RabbitMQ,MQ,五种工作模式

第二类:发布订阅(publish,Subscribe)根据交换机类型分为三种

Fanout Exchange(广播),Direct Exchange(路由),Topic Exchange(主题)

RabbbitMQ基本使用及其五种工作模型,java,开发语言,RabbitMQ,MQ,五种工作模式 RabbbitMQ基本使用及其五种工作模型,java,开发语言,RabbitMQ,MQ,五种工作模式 RabbbitMQ基本使用及其五种工作模型,java,开发语言,RabbitMQ,MQ,五种工作模式

更多可参考官方文档: 

RabbitMQ Tutorials | RabbitMQ

RabbitMQ入门案例

我们使用RabbitMQ参考官方文档,完成一个hello world案例

引入依赖:

     <!--包含RabbitMQ-->
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-amqp</artifactId>
        </dependency>

创建一个测试类publisher,用来发送消息代码如下:

public class PublisherTest {
    @Test
    public void testSendMessage() throws IOException, TimeoutException {
        // 1.建立连接
        ConnectionFactory factory = new ConnectionFactory();
        // 1.1.设置连接参数,分别是:主机名、端口号、vhost、用户名、密码
        factory.setHost("192.168.*.*");
        factory.setPort(5672);
        factory.setVirtualHost("/");
        factory.setUsername("***");
        factory.setPassword("****");
        // 1.2.建立连接
        Connection connection = factory.newConnection();
        // 2.创建通道Channel
        Channel channel = connection.createChannel();
        // 3.创建队列
        String queueName = "simple.queue";
        channel.queueDeclare(queueName, false, false, false, null);
        // 4.发送消息
        String message = "hello, rabbitmq!";
        channel.basicPublish("", queueName, null, message.getBytes());
        System.out.println("发送消息成功:【" + message + "】");
        // 5.关闭通道和连接
        channel.close();
        connection.close();

    }
}

Debug执行观察:

1,连接工厂初始化完成之后,他会创建一个连接,连接上RabbitMQ ,这时界面显示如下

RabbbitMQ基本使用及其五种工作模型,java,开发语言,RabbitMQ,MQ,五种工作模式

2,之后他会创建一个Channel ,用于操作RabbbitMQ

RabbbitMQ基本使用及其五种工作模型,java,开发语言,RabbitMQ,MQ,五种工作模式

3,之后会创建一个我们定义好的消息队列simple.queue:

RabbbitMQ基本使用及其五种工作模型,java,开发语言,RabbitMQ,MQ,五种工作模式

4,之后发送消息hello,rabbitmq!

RabbbitMQ基本使用及其五种工作模型,java,开发语言,RabbitMQ,MQ,五种工作模式

5,完成之后关闭连接,但是消息依旧存在与消息队列中

之后创建一个测试类consumer,用来接受消息代码如下:

public class ConsumerTest {

    public static void main(String[] args) throws IOException, TimeoutException {
        // 1.建立连接
        ConnectionFactory factory = new ConnectionFactory();
        // 1.1.设置连接参数,分别是:主机名、端口号、vhost、用户名、密码
        factory.setHost("192.168.*.*");
        factory.setPort(5672);
        factory.setVirtualHost("/");
        factory.setUsername("****");
        factory.setPassword("****");
        // 1.2.建立连接
        Connection connection = factory.newConnection();
        // 2.创建通道Channel
        Channel channel = connection.createChannel();
        // 3.创建队列
        String queueName = "simple.queue";
        channel.queueDeclare(queueName, false, false, false, null);
        // 4.订阅消息
        channel.basicConsume(queueName, true, new DefaultConsumer(channel){
            @Override
            public void handleDelivery(String consumerTag, Envelope envelope,
                                       AMQP.BasicProperties properties, byte[] body) throws IOException {
                // 5.处理消息
                String message = new String(body);
                System.out.println("接收到消息:【" + message + "】");
            }
        });
        System.out.println("等待接收消息。。。。");
    }
}

 跟publisher一样创建工厂,建立连接,这里需要说明的是,之所以还要创建一个队列是因为在实际执行过程中,发布者和消费者又可能执行顺序不一致,所以我们消费者也需要创建一个队列,不过这个队列只会有一个,如果创建了就不再创建。之后消费者接受消息处理,消息队列中消息清空。

RabbbitMQ基本使用及其五种工作模型,java,开发语言,RabbitMQ,MQ,五种工作模式

RabbbitMQ基本使用及其五种工作模型,java,开发语言,RabbitMQ,MQ,五种工作模式

根据上述代码我们可以看到官方的demo确实是有带你复杂繁琐,实在是很不友好啊,于是我们使用一种简单的方式来操作RabbitMQ!

SpringAMQP

AMQP(Advanced Message Queuing Protocal):高级消息队列协议,是应用程序之间传递业务消息的开放标准/规范,和语言和平台无关。

SpringAMQP:是基于AMQP协议定义的一套API规范,提供了模板用来发送和接受消息,包含两部分,其中spring-amop是基础抽象,spring-rabbit是底层的默认实现。

SpringAMQP实现基础消息队列功能

消息发送

1,引入依赖

  <!--AMQP依赖,包含RabbitMQ-->
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-amqp</artifactId>
        </dependency>

2,在配置文件中配置RabbitMQ的信息

spring:
  rabbitmq:
    host: 192.168.121.10 #主机名
    port: 5672 #端口
    virtual-host: /  #虚拟主机
    username: *** #用户名
    password: **** #密码
  

3,使用RabbitTemplate来发送消息(spring提供的发送消息的模板)

这里我们编写一个测试类来测试

@SpringBootTest
@RunWith(SpringRunner.class)
public class AMQPTest {
    @Autowired
    private RabbitTemplate rabbitTemplate;

    @Test
    public void sendMessage(){
        String queueName="simple.queue";//定义队列名
        String message="hello SpringAMQP!!";
        rabbitTemplate.convertAndSend(queueName,message);
    }
}

 可以看到消息成功发送

RabbbitMQ基本使用及其五种工作模型,java,开发语言,RabbitMQ,MQ,五种工作模式

消息接收

依旧是先引依赖,添加RabbitMQ相关配置

之后书写监听消息代码,创建一个类,交给spring管理,定义一个方法加@RabbitListener注解指定接受消息的队列,可以传递数组,指定多个队列

@Component
public class ListenerMessage {
    @RabbitListener(queues = {"simple.queue"})//可以监听多个队列
    public void ListenerSimpleQueue(String message){
        System.out.println("接收到的消息是;"+message);
    }
}

可以看到成功接受消息

RabbbitMQ基本使用及其五种工作模型,java,开发语言,RabbitMQ,MQ,五种工作模式

 WordQueue 工作队列

Work模型-多个消费者绑定到同一个队列,同一个消息只会被同一个消费者处理

RabbbitMQ基本使用及其五种工作模型,java,开发语言,RabbitMQ,MQ,五种工作模式

我们来做一个测试,在一秒内发送50条消息,定义两个消费者,同时处理,一个消费者每秒处理50条消息,另一个消费者每秒处理20个消息,按照常理来说应该是多劳多得,也就是处理能力强的处理更多消息,弱的处理更少消息,我们修改之前的代码:

修改之后的消息发送代码:

@Test
    public void sendWorkMessage() throws InterruptedException {
        String queueName="simple.queue";//定义队列名
        String message="Work Message__";
        for (int i = 1; i <= 50; i++) {
            rabbitTemplate.convertAndSend(queueName,message+i);
            Thread.sleep(20);
        }
    }

修改之后的消息接收代码:

@Component
public class ListenerMessage {

    @RabbitListener(queues = {"simple.queue"})//可以监听多个队列
    public void ListenerWorkSimpleQueue1(String message) throws InterruptedException {
        System.out.println("consumer1接收到的消息是;"+message+"  时间:"+LocalDateTime.now());
        Thread.sleep(20);
    }
    @RabbitListener(queues = {"simple.queue"})//可以监听多个队列
    public void ListenerWorkSimpleQueue2(String message) throws InterruptedException {
        System.err.println("consumer2接收到的消息是;"+message+"  时间:"+LocalDateTime.now());
        Thread.sleep(200);
    }
}

代码执行结果显示,consumer1和consumer2处理的消息是一样的并不会向我们设想的那样,能力强的处理多能力弱的处理少,是什么原因导致的呢??

这是由于RabbitMQ的消息预取机制,就是说在消息到达消息队列的时候两个消费者会分别从消息队列中一次性取完所有的消息理论上来说是无上限的,所以我们需要修改机制让消费者一次性例如说取一个消息,等这个消息处理完之后在去取下一个即可,

在配置文件中修改如下:

spring:
  rabbitmq:
    host: 192.168.101.100 #主机名
    port: 5672 #端口
    virtual-host: /  #虚拟主机
    username: qmlx #用户名
    password: QMLX-9999 #密码
    listener:
      direct:
        prefetch: 1  #每次取一个消息,取完之后在取

 发布(Publish)订阅(Subscribe)

发布订阅模式和之前案例的区别就是上述模型一个消息只能发送给一个consumer,而发布订阅模型则是将同一个消息发送给多个消费者,实现方式是假如exchange(交换机)

 RabbbitMQ基本使用及其五种工作模型,java,开发语言,RabbitMQ,MQ,五种工作模式

常见的exchange类型包括:

Fanout:广播

Direct:路由

Topic:话题

注意:exchange只会负责消息路由,而不是存储,路由失败则消息丢失

🧇发布订阅-Fanout Exchange

Fanout Exchange他会将接收到的消息路由到每一个与其绑定的queue,广播模式。

实现思路:

1,在consumer服务中,编写一个配置类,声明两个队列,一个交换机,并将交换机绑定到队列上


@Configuration
public class FanoutConfig {
    @Bean
    public FanoutExchange fanoutExchange(){
        return  new FanoutExchange("fanout.exchange");
    }

    //定义两个队列
    @Bean
    public Queue fanoutQueue1(){
        return new Queue("fanout.queue1");
    }
    @Bean
    public Queue fanoutQueue2(){
        return new Queue("fanout.queue2");
    }

    //将队列绑定在交换机上面
    @Bean
    public Binding bindingQueue1(Queue fanoutQueue1,FanoutExchange fanoutExchange){
    //按照类型和名称传入参数
        return BindingBuilder.bind(fanoutQueue1).to(fanoutExchange);
    }

    @Bean
    public Binding bindingQueue2(Queue fanoutQueue2,FanoutExchange fanoutExchange){
        return BindingBuilder.bind(fanoutQueue2).to(fanoutExchange);
    }


}

启动项目之后spring会自动加载交换机和队列

RabbbitMQ基本使用及其五种工作模型,java,开发语言,RabbitMQ,MQ,五种工作模式

RabbbitMQ基本使用及其五种工作模型,java,开发语言,RabbitMQ,MQ,五种工作模式

 2,编写consumer代码,监听定义好的两个队列


@Component
public class ListenerMessage {

     @RabbitListener(queues = {"fanout.queue1"})
    public void ListenerFanoutQueue1(String message){
        System.out.println("consumer1接受到的消息是:"+message);
    }

    @RabbitListener(queues = {"fanout.queue2"})
    public void ListenerFanoutQueue2(String message){
        System.err.println("consumer2接受到的消息是:"+message);
    }
}

3,编写publisher代码,发送消息

之前消息是发送到队列中,现在消息发送给交换机

    @Test
    public void sendFanoutMessage() throws InterruptedException {
        String exchangeName="fanout.exchange";//定义队列名
        String message="FanoutExchange Message!!";
        rabbitTemplate.convertAndSend(exchangeName,null,message);

    }

可以看到两个消费者同时接受到了消息!!!

RabbbitMQ基本使用及其五种工作模型,java,开发语言,RabbitMQ,MQ,五种工作模式

交换机的作用:

1,接受publisher发送的消息

2,将消息按照规则路由发送给每一个与之绑定的队列

3,不能缓存信息,路由失败则消息丢失

🧇发布订阅-Direct Exchange

Direct Exchange:交换机将接收到的消息根据路由规则到指定的Queue,称之为路由模式(routes)

        每一个Queue都和Exchange设定一个BindingKey

        发布者发送消息时,指定消息的BindingKey

        Exchange将消息路由到BindKey和消息的BindingKey一致的队列

RabbbitMQ基本使用及其五种工作模型,java,开发语言,RabbitMQ,MQ,五种工作模式

实现思路:

1,指定消息接收者绑定交换机和队列,但是需要指定BindingKey

这次直接使用注解实现,不必那莫繁琐

@Component
public class ListenerDirectMessage {

    @RabbitListener(bindings = @QueueBinding(
            value = @Queue(name = "direct.queue1"),
            exchange = @Exchange(name = "direct.exchange",type = ExchangeTypes.DIRECT),
            key = {"red","blue"}
    ))
    public void listensterDirectQueue1(String message){
        System.out.println("消费者1接收到Direct的消息是:"+message);
    }

    @RabbitListener(bindings = @QueueBinding(
            value = @Queue(name = "direct.queue2"),
            exchange = @Exchange(name = "direct.exchange",type = ExchangeTypes.DIRECT),
            key = {"red","blue"}
    ))
    public void listensterDirectQueue2(String message){
        System.out.println("消费者2接收到Direct的消息是:"+message);
    }
}

‘2,定义publisher发布消息

@Test
    public void sendDirectMessage() throws InterruptedException {
        String exchangeName="direct.exchange";//定义队列名
        String message="DirectExchange Message!!";
        String routinfKey="blue";
        rabbitTemplate.convertAndSend(exchangeName,routinfKey,message);
        rabbitTemplate.convertAndSend(exchangeName,null,message);

    }

发送时需要指定routingKey即可,同一个队列定义时可指定多个BindingKey

注意:Direct Exchange只会发送给routingKeyBindingKey一致的队列

🧇发布订阅-Topic Exchange

Topic Exchange:Topic Exchange和Direct key类似,区别在于他的routingKey必须是多个单词的列表,并且必须是以 分割,并且Queue和Exchange指定时课使用通配符

RabbbitMQ基本使用及其五种工作模型,java,开发语言,RabbitMQ,MQ,五种工作模式

实现思路:

1,实现消费者代码


@Component
public class ListenerTopicMessage {

    @RabbitListener(bindings = @QueueBinding(
            value = @Queue(name = "topic.queue1"),
            exchange = @Exchange(name = "topic.exchange",type = ExchangeTypes.TOPIC),
            key = "china.#"
    ))
    public void listensterDirectQueue1(String message){
        System.out.println("消费者1接收到topic的消息是:"+message);
    }

    @RabbitListener(bindings = @QueueBinding(
            value = @Queue(name = "topic.queue2"),
            exchange = @Exchange(name = "topic.exchange",type = ExchangeTypes.TOPIC),
            key = "#.news"
    ))
    public void listensterDirectQueue2(String message){
        System.out.println("消费者2接收到topic的消息是:"+message);
    }
}

2,实现publisher代码

@Test
    public void sendtopictMessage() throws InterruptedException {
        String exchangeName="topic.exchange";//定义队列名
        String message="TopicExchange Message!!";
        String routinfKey="china.news";
        rabbitTemplate.convertAndSend(exchangeName,routinfKey,message);
        //rabbitTemplate.convertAndSend(exchangeName,null,message);

    }

至此RabbitMQ的安装使用及其五种基本工作模式搞定!!!!文章来源地址https://www.toymoban.com/news/detail-852855.html

到了这里,关于RabbbitMQ基本使用及其五种工作模型的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • Linux下Web服务器工作模型及Nginx工作原理详解

    在Linux环境下,Web服务器处理并发连接请求的工作模型主要有阻塞、非阻塞、同步、异步等方式。以下是对各种工作模型的浅析: 同步阻塞 I/O: 类比于在餐厅等饭,需要在取餐处等待,期间不能进行其他事情。 同步非阻塞 I/O: 类比于在餐厅等饭,可以进行其他事情,但需

    2024年02月03日
    浏览(31)
  • TCP模型和工作沟通方式

    我们如何与客户沟通?理科生和技术人员可能在沟通技巧方面有所欠缺。 那么我们如何理解和掌握沟通的原则和技巧呢?我发现TCP网络交互模型很好的描述了沟通的原则和要点。下面我们就从TCP来讲沟通的过程。 TCP的客户端就像客户(甲方),TCP的服务端就如同乙方。网络

    2024年02月15日
    浏览(24)
  • 全面解析大语言模型的工作原理

           当ChatGPT在去年秋天推出时,在科技行业乃至世界范围内引起了轰动。当时,机器学习研究人员尝试研发了多年的语言大模型(LLM),但普通大众并未十分关注,也没有意识到它们变得多强大。        如今,几乎每个人都听说过LLM,并有数千万人用过它们,但是,了

    2024年02月14日
    浏览(22)
  • Rabbitmq入门与应用(二)-RabbitMQ工作模型

    RabbitMQ Tutorials — RabbitMQ Broker RabbitMQ服务。 Connection 生产者或是服务者都需要与Broker建立的TCP连接。 Channel 保持的TCP长连接里面去创建和释放Channel,从而减少资源的消耗。其中Channel是相互隔离的,不能共享。 Queue Queue是生产者与消费者的中间交互队列,生产者发送的消息到达

    2024年02月21日
    浏览(30)
  • 【生活工作经验 十】ChatGPT模型对话初探

    最近探索了下全球大火的ChatGPT,想对此做个初步了解 当今社会,自然语言处理技术得到了迅速的发展,人工智能技术也越来越受到关注。其中,基于深度学习的大型语言模型,如GPT(Generative Pre-trained Transformer)在自然语言处理领域中取得了很大的成功。ChatGPT就是其中的一种

    2023年04月21日
    浏览(31)
  • 利用大语言模型(LLM )提高工作效率

    日常工作就是面向 google/ 百度编程,除了给变量命名是手动输入,大多时候就是通过搜索引擎拷贝别人的代码,或者找到旧项目一段代码拷贝过来使用。这无疑是开发人员的真实写照;然而,通过搜索引擎搜索答案,无疑是粪堆里淘金子,遇到简单的问题,一次搜索点三个连

    2024年02月05日
    浏览(40)
  • 文本生成图像工作简述4--扩散模型、自回归模型、生成对抗网络的对比调研

    基于近年来图像处理和语言理解方面的技术突破,融合图像和文本处理的多模态任务获得了广泛的关注并取得了显著成功。 文本生成图像(text-to-image)是图像和文本处理的多模态任务的一项子任务,其根据给定文本生成符合描述的真实图像,具有巨大的应用潜力,如 视觉推

    2023年04月08日
    浏览(31)
  • 一文了解大模型工作原理——以ChatGPT为例

    2022年11月30日,ChatGPT模型问世后,立刻在全球范围内掀起了轩然大波。无论AI从业者还是非从业者,都在热议ChatGPT极具冲击力的交互体验和惊人的生成内容。这使得广大群众重新认识到人工智能的潜力和价值。对于AI从业者来说,ChatGPT模型成为一种思路的扩充,大模型不再是

    2024年02月09日
    浏览(30)
  • RabbitMQ笔记一7-11章RabbitMQ工作模型【动力节点】

    broker 相当于mysql服务器,virtual host相当于数据库(可以有多个数据库) queue相当于表,消息相当于记录。   消息队列有三个核心要素:  消息生产者 、 消息队列 、 消息消费者 ; 生产者(Producer):发送消息的应用;(java程序,也可能是别的语言写的程序) 消费者(Con

    2023年04月26日
    浏览(26)
  • RabbitMQ工作队列模型详解:实现高效任务分配与消费

    深入探索RabbitMQ的work消息模型,学习如何在多个消费者之间有效分配任务。掌握能者多劳的策略,优化队列消费效率,提升系统性能。

    2024年02月11日
    浏览(38)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包