Redis實(shí)現(xiàn)訂單過(guò)期刪除的方法步驟
前言
設(shè)計(jì)訂單過(guò)期,不能單純靠Redis,需要兜底策略
代碼實(shí)現(xiàn):
import com.coolplay.trade.dto.req.CancelOrderReq; import lombok.extern.slf4j.Slf4j; import org.apache.commons.collections.CollectionUtils; import org.springframework.data.redis.core.ZSetOperations; import org.springframework.scheduling.annotation.Scheduled; import org.springframework.stereotype.Service; import javax.annotation.Resource; import java.util.Set; import java.util.concurrent.TimeUnit; @Service @Slf4j public class OrderRedisDelayQueueOperator extends AbstractOrderScheduleDelayQueue { ? ? @Resource(name = "redisTemplate") ? ? private ZSetOperations<String, String> orderRedis; ? ? /** ? ? ?* 預(yù)售、現(xiàn)貨生成訂單15分鐘后未支付,需要取消訂單 ? ? ?*/ ? ? private static final String DELAY_QUEUE_NAME = "order"; ? ? /** ? ? ?* 每1秒執(zhí)行一次 ? ? ?*/ ? ? @Override ? ? @Scheduled(cron = "0/1 * * * * ? ") ? ? public void orderEventProcess() { ? ? ? ? if (!redisLock.tryLock(this.getClass().getSimpleName(), TimeUnit.MILLISECONDS, 10, 100)) { ? ? ? ? ? ? return; ? ? ? ? } ? ? ? ? Set<String> dq = orderRedis.range(DELAY_QUEUE_NAME, 0L, Long.MAX_VALUE); ? ? ? ? if (CollectionUtils.isEmpty(dq)) { ? ? ? ? ? ? return; ? ? ? ? } ? ? ? ? for (String orderNo : dq) { ? ? ? ? ? ? Double xs = orderRedis.score(DELAY_QUEUE_NAME, orderNo); ? ? ? ? ? ? Double now = System.currentTimeMillis() * 1.0; ? ? ? ? ? ? if (xs <= now) { ? ? ? ? ? ? ? ? log.info("{} timed out", orderNo); ? ? ? ? ? ? ? ? super.threadPoolTaskExecutor.execute(() -> { ? ? ? ? ? ? ? ? ? ? CancelOrderReq req = new CancelOrderReq(); ? ? ? ? ? ? ? ? ? ? req.setOrderNo(orderNo); ? ? ? ? ? ? ? ? ? ? req.setCancelType(OrderActionEnum.TIME_OUT_CANCEL); ? ? ? ? ? ? ? ? ? ? orderService.cancelOrder(req); ? ? ? ? ? ? ? ? }); ? ? ? ? ? ? } else { ? ? ? ? ? ? ? ? //log.info("{} no time out", orderNo); ? ? ? ? ? ? ? ? //如果最小的都沒(méi)有過(guò)期,剩余的則不用處理了 ? ? ? ? ? ? ? ? break; ? ? ? ? ? ? } ? ? ? ? } ? ? } ? ? public void addToRedis(String orderNo, long delayTime) { ? ? ? ? orderRedis.add(DELAY_QUEUE_NAME, orderNo, delayTime * 1.0); ? ? } ? ? public void removeFromRedis(String orderNo) { ? ? ? ? orderRedis.remove(DELAY_QUEUE_NAME, orderNo); ? ? } }
兜底策略
/** ? ? ?* 取消訂單--10分鐘--20分鐘執(zhí)行一次 ? ? ?*/ ? ? @XxlJob("cancelOrder20Minutes") ? ? public void cancelOrderTenMinutes() { ? ? ? ? log.info("*****[開(kāi)始:下單十分鐘以后系統(tǒng)自動(dòng)取消訂單]*****"); ? ? ? ? Date start = DateUtil.dateRoll(new Date(), Calendar.MINUTE,-20); ? ? ? ? Date end = new Date(); ? ? ? ?List<ClOrder> clorderList =clOrderMapper.selectListAllOrdrWaiting(start,end); ? ? ? ?if(ObjectUtil.isNotEmpty(clorderList)){ ? ? ? ? ? ?for(int i=0;i<clorderList.size();i++){ ? ? ? ? ? ? ? ?ClOrder clOrder = clorderList.get(i); ? ? ? ? ? ? ? ?if(ObjectUtil.isNotEmpty(clOrder)){ ? ? ? ? ? ? ? ? ? ?Date orderTime = clOrder.getOrderTime(); ? ? ? ? ? ? ? ? ? ?long between = cn.hutool.core.date.DateUtil.between(orderTime, new Date(), DateUnit.MINUTE); ? ? ? ? ? ? ? ? ? ?if(between>10){ ? ? ? ? ? ? ? ? ? ? ? ?ClOrder clOrderTemp = new ClOrder(); ? ? ? ? ? ? ? ? ? ? ? ?clOrderTemp.setOrderState("3"); ? ? ? ? ? ? ? ? ? ? ? ?clOrderTemp.setId(clOrder.getId()); ? ? ? ? ? ? ? ? ? ? ? ?clOrderTemp.setMemberId(clOrder.getMemberId()); ? ? ? ? ? ? ? ? ? ? ? ?String msg="您的訂單已經(jīng)取消,訂單金額已發(fā)放至您的賬戶(hù)請(qǐng)查收~"; ? ? ? ? ? ? ? ? ? ? ? ?try { ? ? ? ? ? ? ? ? ? ? ? ? ? ?boolean b = orderService.cancelOrder(clOrderTemp,msg); ? ? ? ? ? ? ? ? ? ? ? ? ? ?if(!b){ ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ?log.info("[訂單失效:定時(shí)任務(wù)兜底策略更新失敗]**訂單ID: {}",clOrderTemp.getId()); ? ? ? ? ? ? ? ? ? ? ? ? ? ?} ? ? ? ? ? ? ? ? ? ? ? ? ? ?log.info("[Redis訂單取消訂單失效,定時(shí)任務(wù)兜底策略生效]"); ? ? ? ? ? ? ? ? ? ? ? ?}catch (Exception e){ ? ? ? ? ? ? ? ? ? ? ? ? ? ?log.info("[訂單失效:定時(shí)任務(wù)兜底策略更新失敗]**訂單ID: {}",clOrderTemp.getId()); ? ? ? ? ? ? ? ? ? ? ? ? ? ?e.printStackTrace(); ? ? ? ? ? ? ? ? ? ? ? ?} ? ? ? ? ? ? ? ? ? ?} ? ? ? ? ? ? ? ?} ? ? ? ? ? ?} ? ? ? ?} ? ? ? ? log.info("*****[結(jié)束:下單十分鐘以后系統(tǒng)自動(dòng)取消訂單]*****"); ? ? }
到此這篇關(guān)于Redis實(shí)現(xiàn)訂單過(guò)期刪除的方法步驟的文章就介紹到這了,更多相關(guān)Redis 訂單過(guò)期刪除內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Redis中哈希結(jié)構(gòu)(Dict)的實(shí)現(xiàn)
本文主要介紹了Redis中哈希結(jié)構(gòu)(Dict)的實(shí)現(xiàn),文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2023-06-06Redis分布式鎖如何自動(dòng)續(xù)期的實(shí)現(xiàn)
本文主要介紹了Redis分布式鎖如何自動(dòng)續(xù)期的實(shí)現(xiàn),文中通過(guò)示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2021-12-12一文快速搞懂Redis的幾種數(shù)據(jù)類(lèi)型方式
這篇文章主要介紹了一文快速搞懂Redis的幾種數(shù)據(jù)類(lèi)型方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2022-10-10redis5.0以上基于密碼認(rèn)證的集群cluster方式
這篇文章主要介紹了redis5.0以上基于密碼認(rèn)證的集群cluster方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2023-11-11使用攔截器+Redis實(shí)現(xiàn)接口冪思路詳解
這篇文章主要介紹了使用攔截器+Redis實(shí)現(xiàn)接口冪等,接口冪等有很多種實(shí)現(xiàn)方式,攔截器/AOP+Redis,攔截器/AOP+本地緩存等等,本文講解一下通過(guò)攔截器+Redis實(shí)現(xiàn)冪等的方式,需要的朋友可以參考下2023-08-08詳解redis數(shù)據(jù)結(jié)構(gòu)之sds
sds是Simple Dynamic String的縮寫(xiě),譯為簡(jiǎn)單動(dòng)態(tài)字符串,redis使用該結(jié)構(gòu)保存字符串,不同于c中的字符串,redis使用該結(jié)構(gòu)來(lái)更方便的進(jìn)行字符串的處理,需要的朋友可以參考下2017-05-05Linux、Windows下Redis的安裝即Redis的基本使用詳解
Redis是一個(gè)基于內(nèi)存的key-value結(jié)構(gòu)數(shù)據(jù)庫(kù),Redis 是互聯(lián)網(wǎng)技術(shù)領(lǐng)域使用最為廣泛的存儲(chǔ)中間件,這篇文章主要介紹了Linux、Windows下Redis的安裝即Redis的基本使用詳解,需要的朋友可以參考下2022-09-09