RabbitMQ之高级特性

news2025/1/15 16:49:54

文章目录

  • 一、消息确认机制
    • 🎉1.1 消息发送确认(生产者)
      • 🔹confirm 确认模式
      • 🔹return 回退模式
    • 🚩1.2 消息接收确认(消费者)
      • 🔸none 自动确认
      • 🔸auto 异常确认
      • 🔸manual 手动确认
  • 二、消费端限流 (prefetch)
  • 三、设置队列参数
    • 🍃3.1 消息TTL过期
    • 🍃3.2 队列最大长度
  • 四、死信队列
    • 🍒4.1 构建死信队列
    • 🍒4.2 模拟死信消息
    • 🍒4.3 实现延迟队列
  • 五、消息追踪
    • 🍀5.1 Firehose: amq.rabbitmq.trace
    • 🍀5.2 rabbitmq_tracing 插件
  • 六、应用问题
    • 🌹6.1 消息可靠性保障
      • 🔸消息补偿机制
    • 🌹6.2 消息幂等性保障
      • 🔹乐观锁解决方案


提示:以下是本篇文章正文内容,RabbitMQ 系列学习将会持续更新

在这里插入图片描述
官网:https://www.rabbitmq.com

一、消息确认机制

RabbitMQ 消息确定主要分为两部分:

  • 第一种是消息发送确认。这种是用来确认生产者将消息发送给交换器,交换器传递给队列的过程中,消息是否成功投递。
    1. 确认发送的第一步是确认是否到达交换器。
    2. 确认发送的第二步是确认是否到达队列。
  • 第二种是消费接收确认。这种是确认消费者是否成功消费了队列中的消息。

在这里插入图片描述

🎉1.1 消息发送确认(生产者)

  • 消息从 producer 到 exchange 则会返回一个 confirmCallback
  • 消息从 exchange 到 queue 投递失败则会返回一个 returnCallback

🔹confirm 确认模式

①开启确认模式

spring:
  rabbitmq:
    addresses: 1.15.76.95
    username: admin
    password: 123456
    virtual-host: /test
    # 消息可靠传递: 开启确认模式
    publisher-confirm-type: correlated
  • publisher-confirm-type: none:表示禁用发布确认模式,默认值,使用此模式之后,不管消息有没有发送到 Broker 都不会触发 ConfirmCallback 回调。
  • publisher-confirm-type: correlated:表示消息成功到达 Broker 后触发 ConfirmCalllBack 回调。
  • publisher-confirm-type: simple:如果消息成功到达 Broker 后一样会触发 ConfirmCalllBack 回调,发布消息成功后使用 rabbitTemplate 调用 waitForConfirms 或 waitForConfirmsOrDie 方法等待 broker 节点返回发送结果,根据返回结果来判定下一步的逻辑。
    注意:waitForConfirmsOrDie 方法如果返回 false 则会关闭 channel 信道,则接下来无法发送消息到 broker。

②设置 ConfirmCallback 函数,然后发送消息。

@Test
void publisher1() {
    // 1.定义回调
    template.setConfirmCallback(new RabbitTemplate.ConfirmCallback() {
        /**
         * confirm方法参数:
         * @param correlationData 相关配置信息
         * @param ack 交换机是否成功收到消息
         * @param cause 交换机接收失败的原因
         */
        @Override
        public void confirm(CorrelationData correlationData, boolean ack, String cause) {
            System.out.println("执行confirm()方法: ");
            if(ack) {
                System.out.println("交换机成功收到消息!");
            }else {
                System.out.println("交换机没有收到消息,失败原因: " + cause);
            }
        }
    });
    // 2.发送消息
    template.convertAndSend("amq.direct", "my-yyds", "Hello,World");
    // 休眠一会儿
	Thread.sleep(10000);
}

③运行测试:

a. 我们成功发送了消息,但是控制台依然返回了 ack=false
在这里插入图片描述
原因:当发送方法结束,RabbitMQ 相关的资源也就关闭了。虽然我们的消息发送出去,但异步的 ConfirmCallback 却由于资源关闭无法返回确认信息。
在这里插入图片描述

