一種簡單的Java延遲處理器

最近做一個新項目時,有這么一個場景:使用TDDL數(shù)據(jù)庫分庫分表,且按SQL讀寫分離。在為系統(tǒng)做緩存層時,考慮到并發(fā)讀寫,可能會出現(xiàn)這么個問題:

緩存未設(shè)置或已過期時,寫數(shù)據(jù),主庫已完成但未拉起緩存,從庫也未完成同步,此時有查詢請求,將會訪問到從庫,把舊值查詢出來,如果此時寫數(shù)據(jù)線程已經(jīng)用最新數(shù)據(jù)拉起緩存,那讀數(shù)據(jù)線程將會用舊值將緩存覆蓋,導(dǎo)致緩存與數(shù)據(jù)庫不一致,在緩存失效或下次修改之前,緩存數(shù)據(jù)將一直是過期的。

這種情況雖然比較少見,但還是有一定概率出現(xiàn)的。為了屏蔽主從同步可能帶來的問題,一種比較簡單的解決方法是延遲更新緩存,給數(shù)據(jù)庫一定時間完成同步再進(jìn)行緩存設(shè)置,這樣就能保證最后緩存的是新數(shù)據(jù)。當(dāng)然這種方案只能解決數(shù)據(jù)庫同步延遲不大的普通情況。

如果是一些讀多寫少的數(shù)據(jù),也可以在修改完成改一遍緩存,再放到延遲任務(wù)改一遍。

為此實現(xiàn)了一個簡單的延遲任務(wù)處理器,主要思路是將任務(wù)放入DelayQueue,由一個專用線程從DelayQueue中取出到期的任務(wù),放入執(zhí)行線程池中執(zhí)行。整體比較簡單。

import com.google.common.util.concurrent.ThreadFactoryBuilder;
import java.lang.Thread.UncaughtExceptionHandler;
import java.util.List;
import java.util.concurrent.DelayQueue;
import java.util.concurrent.Delayed;
import java.util.concurrent.LinkedBlockingQueue;
import java.util.concurrent.ThreadPoolExecutor;
import java.util.concurrent.TimeUnit;
import org.apache.commons.collections.CollectionUtils;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.beans.factory.DisposableBean;
import org.springframework.beans.factory.InitializingBean;

/**
 * 延遲任務(wù)處理器
 * @author jerryli
 * @date 2018/3/27
 */
public class DelayedTaskExecutor implements InitializingBean, DisposableBean{

    private static final Logger logger = LoggerFactory.getLogger(DelayedTaskExecutor.class);

    private String name;

    private ThreadPoolExecutor executor;

    private volatile boolean running;

    private ThreadPoolExecutor putExecutor ;

    private DelayQueue<Task> delayQueue = new DelayQueue<>();

    public DelayedTaskExecutor(String name){
        this.name = name;
    }

    public void submit(Runnable runnable, long delayedTime){
        Task task = new Task(runnable,delayedTime);
        delayQueue.offer(task);
        logger.info("{}添加延遲任務(wù) {} 延遲{}ms",this.name,task,delayedTime);
    }

    @Override
    public void destroy() throws Exception {
        running = false;

        if(putExecutor != null){
            putExecutor.shutdown();
            boolean shutdown = putExecutor.awaitTermination(8000, TimeUnit.MILLISECONDS);
            if(!shutdown){
                List<Runnable> notRunList = putExecutor.shutdownNow();
                if(CollectionUtils.isNotEmpty(notRunList)){
                    logger.warn("強制關(guān)閉延遲任務(wù)轉(zhuǎn)移線程...未執(zhí)行任務(wù)數(shù)量{}",notRunList.size());
                }
            }
        }
        if(executor != null){
            executor.shutdown();
            boolean shutdown = executor.awaitTermination(8000, TimeUnit.MILLISECONDS);
            if(!shutdown){
                List<Runnable> notRunList = executor.shutdownNow();
                if(CollectionUtils.isNotEmpty(notRunList)){
                    logger.warn("強制關(guān)閉延遲任務(wù)執(zhí)行線程池...未執(zhí)行任務(wù)數(shù)量{}",notRunList.size());
                }
            }
        }
    }

