使用場景
1、下單成功,30分鐘未支付。付款逾時,自動取消訂單
2、訂單簽收,簽收後7天未進(jìn)行評估。訂單逾時未評估,系統(tǒng)預(yù)設(shè)好評
3、下單成功,商家5分鐘未接單,訂單取消
4、配送逾時 xff0c;推送簡訊提醒
……
對於延時比較長的場景、即時性不高的場景,我們可以採用任務(wù)調(diào)度的方式定時輪詢處理。如:xxl-job
今天我們採用比較簡單、輕量級的方式,使用 Redis 的延遲佇列來處理。當(dāng)然有更好的解決方案,可根據(jù)公司的技術(shù)選項和業(yè)務(wù)體系選擇最優(yōu)方案。如:使用訊息中間件Kafka、RabbitMQ?的延遲隊列
先不討論其實現(xiàn)原理,直接實戰(zhàn)上程式碼先實現(xiàn)基於Redis 的延遲隊列
1、引入Redisson 依賴
<dependency> <groupId>org.redisson</groupId> <artifactId>redisson-spring-boot-starter</artifactId> <version>3.10.5</version> </dependency>
2、Nacos 配置?Redis 連線
spring: redis: host: 127.0.0.1 port: 6379 password: 123456 database: 12 timeout: 3000
3、建立RedissonConfig 設(shè)定
/** * Created by LPB on 2020/04/20. */ @Configuration public class RedissonConfig { @Value("${spring.redis.host}") private String host; @Value("${spring.redis.port}") private int port; @Value("${spring.redis.database}") private int database; @Value("${spring.redis.password}") private String password; @Bean public RedissonClient redissonClient() { Config config = new Config(); config.useSingleServer() .setAddress("redis://" + host + ":" + port) .setDatabase(database) .setPassword(password); return Redisson.create(config); } }
4、封裝Redis 延遲佇列工具類別
/** * redis延遲隊列工具 * Created by LPB on 2021/04/20. */ @Slf4j @Component public class RedisDelayQueueUtil { @Autowired private RedissonClient redissonClient; /** * 添加延遲隊列 * @param value 隊列值 * @param delay 延遲時間 * @param timeUnit 時間單位 * @param queueCode 隊列鍵 * @param <T> */ public <T> void addDelayQueue(T value, long delay, TimeUnit timeUnit, String queueCode){ try { RBlockingDeque<Object> blockingDeque = redissonClient.getBlockingDeque(queueCode); RDelayedQueue<Object> delayedQueue = redissonClient.getDelayedQueue(blockingDeque); delayedQueue.offer(value, delay, timeUnit); log.info("(添加延時隊列成功) 隊列鍵:{},隊列值:{},延遲時間:{}", queueCode, value, timeUnit.toSeconds(delay) + "秒"); } catch (Exception e) { log.error("(添加延時隊列失敗) {}", e.getMessage()); throw new RuntimeException("(添加延時隊列失敗)"); } } /** * 獲取延遲隊列 * @param queueCode * @param <T> * @return * @throws InterruptedException */ public <T> T getDelayQueue(String queueCode) throws InterruptedException { RBlockingDeque<Map> blockingDeque = redissonClient.getBlockingDeque(queueCode); T value = (T) blockingDeque.take(); return value; } }ee
5、建立延遲佇列業(yè)務(wù)枚舉
/** * 延遲隊列業(yè)務(wù)枚舉 * Created by LPB on 2021/04/20. */ @Getter @NoArgsConstructor @AllArgsConstructor public enum RedisDelayQueueEnum { ORDER_PAYMENT_TIMEOUT("ORDER_PAYMENT_TIMEOUT","訂單支付超時,自動取消訂單", "orderPaymentTimeout"), ORDER_TIMEOUT_NOT_EVALUATED("ORDER_TIMEOUT_NOT_EVALUATED", "訂單超時未評價,系統(tǒng)默認(rèn)好評", "orderTimeoutNotEvaluated"); /** * 延遲隊列 Redis Key */ private String code; /** * 中文描述 */ private String name; /** * 延遲隊列具體業(yè)務(wù)實現(xiàn)的 Bean * 可通過 Spring 的上下文獲取 */ private String beanId; }
6、定義延遲佇列執(zhí)行器
/** * 延遲隊列執(zhí)行器 * Created by LPB on 2021/04/20. */ public interface RedisDelayQueueHandle<T> { void execute(T t); }
7、建立在列舉中定義的Bean,並實作延??遲佇列執(zhí)行器
OrderPaymentTimeout:訂單支付逾時延遲佇列處理類別
/** * 訂單支付超時處理類 * Created by LPB on 2021/04/20. */ @Component @Slf4j public class OrderPaymentTimeout implements RedisDelayQueueHandle<Map> { @Override public void execute(Map map) { log.info("(收到訂單支付超時延遲消息) {}", map); // TODO 訂單支付超時,自動取消訂單處理業(yè)務(wù)... } }
DelayQueueProcessorForUnevaluatedOrders: 處理未評價訂單的延遲佇列處理類別,用於訂單逾時未評價的情況
/** * 訂單超時未評價處理類 * Created by LPB on 2021/04/20. */ @Component @Slf4j public class OrderTimeoutNotEvaluated implements RedisDelayQueueHandle<Map> { @Override public void execute(Map map) { log.info("(收到訂單超時未評價延遲消息) {}", map); // TODO 訂單超時未評價,系統(tǒng)默認(rèn)好評處理業(yè)務(wù)... } }
8、建立延遲隊列消費性執(zhí)行緒,專案啟動完成後開啟
/** * 啟動延遲隊列 * Created by LPB on 2021/04/20. */ @Slf4j @Component public class RedisDelayQueueRunner implements CommandLineRunner { @Autowired private RedisDelayQueueUtil redisDelayQueueUtil; @Override public void run(String... args) { new Thread(() -> { while (true){ try { RedisDelayQueueEnum[] queueEnums = RedisDelayQueueEnum.values(); for (RedisDelayQueueEnum queueEnum : queueEnums) { Object value = redisDelayQueueUtil.getDelayQueue(queueEnum.getCode()); if (value != null) { RedisDelayQueueHandle redisDelayQueueHandle = SpringUtil.getBean(queueEnum.getBeanId()); redisDelayQueueHandle.execute(value); } } } catch (InterruptedException e) { log.error("(Redis延遲隊列異常中斷) {}", e.getMessage()); } } }).start(); log.info("(Redis延遲隊列啟動成功)"); } }
以上步驟,Redis 延遲佇列核心程式碼已經(jīng)完成,下面我們寫一個測試介面,用PostMan 模擬測試一下
9、建立一個測試介面,模擬新增延遲佇列
/** * 延遲隊列測試 * Created by LPB on 2020/04/20. */ @RestController public class RedisDelayQueueController { @Autowired private RedisDelayQueueUtil redisDelayQueueUtil; @PostMapping("/addQueue") public void addQueue() { Map<String, String> map1 = new HashMap<>(); map1.put("orderId", "100"); map1.put("remark", "訂單支付超時,自動取消訂單"); Map<String, String> map2 = new HashMap<>(); map2.put("orderId", "200"); map2.put("remark", "訂單超時未評價,系統(tǒng)默認(rèn)好評"); // 添加訂單支付超時,自動取消訂單延遲隊列。為了測試效果,延遲10秒鐘 redisDelayQueueUtil.addDelayQueue(map1, 10, TimeUnit.SECONDS, RedisDelayQueueEnum.ORDER_PAYMENT_TIMEOUT.getCode()); // 訂單超時未評價,系統(tǒng)默認(rèn)好評。為了測試效果,延遲20秒鐘 redisDelayQueueUtil.addDelayQueue(map2, 20, TimeUnit.SECONDS, RedisDelayQueueEnum.ORDER_TIMEOUT_NOT_EVALUATED.getCode()); } }
10、啟動SpringBoot 專案,用PostMan 呼叫介面新增延遲佇列
透過Redis 用戶端可看到兩個延遲佇列已新增成功
檢視IDEA 控制臺日誌可看到延遲佇列已消費成功
以上是SpringBoot怎麼整合Redisson實現(xiàn)延遲隊列的詳細(xì)內(nèi)容。更多資訊請關(guān)注PHP中文網(wǎng)其他相關(guān)文章!

熱AI工具

Undress AI Tool
免費脫衣圖片

Undresser.AI Undress
人工智慧驅(qū)動的應(yīng)用程序,用於創(chuàng)建逼真的裸體照片

AI Clothes Remover
用於從照片中去除衣服的線上人工智慧工具。

Clothoff.io
AI脫衣器

Video Face Swap
使用我們完全免費的人工智慧換臉工具,輕鬆在任何影片中換臉!

熱門文章

熱工具

記事本++7.3.1
好用且免費的程式碼編輯器

SublimeText3漢化版
中文版,非常好用

禪工作室 13.0.1
強(qiáng)大的PHP整合開發(fā)環(huán)境

Dreamweaver CS6
視覺化網(wǎng)頁開發(fā)工具

SublimeText3 Mac版
神級程式碼編輯軟體(SublimeText3)

Jasypt介紹Jasypt是一個java庫,它允許開發(fā)員以最少的努力為他/她的專案添加基本的加密功能,並且不需要對加密工作原理有深入的了解用於單向和雙向加密的高安全性、基於標(biāo)準(zhǔn)的加密技術(shù)。加密密碼,文本,數(shù)字,二進(jìn)位檔案...適合整合到基於Spring的應(yīng)用程式中,開放API,用於任何JCE提供者...添加如下依賴:com.github.ulisesbocchiojasypt-spring-boot-starter2. 1.1Jasypt好處保護(hù)我們的系統(tǒng)安全,即使程式碼洩露,也可以保證資料來源的

一、Redis實現(xiàn)分散式鎖原理為什麼需要分散式鎖在聊分散式鎖之前,有必要先解釋一下,為什麼需要分散式鎖。與分散式鎖相對就的是單機(jī)鎖,我們在寫多執(zhí)行緒程式時,避免同時操作一個共享變數(shù)產(chǎn)生資料問題,通常會使用一把鎖來互斥以保證共享變數(shù)的正確性,其使用範(fàn)圍是在同一個進(jìn)程中。如果換做是多個進(jìn)程,需要同時操作一個共享資源,如何互斥?現(xiàn)在的業(yè)務(wù)應(yīng)用通常是微服務(wù)架構(gòu),這也意味著一個應(yīng)用會部署多個進(jìn)程,多個進(jìn)程如果需要修改MySQL中的同一行記錄,為了避免操作亂序?qū)е麦v數(shù)據(jù),此時就需要引入分佈式鎖了。想要實現(xiàn)分

1.自訂RedisTemplate1.1、RedisAPI預(yù)設(shè)序列化機(jī)制基於API的Redis快取實作是使用RedisTemplate範(fàn)本進(jìn)行資料快取操作的,這裡開啟RedisTemplate類,查看該類別的源碼資訊publicclassRedisTemplateextendsRedisAccessorimplementsRedisOperations,BeanClassLoaderAware{//聲明了value的各種序列化方式,初始值為空@NullableprivateRedisSe

springboot讀取文件,打成jar包後訪問不到最新開發(fā)出現(xiàn)一種情況,springboot打成jar包後讀取不到文件,原因是打包之後,文件的虛擬路徑是無效的,只能通過流去讀取。文件在resources下publicvoidtest(){Listnames=newArrayList();InputStreamReaderread=null;try{ClassPathResourceresource=newClassPathResource("name.txt");Input

使用場景1、下單成功,30分鐘未支付。支付超時,自動取消訂單2、訂單簽收,簽收後7天未進(jìn)行評估。訂單超時未評價,系統(tǒng)預(yù)設(shè)好評3、下單成功,商家5分鐘未接單,訂單取消4、配送超時,推播簡訊提醒…對於延時比較長的場景、即時性不高的場景,我們可以採用任務(wù)調(diào)度的方式定時輪詢處理。如:xxl-job今天我們採

在Springboot+Mybatis-plus不使用SQL語句進(jìn)行多表添加操作我所遇到的問題準(zhǔn)備工作在測試環(huán)境下模擬思維分解一下:創(chuàng)建出一個帶有參數(shù)的BrandDTO對像模擬對後臺傳遞參數(shù)我所遇到的問題我們都知道,在我們使用Mybatis-plus中進(jìn)行多表操作是極其困難的,如果你不使用Mybatis-plus-join這一類的工具,你只能去配置對應(yīng)的Mapper.xml文件,配置又臭又長的ResultMap,然後再寫對應(yīng)的sql語句,這種方法雖然看上去很麻煩,但具有很高的靈活性,可以讓我們

SpringBoot和SpringMVC都是Java開發(fā)中常用的框架,但它們之間有一些明顯的差異。本文將探究這兩個框架的特點和用途,並對它們的差異進(jìn)行比較。首先,我們來了解一下SpringBoot。 SpringBoot是由Pivotal團(tuán)隊開發(fā)的,它旨在簡化基於Spring框架的應(yīng)用程式的建立和部署。它提供了一種快速、輕量級的方式來建立獨立的、可執(zhí)行

一、@Import引入普通類別@Import引入普通的類別可以幫助我們把普通的類別定義為Bean。 @Import可以加入在@SpringBootApplication(啟動類別)、@Configuration(配置類別)、@Component(組件類別)對應(yīng)的類別上。注意:@RestController、@Service、@Repository都屬於@Component@SpringBootApplication@Import(ImportBean.class)//透過@Import註解把ImportBean