b. 我们可以让主线程休眠一会儿,等 callback 返回确认信息后再关闭资源。
在这里插入图片描述

回到目录…

🔹return 回退模式

①开启回退模式

spring:
  rabbitmq:
    addresses: 1.15.76.95
    username: admin
    password: 123456
    virtual-host: /test
    # 消息可靠传递: 开启确认模式
    publisher-confirm-type: correlated
    # 消息可靠传递: 开启回退模式
    publisher-returns: true

②设置 Exchange 处理消息的模式:如果消息没有路由到 Queue

  1. 默认方式:丢弃消息 。
  2. 如果设置了 rabbitTemplate.setMandatory(true) 参数,则会将消息退回给 producer。并执行回调函数 returnedMessage。

③设置 ReturnCallBack 函数

@Test
void publisher2() throws InterruptedException {

    // 1.设置交换机处理失败消息的模式
    template.setMandatory(true);

    // 2.设置ReturnCallBack函数
    template.setReturnsCallback(new RabbitTemplate.ReturnsCallback() {
        @Override
        public void returnedMessage(ReturnedMessage returnedMessage) {
            System.out.println("消息对象: " + returnedMessage.getMessage());
            System.out.println("错误码: " + returnedMessage.getReplyCode());
            System.out.println("错误信息: " + returnedMessage.getReplyText());
            System.out.println("交换机: " + returnedMessage.getExchange());
            System.out.println("路由键: " + returnedMessage.getRoutingKey());
        }
    });

    // 3.发送消息
    template.convertAndSend("amq.direct", "my-yyds1", "Hello,World");
    Thread.sleep(10000);
}

④运行测试:可以看到路由错误。
在这里插入图片描述

回到目录…

🚩1.2 消息接收确认(消费者)

🔸none 自动确认

  acknowledge-mode=none,默认方式,默认消费者能正确处理所有请求。

🔸auto 异常确认

  acknowledge-mode=auto,根据异常情况决定处理方式。

  1. 如果消费者在消费的过程中没有抛出异常,则自动确认。
  2. 当消费者消费的过程中抛出 AmqpRejectAndDontRequeueException 异常的时候,则消息会被拒绝,且该消息不会重回队列。
  3. 当抛出 ImmediateAcknowledgeAmqpException 异常,消息会被确认。
  4. 如果抛出其他的异常,则消息会被拒绝,但是与前两个不同的是,该消息会重回队列,如果此时只有一个消费者监听该队列,那么该消息重回队列后又会推送给该消费者,会造成死循环的情况。

🔸manual 手动确认

  1. 在该模式下,消费者消费消息后需要根据消费情况给 Broker 返回一个回执,是确认 ack 使 Broker 删除该条已消费的消息,还是失败确认返回 nack,还是拒绝该消息。
  2. 开启手动确认后,消费者接收到消息后必须返回 ack,只有 RabbitMQ 接收到 ack 后,消息才会从队列中被删除。如果不调用 channel.basicAck() 会出现该消息被重复消费的情况。

①设置手动签收,acknowledge =manual

spring:
  rabbitmq:
    addresses: 1.15.76.95
    username: admin
    password: 123456
    virtual-host: /test
    listener:
      # 设置监听器类型,如不设置将会默认为SimpleRabbitListenerContainerFactory,那么下面的direct配置不生效
      type: direct
      direct:
        # 设置消费端手动确认
        acknowledge-mode: manual

②监听器类实现 ChannelAwareMessageListener 接口

  • 目的是为了实现它的 onMessage(Message message, Channel channel) 方法,因为我们需要 channel 连接器。
  • 也可以不实现接口,直接在定义接收方法时,引入 Channel 参数。

③如果消息成功处理,则调用 channel.basicAck() 签收。

④如果消息处理失败,则调用 channel.basicNack() 拒绝签收,broker 重新发送给 consumer。

