Java中的異步回調(diào)問(wèn)題
有時(shí)候我們執(zhí)行一個(gè)任務(wù)需要很長(zhǎng)時(shí)間,單線程環(huán)境下 會(huì)處于阻塞狀態(tài),嚴(yán)重影響效率,那么可以使用一種非阻塞的處理方式,這就是使用多線程。
多線程情況下,問(wèn)題出來(lái)了,我們不知道線程什么時(shí)候執(zhí)行完畢,或者不知道如何處理子線程的結(jié)果
那么就有以下兩種方式:異步回調(diào)和同步等待線程處理結(jié)果兩種方式
1. 異步回調(diào)
直接上代碼,固定格式:主業(yè)務(wù)代碼和一個(gè)處理結(jié)果的接口
/** * Callback * 回調(diào)一般是異步處理的一種技術(shù)。 * 一個(gè)回調(diào)是被傳遞到并且執(zhí)行完該方法。 這種方式只能異步回調(diào), * 如果需要同步等待線程處理結(jié)果可以使用下面介紹的Futures */ interface MyCallback { void doCallback(Map<String, Object> params); } public class TestAsyncCallBack { static ExecutorService es = Executors.newFixedThreadPool(2); public static void doSomething(MyCallback callback) { // 初始化一個(gè)線程 Thread t = new Thread() { public void run() { // 這里是業(yè)務(wù)邏輯處理 System.out.println("子線任務(wù)開(kāi)始執(zhí)行:" + Thread.currentThread().getId()); // 為了能看出效果 ,讓當(dāng)前線程阻塞5秒 try { Thread.sleep(5000); } catch (InterruptedException e) { e.printStackTrace(); } System.out.println("子線任務(wù)結(jié)束執(zhí)行:"); // 處理完業(yè)務(wù)邏輯, Map<String, Object> params = new HashMap<String, Object>(); params.put("a1", "子線程處理結(jié)果"); callback.doCallback(params); } }; es.execute(t); //一定要調(diào)用這個(gè)方法,不然executorService.isTerminated()永遠(yuǎn)不為true es.shutdown(); } public static void main(String[] args) { // 內(nèi)部類 等價(jià)于 new MyCallBack(){...}, 主要作用就是重寫(xiě)doCallback方法 doSomething((params) -> { System.out.println("單個(gè)線程也已經(jīng)處理完畢了,返回參數(shù)a1=" + params.get("a1")); }); System.out.println("主線任務(wù)已經(jīng)執(zhí)行完了:" + Thread.currentThread().getId()); } }
2. 使用Future同步等待執(zhí)行結(jié)果
主要是利用了Callable接口開(kāi)啟進(jìn)行多線程,call方法可以返回結(jié)果
/** * Futures是一個(gè)抽象的概念,它表示一個(gè)值,該值可能在某一點(diǎn)變得可用。一個(gè)Future要么獲得計(jì)算完的結(jié)果,要么獲得計(jì)算失敗后的異常 * 每傳遞一個(gè)Runnable對(duì)象到ExecutorService.submit()方法就會(huì)得到一個(gè)回調(diào)的Future,你能使用它檢測(cè)是否執(zhí)行,這種方法可以是同步等待線處理結(jié)果 */ public class TestFuture { public static void main(String[] args) { Callable<Result> callable = new Callable<Result>() { @Override public Result call() throws Exception { //這里是業(yè)務(wù)邏輯處理 //讓當(dāng)前線程阻塞1秒看下效果 Thread.sleep(5000); return new Result("張三"); } }; ExecutorService executorService = Executors.newFixedThreadPool(5); Future<Result> resultFuture = executorService.submit(callable); executorService.shutdown(); /** * 無(wú)限循環(huán)等待任務(wù)處理完畢 如果已經(jīng)處理完畢 isDone返回true */ while (!resultFuture.isDone()) { try { Result result = resultFuture.get(); // 在這一步阻塞知直到得到子線程返回結(jié)果 System.out.println(result.getName()); } catch (InterruptedException e) { e.printStackTrace(); } catch (ExecutionException e) { e.printStackTrace(); } } } } class Result { private String name; public Result(String name) { this.name = name; } public String getName() { return name; } }
總結(jié)
以上為個(gè)人經(jīng)驗(yàn),希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。
相關(guān)文章
Java實(shí)現(xiàn)一個(gè)簡(jiǎn)單的線程池代碼示例
線程池是管理線程的一個(gè)池子,通過(guò)阻塞隊(duì)列管理任務(wù),主要參數(shù)包括corePoolSize、maximumPoolSize、keepAliveTime等,這篇文章主要介紹了Java實(shí)現(xiàn)一個(gè)簡(jiǎn)單的線程池的相關(guān)資料,需要的朋友可以參考下2024-09-09mybatis+springboot發(fā)布postgresql數(shù)據(jù)的實(shí)現(xiàn)
本文主要介紹了mybatis+springboot發(fā)布postgresql數(shù)據(jù)的實(shí)現(xiàn),文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2023-11-11Java?Calendar類使用之日期和時(shí)間處理指南
這篇文章主要給大家介紹了關(guān)于Java?Calendar類使用之日期和時(shí)間處理指南的相關(guān)資料,Calendar類是Java中用于處理日期和時(shí)間的抽象類,它提供了一種獨(dú)立于特定日歷系統(tǒng)的方式來(lái)處理日期和時(shí)間,需要的朋友可以參考下2023-12-12EntityWrapper如何在and條件中嵌套o(hù)r語(yǔ)句
這篇文章主要介紹了EntityWrapper如何在and條件中嵌套o(hù)r語(yǔ)句,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2022-03-03SpringBoot整合Redis實(shí)現(xiàn)高并發(fā)數(shù)據(jù)緩存的示例講解
這篇文章主要介紹了SpringBoot整合Redis實(shí)現(xiàn)高并發(fā)數(shù)據(jù)緩存,本文通過(guò)實(shí)例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2023-03-03Mybatis Plus條件構(gòu)造器ConditionConstructor用法實(shí)例解析
這篇文章主要介紹了Mybatis Plus條件構(gòu)造器ConditionConstructor用法實(shí)例解析,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2020-08-08