>  기사  >  데이터 베이스  >  Redis에서 지연 대기열을 구현하는 방법은 무엇입니까?

Redis에서 지연 대기열을 구현하는 방법은 무엇입니까?

WBOY
WBOY앞으로
2023-05-30 11:29:252328검색

1. 서문

1.1 지연 대기열이란 무엇입니까

지연 대기열과 일반 대기열의 가장 큰 차이점은 지연 속성에 반영됩니다. 대기열에 추가되는 순서입니다. 지연 대기열의 요소에는 대기열에 추가될 때 지연 시간이 할당됩니다. 이는 지정된 시간 이후에 처리되기를 희망함을 나타냅니다. 지연 대기열의 구조는 기존 대기열보다 시간 가중 순서가 지정된 힙 구조와 유사합니다.

1.2. 애플리케이션 시나리오

일부 비즈니스 시나리오에서는 특정 시간 노드 또는 일정 기간 후에 실행해야 하는 기능을 자주 접하게 됩니다. 예를 들면 다음과 같습니다.

새 주문을 생성하세요. 지정된 시간 내에 결제가 완료되지 않으면 테이크아웃이 자동으로 취소되거나 예상 도착 시간 10분 전에 승객이나 운전기사에게 알려야 합니다. 빠른 배송 후에도 사용자가 지정된 시간 내에 영수증을 확인하지 않으면 상품이 자동으로 영수증을 확인합니다. 일일 주간 보고서는 마감일 30분 전에 최대한 빨리 제출하도록 알려줍니다

1.3. 지연 대기열을 사용하는 이유

일부 데이터 볼륨의 경우 규모가 작고 데이터의 적시성이 많이 필요하지 않은 프로젝트의 경우 가장 간단하고 효과적인 방법은 비즈니스 실현을 위해 데이터베이스를 스캔하는 예약된 작업을 작성하는 것입니다. 데이터의 양이 수백만 또는 수천만에 도달하면 데이터베이스를 정기적으로 검사하면 타격을 받기 쉽습니다. 데이터가 이 수준에 도달하면 정기적으로 테이블을 스캔하는 것이 매우 비효율적이라는 것을 모두가 알고 있다고 생각합니다. 타이밍 간격이 상대적으로 작은 상황에서도 스캔이 완료되기 전에 다음 스캔이 시작됩니다. 이때 지연큐를 사용하는 것이 매우 효과적일 수 있습니다.

지연 대기열을 구현하는 여러 가지 방법

  • Quartz 예약 작업

  • DelayQueue 지연 대기열

  • Redis 정렬 세트 Redis

  • 만료된 키 수신 콜백

  • abbitMQ 배달 못한 편지 대기열

  • RabbitMQ는 플러그인

  • 휠 타임 휠 알고리즘

2을 기반으로 지연 대기열을 구현합니다. Redis 정렬 집합

Redis에서 zet은 정렬된 집합이며 정렬된 특성을 사용하여 zset에 작업을 추가할 수 있습니다. 작업 만료 시간을 점수로 사용, zset의 기본 순서 기능 사용, 점수 값이 가장 작은 요소(즉, 가장 최근에 만료된 작업)를 획득하고 시스템 시간과 작업 만료 시간을 판단 만료 시간에 도달하면 비즈니스를 실행하고 만료된 작업을 삭제한 후 다음 요소를 계속 판단합니다. 만료되지 않은 경우 일정 시간(예: 1초) 동안 휴면 상태로 유지됩니다. 비어 있으면 일정 시간 동안 잠자기 상태가 됩니다.

Redis에서 지연 대기열을 구현하는 방법은 무엇입니까?

zadd 명령을 통해 대기열 지연 대기열에 요소를 추가하고 요소의 만료 시간을 나타내도록 점수 값을 설정합니다. 3개의 order1, order2 및 order3을 지연 대기열에 추가합니다. 이 순서는 10초, 20초 후에 만료됩니다. , 각각 30초입니다.

zadd Delayqueue 3 order3

소비자는 대기열 지연 대기열을 폴링하고 요소를 정렬한 후 최소 시간을 현재 시간과 비교합니다. 현재 시간보다 작으면 키가 만료되어 제거되었음을 의미합니다.

/**
 * 消费消息
 */
public void pollOrderQueue() {
    while (true) {
        Set<Tuple> set = jedis.zrangeWithScores(DELAY_QUEUE, 0, 0);
        String value = ((Tuple) set.toArray()[0]).getElement();
        int score = (int) ((Tuple) set.toArray()[0]).getScore();
        Calendar cal = Calendar.getInstance();
        int nowSecond = (int) (cal.getTimeInMillis() / 1000);
        if (nowSecond >= score) {
            jedis.zrem(DELAY_QUEUE, value);
            System.out.println(sdf.format(new Date()) + " removed key:" + value);
        }
        if (jedis.zcard(DELAY_QUEUE) <= 0) {
            System.out.println(sdf.format(new Date()) + " zset empty ");
            return;
        }
        Thread.sleep(1000);
    }
}