@Component
public class AckListener {

    @RabbitListener(queues = "yyds")
    public void onMessage(Message message, Channel channel) throws Exception {

        // 先获取当前的消息标签
        long deliveryTag = message.getMessageProperties().getDeliveryTag();

        try {
            // 1.接收消息
            System.out.println(new String(message.getBody()));

            // 2.处理业务逻辑
            System.out.println("处理业务逻辑...");
			//int a = 3 / 0; //模拟出错的场景
			
            // 3.手动签收
            channel.basicAck(deliveryTag, false);
        }catch(Exception e) {
            // 4.拒绝签收
            System.out.println("消息接收失败,让消息重回队列");
            channel.basicNack(deliveryTag, false, true);
        }
    }
}
消息确认的方法参数一参数二参数三描述
channel.basicAck(deliverTag, true)消息标签是否接收多条消息———确认应答
channel.basicNack(deliverTag, false, true)消息标签是否接收多条消息是否让消息重回队列拒绝应答
channel.basicReject(deliverTag, false)消息标签是否让消息重回队列———拒绝消息
channel.basicRecover(true)是否恢复给其它消费者——————恢复消息

⑤启动测试:

如果接收者成功消费消息,则没有任何返回。
在这里插入图片描述

如果接收者消费消息失败,则采用 Nack 让消息重回队列,并且自己可以再次消费重回后的消息。
在这里插入图片描述

回到目录…

二、消费端限流 (prefetch)

在这里插入图片描述

和单纯 ACK 区别:ACK 会一次拉取所有消息,而 prefetch 可以设置每次消费的消息数量

  • 用于指定消费端处理消息的速率。
  • 用于保证系统稳定性,削峰填谷时的处理速率。

该功能必须设置 ACK 方式:手动确认 acknowledge-mode: manualprefetch=条数

spring:
  rabbitmq:
    listener:
      type: direct
      direct:
        acknowledge-mode: manual #手动确认
        prefetch: 3 #消费者每次消费的数量

②我们先批量插入20条消息

@Test
void publisher() {
    for (int i = 0; i < 20; i++) {
        template.convertAndSend("amq.direct", "my-yyds", "Hello,World: " + i);
    }
}

在这里插入图片描述

③设置监听器:消费者手动确认消息

@Component
public class QosListener {
    @RabbitListener(queues = "yyds")
    public void onMessage(Message message, Channel channel) throws IOException, InterruptedException {
        // 1.接收消息
        System.out.println("接收消息: " + new String(message.getBody()));

        // 2.处理业务
        System.out.println("处理业务中...");
        Thread.sleep(1000);

        // 3.签收
        channel.basicAck(message.getMessageProperties().getDeliveryTag(), true); //设置参数可以签收多条消息
    }
}

④启动测试:可以看到我们接收到的消息是3条/批
在这里插入图片描述

回到目录…

三、设置队列参数

🍃3.1 消息TTL过期

RabbitMQ 支持将超过一定时间没被消费的消息自动删除,这需要消息设定 TTL 值,如果消息的存活时间超过了 Time To Live 值,就会被自动删除。如果有死信队列,那么就会进入到死信队列中。

①直接给队列设定 TTL 值(毫秒为单位): 过期会删除队列中的所有消息。

@Bean("yydsQueue")
public Queue queue(){
    return QueueBuilder
            .nonDurable("yyds")
            .deadLetterExchange("dlx.direct")
            .deadLetterRoutingKey("dl-yyds")
            .ttl(10000)   //如果10秒没处理,就自动删除
            .build();
}

现在我们删除之前的 yyds 队列再重启测试:可以发现 yyds 队列已经具有 TTL 特性了。
在这里插入图片描述

②生产者给某个消息设定过期时间: 使用 expiration 参数。只有消息在双端时,才是计时。

