springboot實現多例項crontab搶佔定時任務(例項程式碼)
阿新 • • 發佈:2020-01-10
github: https://github.com/jiasion/eslog
wechat:minghui-666
利用redisson實現多例項搶佔定時任務
pom.xml
<dependency> <groupId>org.redisson</groupId> <artifactId>redisson</artifactId> <version>3.12.0</version> </dependency>
Kernel.java - 重寫多執行緒排程
package com.brand.log.scheduler; import org.springframework.context.annotation.Configuration; import org.springframework.scheduling.annotation.SchedulingConfigurer; import org.springframework.scheduling.config.ScheduledTaskRegistrar; import java.util.concurrent.Executors; @Configuration public class Kernel implements SchedulingConfigurer { @Override public void configureTasks(ScheduledTaskRegistrar taskRegistrar) { //設定一個長度10的定時任務執行緒池 taskRegistrar.setScheduler(Executors.newScheduledThreadPool(4)); } }
RedissonManager.java - 分散式鎖的實現
package com.brand.log.util; import lombok.extern.slf4j.Slf4j; import org.redisson.Redisson; import org.redisson.config.Config; import org.springframework.beans.factory.annotation.Value; import org.springframework.stereotype.Component; import javax.annotation.PostConstruct; @Component @Slf4j public class RedissonManager { @Value("${spring.redis.host}") private String host; @Value("${spring.redis.port}") private int port; private Redisson redisson = null; private Config config = new Config(); @PostConstruct private void init() { try { config.useSingleServer().setAddress("redis://" + host + ":" + port); log.info("redisson address {} {}",host,port); redisson = (Redisson) Redisson.create(config); log.info("Redisson 初始化完成"); } catch (Exception e) { log.error("init Redisson error ",e); } } public Redisson getRedisson() { return redisson; } }
CronSynData.java
package com.brand.log.scheduler; import com.brand.log.util.DateFormatV1; import com.brand.log.util.RedisUtil; import com.brand.log.util.RedissonManager; import lombok.extern.slf4j.Slf4j; import org.redisson.Redisson; import org.redisson.api.RLock; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.scheduling.annotation.Scheduled; import org.springframework.stereotype.Component; import java.util.concurrent.TimeUnit; @Component @Slf4j public class CronSynData { @Autowired RedissonManager redissonManager; @Autowired RedisUtil redisUtil; @Autowired DateFormatV1 dateFormatV1; private String lokFlag = ".handleKernel"; private Redisson redisson = null; /* * java定時指令碼掛靠例項 * 多例項會有重複呼叫問題 + 使用Redisson實現分散式鎖 * 業務邏輯必須加鎖 + 且需要保證 tryLock 等待時間小於cron的最小間隔執行時間 * */ @Scheduled(cron = "*/10 * * * * *") public void handleKernel() { redisson = redissonManager.getRedisson(); if (redisson != null) { RLock lock = redisson.getLock(this.getClass().getName() + lokFlag); Boolean stat = false; try { // 嘗試加鎖,立即返回,最多等待5s自動解鎖 stat = lock.tryLock(0,5,TimeUnit.SECONDS); if (stat) { log.info("{} 取鎖成功!{}",this.getClass().getName(),Thread.currentThread().getName()); redisUtil.checkCount("log:limit_",dateFormatV1.getDate("HH","GMT+8"),60*10,1000); } else { log.info("{}沒有獲取到鎖:{}",Thread.currentThread().getName()); } } catch (InterruptedException e) { log.error("Redisson 獲取分散式鎖異常",e); if (!stat){ return; } lock.unlock(); } } } }
kibana - 6個例項
總結
以上所述是小編給大家介紹的springboot實現多例項crontab搶佔定時任務,希望對大家有所幫助,如果大家有任何疑問請給我留言,小編會及時回覆大家的。在此也非常感謝大家對我們網站的支援!
如果你覺得本文對你有幫助,歡迎轉載,煩請註明出處,謝謝!