실행 결과가 예상대로인 것을 확인합니다:

2020-05-07 13:24:09 추가 완료.

2020-05-07 13:24:19 제거된 키:order1
2020-05-07 13: 24:29 제거된 키:order2
2020-05-07 13:24:39 제거된 키:order3
2020-05-07 13:24:39 zset 비어 있음

3 Redis 만료된 키 모니터링 콜백

Redis 키 만료 콜백 이벤트는 대기열을 지연시키는 효과도 얻을 수 있습니다. 간단히 말해 키가 만료되면 콜백 이벤트가 트리거됩니다.

notify-keyspace-events Ex를 활성화하려면 redis.conf 파일을 편집해야 합니다. 통지-키스페이스-이벤트 Ex

Redis 수신 구성, Bean RedisMessageListenerContainer를 삽입합니다.

두 번째로 Redis 리스너를 구성합니다. 마지막으로 Redis 키 만료 수신 콜백 메서드를 작성합니다.

@Configuration
public class RedisListenerConfig {
@Bean
RedisMessageListenerContainer container(RedisConnectionFactory connectionFactory) {
        RedisMessageListenerContainer container = new RedisMessageListenerContainer();
        container.setConnectionFactory(connectionFactory);
         return container;
    }
}

Redis 만료 콜백 수신 메서드를 작성하려면 MQ 메시지 수신과 다소 유사한 KeyExpirationEventMessageListener를 상속해야 합니다.

@Component
public class RedisKeyExpirationListener extends KeyExpirationEventMessageListener {
public RedisKeyExpirationListener(RedisMessageListenerContainer listenerContainer) {
    super(listenerContainer);
}
@Override
public void onMessage(Message message, byte[] pattern) {
      String expiredKey = message.toString();
      System.out.println("监听到key:" + expiredKey + "已过期");
    }
}

이 코드는 매우 간단하게 작성되었습니다. 다음으로 redis-cli 클라이언트에 키를 추가하고 만료 시간을 3초로 설정합니다.

set xiaofu 123 ex 3

콘솔에서 이 만료된 키를 성공적으로 모니터링했습니다.

모니터링되는 만료된 키는 다음과 같습니다. xiaofu

4. Quartz 예약 작업

Quartz는 Redis와 RabbitMQ가 널리 사용되지 않았을 때 시간 경과에 따른 결제가 없는 주문 취소 기능이 예약 작업에 의해 구현되었습니다. 의.

Quartz 종속 항목 가져오기

<dependency>
   <groupId>org.springframework.boot</groupId>
   <artifactId>spring-boot-starter-quartz</artifactId>