@Test
void publisher() {

    MessagePostProcessor messagePostProcessor = new MessagePostProcessor() {
        @Override
        public Message postProcessMessage(Message message) throws AmqpException {
            message.getMessageProperties().setExpiration("10000"); //设置过期时间 ms
            return message;
        }
    };
    template.convertAndSend("amq.direct", "my-yyds", "Hello,TLL", messagePostProcessor);
}

现在删除之前的 yyds 队列再重启测试:发现队列并没有 TTL 特性。
在这里插入图片描述
只是我们发布的消息会过期,不影响其它消息。
在这里插入图片描述

这种情况就不会过期:当该消息在队列头部时(消费时),会单独判断这一消息是否过期。
在这里插入图片描述

回到目录…

🍃3.2 队列最大长度

我们来看一下当消息队列长度达到最大的情况,现在我们将消息队列的长度进行限制:

@Bean("yydsQueue")
public Queue queue(){
    return QueueBuilder
            .nonDurable("yyds")
            .maxLength(3)   //将最大长度设定为3
            .build();
}

现在我们重启一下:可以发现 yyds 队列已经具有 Limit 特性了。
在这里插入图片描述

我们向 yyds 队列中依次插入4条消息:person-1person-2person-3person-4
在这里插入图片描述

可以看到因为长度限制为3,所以最开始的消息直接被丢弃了。
在这里插入图片描述

回到目录…

四、死信队列

使用场景
 ● 如果消息队列中的数据迟迟没有消费者来处理,那么就会一直占用消息队列的空间。
 ● 比如我们模拟一下抢车票的场景,用户下单高铁票之后,会进行抢座,然后再进行付款,但是如果用户下单之后并没有及时的付款,这张票不可能一直让这个用户占用着,因为你不买别人还要买呢,所以会在一段时间后超时,让这张票可以继续被其他人购买。
 ● 这时,我们就可以使用死信队列,将那些用户超时未付款的或是用户主动取消的订单,进行进一步的处理。

消息成为死信的三种情况
 ● 消息被拒绝 (basic.reject / basic.nack),并且 requeue = false 不重回队列
 ● 消息TTL过期
 ● 队列达到最大长度

那么如何构建这样的模式呢?
 实际上本质就是一个死信交换机 + 绑定的死信队列。当正常队列中的消息被判定为死信时,会被发送到对应的死信交换机,然后再通过交换机发送到死信队列中,死信队列也有对应的消费者去处理消息。

在这里插入图片描述

🍒4.1 构建死信队列

①这里我们直接在配置类中创建一个新的死信交换机和死信队列,并进行绑定:

@Configuration
public class RabbitConfiguration {

    @Bean("directDlExchange")   //创建一个新的死信交换机
    public Exchange dlExchange() {
        return ExchangeBuilder.directExchange("dlx.direct").build();
    }

    @Bean("yydsDlQueue")   //创建一个新的死信队列
    public Queue dlQueue() {
        return QueueBuilder
                .nonDurable("dl-yyds") //队列名称
                .build();
    }

    @Bean("dlBinding")   //死信交换机和死信队列进绑定
    public Binding dlBinding(@Qualifier("directDlExchange") Exchange exchange,
                             @Qualifier("yydsDlQueue") Queue queue) {
        return BindingBuilder
                .bind(queue)
                .to(exchange)
                .with("dl-yyds") //自定义routingKey
                .noargs();
    }

	@Bean("directExchange")
	.......................

    @Bean("yydsQueue")   //在普通消息队列中指定死信交换机
    public Queue queue(){
        return QueueBuilder
                .nonDurable("yyds")
                .deadLetterExchange("dlx.direct")   //指定死信交换机
                .deadLetterRoutingKey("dl-yyds")   //指定死信RoutingKey
                .build();
    }
  
  	@Bean("binding")
  	................
}

②接着我们将监听器修改为死信队列监听:

@Component
public class ConsumeListener {
    @RabbitListener(queues = "dl-yyds", messageConverter = "jacksonConverter")
    public void receiver(User user){
        System.out.println("死信队列监听器: " + user);
    }
}