    @Override
    public void afterPropertiesSet() throws Exception {
        running = true;

        executor = new ThreadPoolExecutor(Runtime.getRuntime().availableProcessors() + 1,
            Runtime.getRuntime().availableProcessors() * 2,0,TimeUnit.MILLISECONDS, new LinkedBlockingQueue<>(),
            new ThreadFactoryBuilder()
                .setNameFormat(this.name + "-pool-%d")
                .setUncaughtExceptionHandler(new UncaughtExceptionHandler() {
                    @Override
                    public void uncaughtException(Thread t, Throwable e) {
                        logger.error("延遲任務(wù)執(zhí)行線程異常,thread={}",t,e);
                    }
                })
                .build());

        putExecutor = new ThreadPoolExecutor(1,1,0,TimeUnit.MILLISECONDS, new LinkedBlockingQueue<>(),
            new ThreadFactoryBuilder()
                .setNameFormat(this.name + "-submiter")
                .setUncaughtExceptionHandler(new UncaughtExceptionHandler() {
                    @Override
                    public void uncaughtException(Thread t, Throwable e) {
                        logger.error("延遲任務(wù)轉(zhuǎn)移線程異常,thread={}",t,e);
                    }
                })
                .build());

        putExecutor.execute(new Runnable() {
            @Override
            public void run() {
                while(running || delayQueue.size() > 0){
                    try {
                        Task task = delayQueue.poll(1000, TimeUnit.MILLISECONDS);
                        if(task == null){
                            continue;
                        }
                        logger.info("開始執(zhí)行延遲任務(wù) {}",task);
                        executor.execute(task.getRunnable());
                    } catch (InterruptedException e) {
                        logger.error("延遲任務(wù)轉(zhuǎn)移線程異常interrupted",e);
                    } catch (Throwable e){
                        logger.error("延遲任務(wù)轉(zhuǎn)移線程異常",e);
                    }
                }
                logger.info("延遲任務(wù)轉(zhuǎn)移線程退出");
            }
        });
    }

    class Task implements Delayed{

        // 提交時間
        private long submitTime;

        private long runningTime;

        private Runnable runnable;

        Task(Runnable runnable,long delayTime){
            this.runnable = runnable;
            this.submitTime = System.currentTimeMillis();
            this.runningTime = submitTime + delayTime;
        }

        @Override
        public long getDelay(TimeUnit unit) {
            return unit.convert(runningTime - System.currentTimeMillis(), TimeUnit.MILLISECONDS);
        }

        @Override
        public int compareTo(Delayed o) {
            if(o == null || !(o instanceof Task)){
                return 1;
            }
            if(o == this){
                return 0;
            }
            Task otherTask = (Task) o;
            if(this.runningTime > otherTask.runningTime){
                return 1;
            }else if(this.runningTime == otherTask.runningTime){
                return 0;
            }else{
                return -1;
            }
        }

        public Runnable getRunnable() {
            return runnable;
        }
    }

}

測試代碼:

public static void main(String[] args) throws Exception {
        // 使用Spring可不用調(diào)用afterPropertiesSet和destroy方法
        DelayedTaskExecutor delayedTaskExecutor = new DelayedTaskExecutor("cache-delay");
        delayedTaskExecutor.afterPropertiesSet();
        delayedTaskExecutor.submit(() -> System.out.println("hello"), 2000);
        delayedTaskExecutor.destroy();
    }
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
平臺聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點,簡書系信息發(fā)布平臺,僅提供信息存儲服務(wù)。
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 227,818評論 6 531
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 98,185評論 3 414
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事?!?“怎么了?”我有些...
    開封第一講書人閱讀 175,656評論 0 373
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經(jīng)常有香客問我,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 62,647評論 1 309
  • 正文 為了忘掉前任,我火速辦了婚禮,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當(dāng)我...
    茶點故事閱讀 71,446評論 6 405
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 54,951評論 1 321
  • 那天,我揣著相機與錄音,去河邊找鬼。 笑死,一個胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,041評論 3 440
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 42,189評論 0 287
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 48,718評論 1 333
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 40,602評論 3 354
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發(fā)現(xiàn)自己被綠了。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 42,800評論 1 369
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 38,316評論 5 358
  • 正文 年R本政府宣布,位于F島的核電站,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點故事閱讀 44,045評論 3 347
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 34,419評論 0 26
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 35,671評論 1 281
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個月前我還...
    沈念sama閱讀 51,420評論 3 390
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 47,755評論 2 371

推薦閱讀更多精彩內(nèi)容