</dependency>
>在启动类中使用@EnableScheduling注解开启定时任务功能。
```java
@SpringBootApplication
@EnableScheduling
public class DelayQueueApplication {
    public static void main(String[] args) {
        SpringApplication.run(DelayQueueApplication.class, args);
    }
}

예약된 작업 쓰기

@Slf4j
@Component
public class QuartzDemo {
    /**
     * 每隔五秒开启一次任务
     */
    @Scheduled(cron = "0/5 * * * * ? ")
    public void process(){
        log.info("--------------定时任务测试--------------");
    }
}

5. DelayQueue 지연 대기열

JDK는 Java.util.concurrent 패키지 아래의 DelayQueue에 있는 지연 대기열을 구현하기 위한 API 세트를 제공합니다.

DelayQueue是一个BlockingQueue(无界阻塞)队列,它本质就是封装了一个PriorityQueue(优先队列),PriorityQueue内部使用完全二叉堆(不知道的自行了解哈)来实现队列元素排序,我们在向DelayQueue队列中添加元素时,会给元素一个Delay(延迟时间)作为排序条件,队列中最小的元素会优先放在队首。队列中的元素只有到了Delay时间才允许从队列中取出。队列中可以放基本数据类型或自定义实体类,在存放基本数据类型时,优先队列中元素默认升序排列,自定义实体类就需要我们根据类属性值比较计算了。 先简单实现一下看看效果,添加三个order入队DelayQueue,分别设置订单在当前时间的5秒、10秒、15秒后取消。

要实现DelayQueue延时队列,队中元素要implements Delayed 接口,这哥接口里只有一个getDelay方法,用于设置延期时间。在Order类中,compareTo方法的作用是对队列中的元素进行排列。

public class Order implements Delayed {
/**
 * 延迟时间
 */
@JsonFormat(locale = "zh", timezone = "GMT+8", pattern = "yyyy-MM-dd HH:mm:ss")
private long time;
String name;
public Order(String name, long time, TimeUnit unit) {
    this.name = name;
    this.time = System.currentTimeMillis() + (time > 0 ? unit.toMillis(time) : 0);
}
@Override
public long getDelay(TimeUnit unit) {
    return time - System.currentTimeMillis();
}
@Override
public int compareTo(Delayed o) {
    Order Order = (Order) o;
    long diff = this.time - Order.time;
    if (diff <= 0) {
        return -1;
    } else {
        return 1;
    }
}
}

DelayQueue的put方法是线程安全的,因为put方法内部使用了ReentrantLock锁进行线程同步。DelayQueue还提供了两种出队的方法poll()和take() , poll()为非阻塞获取,没有到期的元素直接返回null;take()阻塞方式获取,没有到期的元素线程将会等待。

public class DelayQueueDemo {
public static void main(String[] args) throws InterruptedException {
    Order Order1 = new Order("Order1", 5, TimeUnit.SECONDS);
    Order Order2 = new Order("Order2", 10, TimeUnit.SECONDS);
    Order Order3 = new Order("Order3", 15, TimeUnit.SECONDS);
    DelayQueue<Order> delayQueue = new DelayQueue<>();
    delayQueue.put(Order1);
    delayQueue.put(Order2);
    delayQueue.put(Order3);
    System.out.println("订单延迟队列开始时间:" + LocalDateTime.now().format(DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss")));
    while (delayQueue.size() != 0) {
        /**
         * 取队列头部元素是否过期
         */
        Order task = delayQueue.poll();
        if (task != null) {
            System.out.format("订单:{%s}被取消, 取消时间:{%s}\n", task.name,  
            LocalDateTime.now().format(DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss")));
        }
        Thread.sleep(1000);
    }
}
}

上边只是简单的实现入队与出队的操作,实际开发中会有专门的线程,负责消息的入队与消费。

执行后看到结果如下,Order1、Order2、Order3 分别在 5秒、10秒、15秒后被执行,至此就用DelayQueue实现了延时队列。

订单延迟队列开始时间:2020-05-06 14:59:09

订单:{Order1}被取消, 取消时间:{2020-05-06 14:59:14}

订单:{Order2}被取消, 取消时间:{2020-05-06 14:59:19}

订单:{Order3}被取消, 取消时间:{2020-05-06 14:59:24}

6、RabbitMQ 延时队列

利用 RabbitMQ 做延时队列是比较常见的一种方式,而实际上RabbitMQ 自身并没有直接支持提供延迟队列功能,而是通过 RabbitMQ 消息队列的 TTL和 DXL这两个属性间接实现的。

先来认识一下 TTL和 DXL两个概念:

Time To Live(TTL) :

TTL 顾名思义:指的是消息的存活时间,RabbitMQ可以通过x-message-tt参数来设置指定Queue(队列)和 Message(消息)上消息的存活时间,它的值是一个非负整数,单位为微秒。

RabbitMQ 可以从两种维度设置消息过期时间,分别是队列和消息本身

设置队列过期时间,那么队列中所有消息都具有相同的过期时间。可以在队列中为每条消息单独设置过期时间,即使每个消息的TTL不同也可以实现。若队列和队列中消息的TTL同时被设置,则TTL的值以两者中较小的那个为准。如果队列中的消息存储时间超过了预设的TTL过期时间,那么它就会变成Dead Letter(死信)。

Dead Letter Exchanges(DLX)

DLX即死信交换机,绑定在死信交换机上的即死信队列。RabbitMQ的 Queue(队列)可以配置两个参数x-dead-letter-exchange 和 x-dead-letter-routing-key(可选),一旦队列内出现了Dead Letter(死信),则按照这两个参数可以将消息重新路由到另一个Exchange(交换机),让消息重新被消费。

x-dead-letter-exchange:队列中出现Dead Letter后将Dead Letter重新路由转发到指定 exchange(交换机)。

x-dead-letter-routing-key:指定routing-key发送,一般为要指定转发的队列。

队列出现Dead Letter的情况有:

消息或者队列的TTL过期

队列达到最大长度

消息被消费端拒绝(basic.reject or basic.nack)

下边结合一张图看看如何实现超30分钟未支付关单功能,我们将订单消息A0001发送到延迟队列order.delay.queue,并设置x-message-tt消息存活时间为30分钟,当到达30分钟后订单消息A0001成为了Dead Letter(死信),延迟队列检测到有死信,通过配置x-dead-letter-exchange,将死信重新转发到能正常消费的关单队列,直接监听关单队列处理关单逻辑即可。

Redis에서 지연 대기열을 구현하는 방법은 무엇입니까?

发送消息时指定消息延迟的时间

public void send(String delayTimes) {
    amqpTemplate.convertAndSend("order.pay.exchange", "order.pay.queue","大家好我是延迟数据", message -> {
      // 设置延迟毫秒值
      message.getMessageProperties().setExpiration(String.valueOf(delayTimes));
      return message;
    });
  }
}

设置延迟队列出现死信后的转发规则

/**
   * 延时队列
   */
  @Bean(name = "order.delay.queue")
  public Queue getMessageQueue() {
    return QueueBuilder
        .durable(RabbitConstant.DEAD_LETTER_QUEUE)
        // 配置到期后转发的交换
        .withArgument("x-dead-letter-exchange", "order.close.exchange")
        // 配置到期后转发的路由键
        .withArgument("x-dead-letter-routing-key", "order.close.queue")
        .build();
  }

7、时间轮

前面几种实现延迟队列的方法相对简单,比较易于理解。相比之下,时间轮算法稍微有点抽象。kafka、netty都有基于时间轮算法实现延时队列,下边主要实践Netty的延时队列讲一下时间轮是什么原理。

先来看一张时间轮的原理图,解读一下时间轮的几个基本概念

Redis에서 지연 대기열을 구현하는 방법은 무엇입니까?

wheel :时间轮,图中的圆盘可以看作是钟表的刻度。举个例子,如果一圈round的长度为24秒,共分成8个刻度,那么每个刻度代表3秒。那么时间精度就是 3秒。时间长度 / 刻度数值越大,精度越大。

当添加一个定时、延时任务A,假如会延迟25秒后才会执行,可时间轮一圈round 的长度才24秒,那么此时会根据时间轮长度和刻度得到一个圈数 round和对应的指针位置 index,也是就任务A会绕一圈指向0格子上,此时时间轮会记录该任务的round和 index信息。指针处于0格,当 round=0 且 index=0 时不会执行任务A,因为 round=0 不符合条件。

所以每一个格子代表的是一些时间,比如1秒和25秒 都会指向0格子上,而任务则放在每个格子对应的链表中,这点和HashMap的数据有些类似。

Netty构建延时队列主要用HashedWheelTimer,HashedWheelTimer底层数据结构依然是使用DelayedQueue,只是采用时间轮的算法来实现。

下面我们用Netty 简单实现延时队列,HashedWheelTimer构造函数比较多,解释一下各参数的含义。

ThreadFactory :表示用于生成工作线程,一般采用线程池;

tickDuration和unit:每格的时间间隔,默认100ms;

ticksPerWheel:一圈下来有几格,默认512,而如果传入数值的不是2的N次方,则会调整为大于等于该参数的一个2的N次方数值,有利于优化hash值的计算。

public HashedWheelTimer(ThreadFactory threadFactory, long tickDuration, TimeUnit unit, int ticksPerWheel) {
    this(threadFactory, tickDuration, unit, ticksPerWheel, true);
  }

TimerTask:一个定时任务的实现接口,其中run方法包装了定时任务的逻辑。

Timeout:一个定时任务提交到Timer之后返回的句柄,通过这个句柄外部可以取消这个定时任务,并对定时任务的状态进行一些基本的判断。 Timer:是HashedWheelTimer实现的父接口,仅定义了如何提交定时任务和如何停止整个定时机制。

public class NettyDelayQueue {
  public static void main(String[] args) {
    final Timer timer = new HashedWheelTimer(Executors.defaultThreadFactory(), 5, TimeUnit.SECONDS, 2);
    //定时任务
    TimerTask task1 = new TimerTask() {
      public void run(Timeout timeout) throws Exception {
        System.out.println("order1 5s 后执行 ");
        timer.newTimeout(this, 5, TimeUnit.SECONDS);//结束时候再次注册
      }
    };
    timer.newTimeout(task1, 5, TimeUnit.SECONDS);
    TimerTask task2 = new TimerTask() {
      public void run(Timeout timeout) throws Exception {
        System.out.println("order2 10s 后执行");
        timer.newTimeout(this, 10, TimeUnit.SECONDS);//结束时候再注册
      }
    };
    timer.newTimeout(task2, 10, TimeUnit.SECONDS);
    //延迟任务
    timer.newTimeout(new TimerTask() {
      public void run(Timeout timeout) throws Exception {
        System.out.println("order3 15s 后执行一次");
      }
    }, 15, TimeUnit.SECONDS);
  }
}

从执行的结果看,order3、order3延时任务只执行了一次,而order2、order1为定时任务,按照不同的周期重复执行。

order1 5s 后执行
order2 10s 后执行
order3 15s 后执行一次
order1 5s 后执行
order2 10s 后执行

위 내용은 Redis에서 지연 대기열을 구현하는 방법은 무엇입니까?의 상세 내용입니다. 자세한 내용은 PHP 중국어 웹사이트의 기타 관련 기사를 참조하세요!

성명:
이 기사는 yisu.com에서 복제됩니다. 침해가 있는 경우 admin@php.cn으로 문의하시기 바랍니다. 삭제