启动一下:我们可以看到多了一个死信交换机。
在这里插入图片描述

队列列表中也多了一个死信队列,并且 yyds 队列也支持死信队列发送功能了。
在这里插入图片描述

回到目录…

🍒4.2 模拟死信消息

场景一:消息被拒绝

现在我们先向 yyds 队列中发送一个消息:Hello,World
在这里插入图片描述
然后我们取消息的时候拒绝消息,并且不让消息重新排队:
在这里插入图片描述

可以看到拒绝后,如果不让消息重新排队,那么就会直接被丢进死信队列中:
在这里插入图片描述

场景二:消息TTL过期

如果消息的存活时间超过了 Time To Live 值,就会被自动删除。如果有死信队列,那么就会进入到死信队列中。

现在我们将 yyds 消息队列设定 TTL 值(毫秒为单位):

@Bean("yydsQueue")
public Queue queue(){
    return QueueBuilder
            .nonDurable("yyds")
            .deadLetterExchange("dlx.direct")
            .deadLetterRoutingKey("dl-yyds")
            .ttl(10000)   //如果10秒没处理,就自动删除
            .build();
}

启动测试:我们可以看到 死信队列 和 普通队列 (TTLDLX死信交换机、DLK死信routingKey):
在这里插入图片描述

我们向 yyds 队列中插入一个新的消息:Hello,World
在这里插入图片描述
可以看到消息10秒钟之后就不见了,而是被丢进了死信队列中。
在这里插入图片描述

场景三:队列达到最大长度

最后我们来看一下当消息队列长度达到最大的情况,现在我们将消息队列的长度进行限制:

@Bean("yydsQueue")
public Queue queue(){
    return QueueBuilder
            .nonDurable("yyds")
            .deadLetterExchange("dlx.direct")
            .deadLetterRoutingKey("dl-yyds")
            .maxLength(3)   //将最大长度设定为3
            .build();
}

启动测试:我们可以看到 死信队列 和 普通队列(LimDLXDLK):
在这里插入图片描述

我们向 yyds 队列中依次插入4条消息:Message-1Message-2Message-3Message-4
在这里插入图片描述

可以看到因为长度限制为3,所以最开始的消息直接被丢进了死信队列中了。
在这里插入图片描述

回到目录…

🍒4.3 实现延迟队列

延迟队列,即消息进入队列后不会立即被消费,只有到达指定时间后,才会被消费。

场景一:下单后,30分钟未支付,取消订单,回滚库存。
场景二:新用户注册成功7天后,发送短信问候。

我们可以使用 TTL+死信队列 组合实现延迟队列的效果。
在这里插入图片描述

这里就不演示 TTL+死信队列 的组合了,和上面一样,我们来实现一下消费者端吧!

@Component
public class DlxListener {

    @RabbitListener(queues = "dl-yyds")
    public void receiver(Message message, Channel channel) throws Exception {        
        try {
            System.out.println("死信队列监听器: " + new String(message.getBody()));
            System.out.println("======== 业务处理中 ==========");
            System.out.println("根据订单id查询状态:");
            System.out.println("   如果订单支付成功,向物流系统发送发货的请求。");
            System.out.println("   如果订单支付失败,取消订单,回滚库存。");
            System.out.println("======== 确认消息签收 ==========");
            channel.basicAck(message.getMessageProperties().getDeliveryTag(), false);
        }catch(Exception e) {
            System.out.println("出现异常,拒绝签收, 让消息重回队列");
            channel.basicNack(message.getMessageProperties().getDeliveryTag(), false, true);
        }
    }
}

启动测试
在这里插入图片描述
在这里插入图片描述

手动向 yyds 队列中发一条消息:订单:{id:1, 状态:xxx, 下单时间: yyyy-m-d HH:MM:SS}
在这里插入图片描述

消息过期后,成功被死信队列监听到。
在这里插入图片描述

回到目录…

五、消息追踪

🍀5.1 Firehose: amq.rabbitmq.trace

  • Firehose 的机制是将生产者投递给 RabbitMQ 的消息,RabbitMQ 投递给消费者的消息按照指定的格式发送到默认的 Exchange 上。
  • 这个默认的 Exchange 的名称为 amq.rabbitmq.trace,它是一个 topic 类型的内部交换机。
  • 发送到该交换机上的消息的 routingKey 为 publish.exchangename(生产者发布的消息) 和 deliver.queuename(消费者获取的消息)。

应用:amq.rabbitmq.trace 交换机实现消息追踪
在这里插入图片描述
可以看到它也是 topic 类型的,它是一个内部交换机,用于帮助我们记录和追踪生产者和消费者使用消息队列的交换机。

①首先,我们需要在控制台将虚拟主机 /test 的追踪功能开启:

rabbitmqctl trace_on -p /test

②创建一个 trace 消息队列用于接收记录:
在这里插入图片描述

③我们给 amq.rabbitmq.trace 交换机绑定上刚刚的队列: 因为该交换机是内部的,所以只能在 Web 管理页面中绑定
在这里插入图片描述
在这里插入图片描述
由于发送到此交换机上的 routingKey 为 publish.交换机名称deliver.队列名称,分别对应生产者投递到交换机的消息,和消费者从队列上获取的消息,因此这里使用 # 通配符进行绑定。

④现在我们来测试一下,往 yyds 队列中发送消息: 会发现 trace 队列中多了2条信息。
在这里插入图片描述

通过追踪,我们可以很明确地得知消息发送的交换机、routingKey、用户等信息,包括信息本身:
在这里插入图片描述
同样的,消费者在取出数据时也有记录:我们可以明确消费者的地址、端口、具体操作的队列以及取出的消息信息等。
在这里插入图片描述

回到目录…

🍀5.2 rabbitmq_tracing 插件

rabbitmq_tracing 和 Firehose 在实现上如出一辙,只不过 rabbitmq_tracing 的方式比 Firehose 多了一
GUI的包装,更容易使用和管理。

缺点:会降低 RabbitMQ 的整体性能,不适用于生产开发中,适用于测试和调试阶段。

①启用插件

# 查看插件列表
rabbitmq-plugins list

# 启动插件
rabbitmq-plugins enable rabbitmq_tracing

②创建 Trace 追踪队列
在这里插入图片描述
同时多了一条队列,点击查看发现:实际上它也是绑定了我们的 amq.rabbitmq.trace 交换机。
在这里插入图片描述

③测试:向 yyds 队列中发布并消费消息,然后查看 Trace log 日志 (消息的生产者和消费者)
在这里插入图片描述

回到目录…

六、应用问题

🌹6.1 消息可靠性保障

🔸需求:确保消息 100% 发送成功。

🔸消息补偿机制

在这里插入图片描述

🌹6.2 消息幂等性保障

🔹需求:在MQ中,对于相同的消息消费一次和消费多次,最终的结果一致。

🔹乐观锁解决方案

在这里插入图片描述

回到目录…


总结:
提示:这里对文章进行总结:
本文是对RabbitMQ高级特性的学习,我们首先学习了消息确认机制和消费端限流的方法,又通过设置队列的参数实现了死信队列和延时队列。后面又介绍了两种方式实现消息追踪,最后也介绍了消息可靠性和消息幂等性的解决方案。之后的学习内容将持续更新!!!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/415851.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

创略科技联合创始人兼总裁杨辰韵:AIGC、隐私计算赋能数字营销的本质是“以客户为中心”丨数据猿专访...

‍数据智能产业创新服务媒体——聚焦数智 改变商业MarTech概念现身已超十年&#xff0c;伴随着企业数字化转型的大背景&#xff0c;中国MarTech市场也迎来了高速发展。据《2022年中国MarTech市场洞察报告》数据显示&#xff0c;2017-2021年&#xff0c;中国 MarTech产业规模从…

H264码流中 SPS PPS SEI 详解

1 客户端抓包 在做客户端视频解码时&#xff0c;一般都会使用Wireshark抓包工具对接收的H264码流进行分析&#xff0c;如下所示&#xff1a; 在这里我们可以看到对解码视频起关键作用的SPS和PPS。 2、双击SPS内容如下&#xff1a; 那么从上面的sps中我们知道图像的宽&#x…

【从零开始学Skynet】实战篇《球球大作战》(二):结构设计

万丈高楼平地起&#xff0c;既然这是个“大项目”&#xff0c;就要有大项目的样子&#xff0c;就要有所规划&#xff0c;下面先把项目的目录结构搭起来。 1、目录结构 建议把Skynet框架放到一个文件夹里&#xff0c;把所有自己编写的内容都放到外层的文件夹里。建立如下表所示的…

MySQL运维10-MySQL数据的导入导出

文章目录0、概述1、mysqldump导出数据mysql导入数据1.1、使用mysqldump导出数据1.1.1、使用--tables导出指定表1.1.2、使用--tab选项将表定义文件和数据文件分开导出1.1.3、使用--fields-terminated-by选项定义数据分隔符1.1.4、使用--databases选项导出整个库或多个库1.1.5、使…

Python算法设计 - 蒙特卡洛法

版权声明&#xff1a;原创不易&#xff0c;本文禁止抄袭、转载&#xff0c;侵权必究&#xff01; 目录一、蒙特卡洛法二、求圆周率π三、Python算法实现四、作者Info一、蒙特卡洛法 蒙特卡洛方法又称统计模拟法&#xff0c;随机抽样技术&#xff0c;是一种随机模拟方法&#x…

pytorch进阶学习(五):神经网络迁移学习应用的保姆级详细介绍,如何将训练好的模型替换成自己所需模型

代码资源和数据集资源使用进阶学习&#xff08;四&#xff09;中的代码&#xff0c;大家可以配合食用哟~ pytorch进阶学习&#xff08;四&#xff09;&#xff1a;使用不同分类模型进行数据训练&#xff08;alexnet、resnet、vgg等&#xff09;_好喜欢吃红柚子的博客-CSDN博客…

数据结构之八大排序算法

文章目录一.常见的排序二.插入排序三.希尔排序四.选择排序五.堆排序六.冒泡排序七.快速排序八.归并排序九.计数排序十.排序总结一.常见的排序 排序&#xff1a;所谓排序&#xff0c;就是使一串记录&#xff0c;按照其中的某个或某些关键字的大小&#xff0c;递增或递减的排列起…

pytorch性能分析工具Profiler

1. Torch Profiler工具介绍 https://zhuanlan.zhihu.com/p/360479566 PyTorch Profiler 是一个开源工具,可以对大规模深度学习模型进行准确高效的性能分析。包括如下等功能: 分析model的GPU、CPU的使用率各种算子op的时间消耗trace网络在pipeline的CPU和GPU的使用情况Profil…

Educational Codeforces Round 146 (Rated for Div. 2) - B. Long Legs(思维 数学)

题目如下&#xff1a; 题目链接 题解 or 思路&#xff1a; 我们可以发现我们有两个可选的入手方向&#xff1a; 1.正推 2.反推 我们可以发现正推似乎看不出来什么东西&#xff0c;而反推可以发现一个性质&#xff01; 性质如下&#xff1a; 我们假设最终的腿长为 MMM 可以得到…

sql需要注意的地方 以及 云记模块逻辑

标题模糊搜素时 sql语句需要注意的地方 用concat拼; 用户行为 actionName 进入发布云记页面 actionNameview 添加或修改云记 actionNameaddOrUpdate 查询云记详情 actionNamedetail 删除云记 actionNamedelete 分页查询云记列表 …

第05章_数组

第05章_数组 讲师&#xff1a;尚硅谷-宋红康&#xff08;江湖人称&#xff1a;康师傅&#xff09; 官网&#xff1a;http://www.atguigu.com 本章专题与脉络 1. 数组的概述 1.1 为什么需要数组 需求分析1&#xff1a; 需要统计某公司50个员工的工资情况&#xff0c;例如计…

chatGPT中文版入口-chatGPT不可以用的地区

ChatGPT老出现不可用 如果您在使用ChatGPT时发现它经常不可用&#xff0c;可能是由于以下原因&#xff1a; OpenAI API的服务不稳定。由于技术问题、网络问题或维护&#xff08;如软件更新&#xff09;等原因导致OpenAI API服务不稳定&#xff0c;会导致ChatGPT无法使用。 接…

【JSON学习笔记】2.JSON vs XML及JSON的对象和数组

前言 本章介绍JSON vs XML及JSON的对象和数组。 JSON vs XML JSON 和 XML 都用于接收 web 服务端的数据。 JSON 和 XML在写法上有所不同&#xff0c;如下所示&#xff1a; JSON 实例 {"sites": [{ "name":"csdn教程" , "url":&q…

〖Python网络爬虫实战⑪〗- 正则表达式实战(二)

订阅&#xff1a;新手可以订阅我的其他专栏。免费阶段订阅量1000python项目实战 Python编程基础教程系列&#xff08;零基础小白搬砖逆袭) 说明&#xff1a;本专栏持续更新中&#xff0c;目前专栏免费订阅&#xff0c;在转为付费专栏前订阅本专栏的&#xff0c;可以免费订阅付费…

《LeetCode 热题 HOT 100》——寻找两个正序数组的中位数

本期给大家带来的是是《LeetCode 热题 HOT 100》第四题——寻找两个正序数组的中位数的题目讲解&#xff01;&#xff01;&#xff01;&#xff08;&#xff09; 本文目录 &#x1f4a5;题意分析 &#x1f4a5;解题思路&#xff1a; 1、直接法 &#xff08;❌&#xff09; …

2023年证券、基金、银行从业资格证考试计划

一、证券从业&#xff1a; 考试时间&#xff1a;统一测试拟于6月3日-4日举办1次&#xff0c;在全国41个城市举办。 报名网站&#xff1a;考试报名-中国证券业协会 (sac.net.cn) 目标&#xff1a;一般从业资格考试(入门资格考试) 考试人群&#xff1a;即将进入证券业从业的人…

Linux命令·ping

Linux系统的ping命令是常用的网络命令&#xff0c;它通常用来测试与目标主机的连通性&#xff0c;我们经常会说“ping一下某机器&#xff0c;看是不是开着”、不能打开网页时会说“你先ping网关地址192.168.1.1试试”。它通过发送ICMP ECHO_REQUEST数据包到网络主机&#xff08…

【ROS2指南-5】理解ROS2服务

目标&#xff1a;使用命令行工具了解 ROS 2 中的服务。 教程级别&#xff1a;初学者 时间&#xff1a; 10分钟 内容 背景 先决条件 任务 1 设置 2 ros2服务列表 3 ros2服务类型 4 ros2 服务查找 5 ros2界面展示 6 ros2 服务调用 概括 下一步 相关内容 背景 服务是 …

Ubuntu18.04安装linux-lab

Ubuntu18.04安装linux-lab 文章目录Ubuntu18.04安装linux-labdocker安装linux-lab安装意外事件流处理参考资料本文主要目的是搭建linux内核实验环境 因为工作需要所以学习linux内核&#xff0c;目前主要根据《linux内核完全注释》和《自己动手写操作系统》进行学习&#xff0c…

代码随想录【链表】---->反转链表、两两交换链表中的节点

文章目录206. 反转链表思路双指针实现递归写法24. 两两交换链表中的节点思路代码实现206. 反转链表 题目LeetCode206. 反转链表 思路 翻转链表实际上只需要将每一个节点的指针域指向前一个节点即可&#xff0c;原来第一个节点的指针域指向NULL指针 原头节点是1&#xff0c…