spring?Cloud微服務(wù)阿里開(kāi)源TTL身份信息的線程間復(fù)用
引言
前面在介紹分布式鏈路追蹤時(shí)講過(guò)異步調(diào)用會(huì)丟失鏈路信息,最終的解決方案是使用對(duì)應(yīng)的包裝類重新包裝一下,如下:
RunnableWrapper
CallableWrapper
SupplierWrapper
還有openFeign異步請(qǐng)求丟失上文的問(wèn)題,這些問(wèn)題追根究底都是ThreadLocal惹得禍。
由于ThreadLocal只能保存當(dāng)前線程的信息,不能實(shí)現(xiàn)父子線程的繼承。
說(shuō)到這,很多人想到了InheritableThreadLocal,確實(shí)InheritableThreadLocal能夠?qū)崿F(xiàn)父子線程間傳遞本地變量,但是.....
但是你的程序如果采用線程池,則存在著線程復(fù)用的情況,這時(shí)就不一定能夠?qū)崿F(xiàn)父子線程間傳遞了,因?yàn)樵诰€程在線程池中的存在不是每次使用都會(huì)進(jìn)行創(chuàng)建,InheritableThreadlocal是在線程初始化時(shí)intertableThreadLocals=true才會(huì)進(jìn)行拷貝傳遞。
所以若本次使用的子線程是已經(jīng)被池化的線程,從線程池中取出線下進(jìn)行使用,是沒(méi)有經(jīng)過(guò)初始化的過(guò)程,也就不會(huì)進(jìn)行父子線程的本地變量拷貝。
由于在日常應(yīng)用場(chǎng)景中,絕大多數(shù)都是會(huì)采用線程池的方式進(jìn)行資源的有效管理。
今天就來(lái)聊一聊阿里的ThansmittableThreadLocal是如何解決線程池中父子線程本地變量傳遞。
InheritableThreadLocal 的問(wèn)題
在介紹ThansmittableThreadLocal之前先來(lái)看一下InheritableThreadLocal 在線程池中的問(wèn)題,如下代碼:
@Test
public?void?test()?throws?Exception?{
????//單一線程池
????ExecutorService?executorService?=?Executors.newSingleThreadExecutor();
????//InheritableThreadLocal存儲(chǔ)
????InheritableThreadLocal<String>?username?=?new?InheritableThreadLocal<>();
????for?(int?i?=?0;?i?<?10;?i++)?{
????username.set("公眾號(hào):腳本之家—"+i);
????Thread.sleep(3000);
????CompletableFuture.runAsync(()->?System.out.println(username.get()),executorService);
???}
}
上述代碼中創(chuàng)建了一個(gè)單一線程池,循環(huán)異步調(diào)用,打印一下username,由于核心線程數(shù)是1,勢(shì)必存在線程的復(fù)用。
打印信息如下:
公眾號(hào):腳本之家—0
公眾號(hào):腳本之家—0
公眾號(hào):腳本之家—0
公眾號(hào):腳本之家—0
公眾號(hào):腳本之家—0
公眾號(hào):腳本之家—0
公眾號(hào):腳本之家—0
公眾號(hào):腳本之家—0
公眾號(hào):腳本之家—0
公眾號(hào):腳本之家—0
看到了嗎?這里并沒(méi)有實(shí)現(xiàn)父子線程間的變量傳遞,這也就是InheritableThreadLocal 的局限性。
TransmittableThreadLocal 使用
TransmittableThreadLocal(TTL):在使用線程池等會(huì)池化復(fù)用線程的執(zhí)行組件情況下,提供ThreadLocal值的傳遞功能,解決異步執(zhí)行時(shí)上下文傳遞的問(wèn)題。
整個(gè)TransmittableThreadLocal庫(kù)的核心功能(用戶API與框架/中間件的集成API、線程池ExecutorService/ForkJoinPool/TimerTask及其線程工廠的Wrapper)。
需求場(chǎng)景:
- 分布式跟蹤系統(tǒng) 或 全鏈路壓測(cè)(即鏈路打標(biāo))
- 日志收集記錄系統(tǒng)上下文
官網(wǎng)地址:https://github.com/alibaba/transmittable-thread-local
下面就以上面的例子改造成TransmittableThreadLocal試一下效果。
首選需要引入對(duì)應(yīng)的依賴,如下:
<dependency> ????<groupId>com.alibaba</groupId> ????<artifactId>transmittable-thread-local</artifactId> </dependency
改造后的代碼如下:
@Test
public?void?test()?throws?Exception?{
????//單一線程池
????ExecutorService?executorService?=?Executors.newSingleThreadExecutor();
????//需要使用TtlExecutors對(duì)線程池包裝一下
????executorService=TtlExecutors.getTtlExecutorService(executorService);
????//TransmittableThreadLocal創(chuàng)建
????TransmittableThreadLocal<String>?username?=?new?TransmittableThreadLocal<>();
????for?(int?i?=?0;?i?<?10;?i++)?{
????username.set("公眾號(hào):https://github.com/alibaba/transmittable-thread-local—"+i);
????Thread.sleep(3000);
????CompletableFuture.runAsync(()->?System.out.println(username.get()),executorService);
??}
}
需要注意的是需要使用TtlExecutors對(duì)線程池進(jìn)行包裝,代碼如下:
executorService=TtlExecutors.getTtlExecutorService(executorService);
運(yùn)行效果如下:
公眾號(hào):腳本之家—0
公眾號(hào):腳本之家—1
公眾號(hào):腳本之家—2
公眾號(hào):腳本之家—3
公眾號(hào):腳本之家—4
公眾號(hào):腳本之家—5
公眾號(hào):腳本之家—6
公眾號(hào):腳本之家—7
公眾號(hào):腳本之家—8
公眾號(hào):腳本之家—9
可以看到已經(jīng)能夠?qū)崿F(xiàn)了線程池中的父子線程的數(shù)據(jù)傳遞。
在每次調(diào)用任務(wù)的時(shí),都會(huì)將當(dāng)前的主線程的TTL數(shù)據(jù)copy到子線程里面,執(zhí)行完成后,再清除掉。同時(shí)子線程里面的修改回到主線程時(shí)其實(shí)并沒(méi)有生效。這樣可以保證每次任務(wù)執(zhí)行的時(shí)候都是互不干涉。
簡(jiǎn)單應(yīng)用
在 Spring Security 往往需要存儲(chǔ)用戶登錄的詳細(xì)信息,這樣在業(yè)務(wù)方法中能夠隨時(shí)獲取用戶的信息。
在前面的Spring Cloud Gateway整合OAuth2.0實(shí)現(xiàn)統(tǒng)一認(rèn)證鑒權(quán) 文章中筆者是將用戶信息直接存儲(chǔ)在Request中,這樣每次請(qǐng)求都能獲取到對(duì)應(yīng)的信息。
其實(shí)Request中的信息存儲(chǔ)也是通過(guò)ThreadLocal完成的,在異步執(zhí)行的時(shí)候還是需要重新轉(zhuǎn)存,這樣一來(lái)代碼就變得復(fù)雜。
那么了解了TransmittableThreadLocal 之后,完全可以使用這個(gè)存儲(chǔ)用戶的登錄信息,實(shí)現(xiàn)如下:
/**
?*?@description?使用TransmittableThreadLocal存儲(chǔ)用戶身份信息LoginVal
?*/
public?class?SecurityContextHolder?{
????//使用TTL存儲(chǔ)身份信息
????private?static?final?TransmittableThreadLocal<LoginVal>?THREAD_LOCAL?=?new?TransmittableThreadLocal<>();
????public?static?void?set(LoginVal?loginVal){
????????THREAD_LOCAL.set(loginVal);
????}
????public?static?LoginVal?get(){
????????return?THREAD_LOCAL.get();
????}
????public?static?void?remove(){
????????THREAD_LOCAL.remove();
????}
}
由于mvc中的一次請(qǐng)求對(duì)應(yīng)一個(gè)線程,因此只需要在攔截器中的設(shè)置和移除TransmittableThreadLocal中的信息,代碼如下:
/**
?*?@description?攔截器,在preHandle中解析請(qǐng)求頭的中的token信息,將其放入SecurityContextHolder中
?*??????????????????????在afterCompletion方法中移除對(duì)應(yīng)的ThreadLocal中信息
?*??????????????????????確保每個(gè)請(qǐng)求的用戶信息獨(dú)立
?*/
@Component
public?class?AuthInterceptor?implements?AsyncHandlerInterceptor?{
????/**
?????*?在執(zhí)行controller方法之前將請(qǐng)求頭中的token信息解析出來(lái),放入SecurityContextHolder中(TransmittableThreadLocal)
?????*/
????@Override
????public?boolean?preHandle(HttpServletRequest?request,?HttpServletResponse?response,?Object?handler)?{
????????if?(!(handler?instanceof?HandlerMethod))
????????????return?true;
????????//獲取請(qǐng)求頭中的加密的用戶信息
????????String?token?=?request.getHeader(OAuthConstant.TOKEN_NAME);
????????if?(StrUtil.isBlank(token))
????????????return?true;
????????//解密
????????String?json?=?Base64.decodeStr(token);
????????//將json解析成LoginVal
????????LoginVal?loginVal?=?TokenUtils.parseJsonToLoginVal(json);
????????//封裝數(shù)據(jù)到ThreadLocal中
????????SecurityContextHolder.set(loginVal);
????????return?true;
????}
????/**
?????*?在視圖渲染之后執(zhí)行,意味著一次請(qǐng)求結(jié)束,清除TTL中的身份信息
?????*/
????@Override
????public?void?afterCompletion(HttpServletRequest?request,?HttpServletResponse?response,?Object?handler,?Exception?ex){
????????SecurityContextHolder.remove();
????}
}
原理
從定義來(lái)看,TransimittableThreadLocal繼承于InheritableThreadLocal,并實(shí)現(xiàn)TtlCopier接口,它里面只有一個(gè)copy方法。所以主要是對(duì)InheritableThreadLocal的擴(kuò)展。
public?class?TransmittableThreadLocal<T>?extends?InheritableThreadLocal<T>?implements?TtlCopier<T>?
在TransimittableThreadLocal中添加holder屬性。這個(gè)屬性的作用就是被標(biāo)記為具備線程傳遞資格的對(duì)象都會(huì)被添加到這個(gè)對(duì)象中。
要標(biāo)記一個(gè)類,比較容易想到的方式,就是給這個(gè)類新增一個(gè)Type字段,還有一個(gè)方法就是將具備這種類型的的對(duì)象都添加到一個(gè)靜態(tài)全局集合中。之后使用時(shí),這個(gè)集合里的所有值都具備這個(gè)標(biāo)記。
//?1.?holder本身是一個(gè)InheritableThreadLocal對(duì)象
//?2.?這個(gè)holder對(duì)象的value是WeakHashMap<TransmittableThreadLocal<Object>,??>
//?? 2.1 WeekHashMap的value總是null,且不可能被使用。
//????2.2?WeekHasshMap支持value=null
private?static?InheritableThreadLocal<WeakHashMap<TransmittableThreadLocal<Object>,??>>?holder?=?new?InheritableThreadLocal<WeakHashMap<TransmittableThreadLocal<Object>,??>>()?{
??@Override
??protected?WeakHashMap<TransmittableThreadLocal<Object>,??>?initialValue()?{
????return?new?WeakHashMap<TransmittableThreadLocal<Object>,?Object>();
??}
?
??/**
???*?重寫了childValue方法,實(shí)現(xiàn)上直接將父線程的屬性作為子線程的本地變量對(duì)象。
???*/
??@Override
??protected?WeakHashMap<TransmittableThreadLocal<Object>,??>?childValue(WeakHashMap<TransmittableThreadLocal<Object>,??>?parentValue)?{
????return?new?WeakHashMap<TransmittableThreadLocal<Object>,?Object>(parentValue);
??}
};
應(yīng)用代碼是通過(guò)TtlExecutors工具類對(duì)線程池對(duì)象進(jìn)行包裝。工具類只是簡(jiǎn)單的判斷,輸入的線程池是否已經(jīng)被包裝過(guò)、非空校驗(yàn)等,然后返回包裝類ExecutorServiceTtlWrapper。根據(jù)不同的線程池類型,有不同和的包裝類。
@Nullable
public?static?ExecutorService?getTtlExecutorService(@Nullable?ExecutorService?executorService)?{
??if?(TtlAgent.isTtlAgentLoaded()?||?executorService?==?null?||?executorService?instanceof?TtlEnhanced)?{
????return?executorService;
??}
??return?new?ExecutorServiceTtlWrapper(executorService);
}
進(jìn)入包裝類ExecutorServiceTtlWrapper??梢宰⒁獾讲徽撌峭ㄟ^(guò)ExecutorServiceTtlWrapper#submit方法或者是ExecutorTtlWrapper#execute方法,都會(huì)將線程對(duì)象包裝成TtlCallable或者TtlRunnable,用于在真正執(zhí)行run方法前做一些業(yè)務(wù)邏輯。
/**
?*?在ExecutorServiceTtlWrapper實(shí)現(xiàn)submit方法
?*/
@NonNull
@Override
public?<T>?Future<T>?submit(@NonNull?Callable<T>?task)?{
??return?executorService.submit(TtlCallable.get(task));
}
/**
?*?在ExecutorTtlWrapper實(shí)現(xiàn)execute方法
?*/
@Override
public?void?execute(@NonNull?Runnable?command)?{
??executor.execute(TtlRunnable.get(command));
}
所以,重點(diǎn)的核心邏輯應(yīng)該是在TtlCallable#call()或者TtlRunnable#run()中。以下以TtlCallable為例,TtlRunnable同理類似。在分析call()方法之前,先看一個(gè)類Transmitter
public?static?class?Transmitter?{
??/**
????*?捕獲當(dāng)前線程中的是所有TransimittableThreadLocal和注冊(cè)ThreadLocal的值。
????*/
??@NonNull
??public?static?Object?capture()?{
????return?new?Snapshot(captureTtlValues(),?captureThreadLocalValues());
??}
?
????/**
????*?捕獲TransimittableThreadLocal的值,將holder中的所有值都添加到HashMap后返回。
????*/
??private?static?HashMap<TransmittableThreadLocal<Object>,?Object>?captureTtlValues()?{
????HashMap<TransmittableThreadLocal<Object>,?Object>?ttl2Value?=?
??????new?HashMap<TransmittableThreadLocal<Object>,?Object>();
????for?(TransmittableThreadLocal<Object>?threadLocal?:?holder.get().keySet())?{
??????ttl2Value.put(threadLocal,?threadLocal.copyValue());
????}
????return?ttl2Value;
??}
??/**
????*?捕獲注冊(cè)的ThreadLocal的值,也就是原本線程中的ThreadLocal,可以注冊(cè)到TTL中,在
????*?進(jìn)行線程池本地變量傳遞時(shí)也會(huì)被傳遞。
????*/
??private?static?HashMap<ThreadLocal<Object>,?Object>?captureThreadLocalValues()?{
????final?HashMap<ThreadLocal<Object>,?Object>?threadLocal2Value?=?
??????new?HashMap<ThreadLocal<Object>,?Object>();
????for(Map.Entry<ThreadLocal<Object>,TtlCopier<Object>>entry:threadLocalHolder.entrySet()){
??????final?ThreadLocal<Object>?threadLocal?=?entry.getKey();
??????final?TtlCopier<Object>?copier?=?entry.getValue();
??????threadLocal2Value.put(threadLocal,?copier.copy(threadLocal.get()));
????}
????return?threadLocal2Value;
??}
??/**
????*?將捕獲到的本地變量進(jìn)行替換子線程的本地變量,并且返回子線程現(xiàn)有的本地變量副本backup。
????*?用于在執(zhí)行run/call方法之后,將本地變量副本恢復(fù)。
????*/
??@NonNull
??public?static?Object?replay(@NonNull?Object?captured)?{
????final?Snapshot?capturedSnapshot?=?(Snapshot)?captured;
????return?new?Snapshot(replayTtlValues(capturedSnapshot.ttl2Value),?
????????????????????????replayThreadLocalValues(capturedSnapshot.threadLocal2Value));
??}
?
??/**
????*?替換TransmittableThreadLocal
????*/
??@NonNull
??private?static?HashMap<TransmittableThreadLocal<Object>,?Object>?replayTtlValues(@NonNull?HashMap<TransmittableThreadLocal<Object>,?Object>?captured)?{
????//?創(chuàng)建副本backup
????HashMap<TransmittableThreadLocal<Object>,?Object>?backup?=?
??????new?HashMap<TransmittableThreadLocal<Object>,?Object>();
????for?(final?Iterator<TransmittableThreadLocal<Object>>?iterator?=?holder.get().keySet().iterator();?iterator.hasNext();?)?{
??????TransmittableThreadLocal<Object>?threadLocal?=?iterator.next();
??????//?對(duì)當(dāng)前線程的本地變量進(jìn)行副本拷貝
??????backup.put(threadLocal,?threadLocal.get());
??????//?若出現(xiàn)調(diào)用線程中不存在某個(gè)線程變量,而線程池中線程有,則刪除線程池中對(duì)應(yīng)的本地變量
??????if?(!captured.containsKey(threadLocal))?{
????????iterator.remove();
????????threadLocal.superRemove();
??????}
????}
????//?將捕獲的TTL值打入線程池獲取到的線程TTL中。
????setTtlValuesTo(captured);
????//?是一個(gè)擴(kuò)展點(diǎn),調(diào)用TTL的beforeExecute方法。默認(rèn)實(shí)現(xiàn)為空
????doExecuteCallback(true);
????return?backup;
??}
??private?static?HashMap<ThreadLocal<Object>,?Object>?replayThreadLocalValues(@NonNull?HashMap<ThreadLocal<Object>,?Object>?captured)?{
????final?HashMap<ThreadLocal<Object>,?Object>?backup?=?
??????new?HashMap<ThreadLocal<Object>,?Object>();
????for?(Map.Entry<ThreadLocal<Object>,?Object>?entry?:?captured.entrySet())?{
??????final?ThreadLocal<Object>?threadLocal?=?entry.getKey();
??????backup.put(threadLocal,?threadLocal.get());
??????final?Object?value?=?entry.getValue();
??????if?(value?==?threadLocalClearMark)?threadLocal.remove();
??????else?threadLocal.set(value);
????}
????return?backup;
??}
??/**
????*?清除單線線程的所有TTL和TL,并返回清除之氣的backup
????*/
??@NonNull
??public?static?Object?clear()?{
????final?HashMap<TransmittableThreadLocal<Object>,?Object>?ttl2Value?=?
??????new?HashMap<TransmittableThreadLocal<Object>,?Object>();
????final?HashMap<ThreadLocal<Object>,?Object>?threadLocal2Value?=?
??????new?HashMap<ThreadLocal<Object>,?Object>();
????for(Map.Entry<ThreadLocal<Object>,TtlCopier<Object>>entry:threadLocalHolder.entrySet()){
??????final?ThreadLocal<Object>?threadLocal?=?entry.getKey();
??????threadLocal2Value.put(threadLocal,?threadLocalClearMark);
????}
????return?replay(new?Snapshot(ttl2Value,?threadLocal2Value));
??}
??/**
????*?還原
????*/
??public?static?void?restore(@NonNull?Object?backup)?{
????final?Snapshot?backupSnapshot?=?(Snapshot)?backup;
????restoreTtlValues(backupSnapshot.ttl2Value);
????restoreThreadLocalValues(backupSnapshot.threadLocal2Value);
??}
??private?static?void?restoreTtlValues(@NonNull?HashMap<TransmittableThreadLocal<Object>,?Object>?backup)?{
????//?擴(kuò)展點(diǎn),調(diào)用TTL的afterExecute
????doExecuteCallback(false);
????for?(final?Iterator<TransmittableThreadLocal<Object>>?iterator?=?holder.get().keySet().iterator();?iterator.hasNext();?)?{
??????TransmittableThreadLocal<Object>?threadLocal?=?iterator.next();
??????if?(!backup.containsKey(threadLocal))?{
????????iterator.remove();
????????threadLocal.superRemove();
??????}
????}
????//?將本地變量恢復(fù)成備份版本
????setTtlValuesTo(backup);
??}
??private?static?void?setTtlValuesTo(@NonNull?HashMap<TransmittableThreadLocal<Object>,?Object>?ttlValues)?{
????for?(Map.Entry<TransmittableThreadLocal<Object>,?Object>?entry?:?ttlValues.entrySet())?{
??????TransmittableThreadLocal<Object>?threadLocal?=?entry.getKey();
??????threadLocal.set(entry.getValue());
????}
??}
??private?static?void?restoreThreadLocalValues(@NonNull?HashMap<ThreadLocal<Object>,?Object>?backup)?{
????for?(Map.Entry<ThreadLocal<Object>,?Object>?entry?:?backup.entrySet())?{
??????final?ThreadLocal<Object>?threadLocal?=?entry.getKey();
??????threadLocal.set(entry.getValue());
????}
??}
??/**
???*?快照類,保存TTL和TL
???*/
??private?static?class?Snapshot?{
????final?HashMap<TransmittableThreadLocal<Object>,?Object>?ttl2Value;
????final?HashMap<ThreadLocal<Object>,?Object>?threadLocal2Value;
????private?Snapshot(HashMap<TransmittableThreadLocal<Object>,?Object>?ttl2Value,
?????????????????????HashMap<ThreadLocal<Object>,?Object>?threadLocal2Value)?{
??????this.ttl2Value?=?ttl2Value;
??????this.threadLocal2Value?=?threadLocal2Value;
????}
??}
進(jìn)入TtlCallable#call()方法。
@Override
public?V?call()?throws?Exception?{
??Object?captured?=?capturedRef.get();
??if?(captured?==?null?||?releaseTtlValueReferenceAfterCall?&&?
??????!capturedRef.compareAndSet(captured,?null))?{
????throw?new?IllegalStateException("TTL?value?reference?is?released?after?call!");
??}
??//?調(diào)用replay方法將捕獲到的當(dāng)前線程的本地變量,傳遞給線程池線程的本地變量,
??//?并且獲取到線程池線程覆蓋之前的本地變量副本。
??Object?backup?=?replay(captured);
??try?{
????//?線程方法調(diào)用
????return?callable.call();
??}?finally?{
????//?使用副本進(jìn)行恢復(fù)。
????restore(backup);
??}
}
到這基本上線程池方式傳遞本地變量的核心代碼已經(jīng)大概看完了。總的來(lái)說(shuō)在創(chuàng)建TtlCallable對(duì)象是,調(diào)用capture()方法捕獲調(diào)用方的本地線程變量,在call()執(zhí)行時(shí),將捕獲到的線程變量,替換到線程池所對(duì)應(yīng)獲取到的線程的本地變量中,并且在執(zhí)行完成之后,將其本地變量恢復(fù)到調(diào)用之前。
總結(jié)
本文介紹了使用阿里開(kāi)源的TransmittableThreadLocal 優(yōu)雅的實(shí)現(xiàn)父子線程的數(shù)據(jù)傳遞,應(yīng)用場(chǎng)景很多,企業(yè)中應(yīng)用也比較廣泛。
- 基于jib-maven-plugin插件快速構(gòu)建微服務(wù)docker鏡像的方法
- 微服務(wù)鏈路追蹤Spring Cloud Sleuth整合Zipkin解析
- Java微服務(wù)Filter過(guò)濾器集成Sentinel實(shí)現(xiàn)網(wǎng)關(guān)限流過(guò)程詳解
- Java微服務(wù)分布式調(diào)度Elastic-job環(huán)境搭建及配置
- Java微服務(wù)Nacos Config配置中心超詳細(xì)講解
- SpringCloud微服務(wù)中跨域配置的方法詳解
- Java Feign微服務(wù)接口調(diào)用方法詳細(xì)講解
- go微服務(wù)PolarisMesh源碼解析服務(wù)端啟動(dòng)流程
- 微服務(wù)Spring Boot 整合 Redis 實(shí)現(xiàn)UV 數(shù)據(jù)統(tǒng)計(jì)的詳細(xì)過(guò)程
- go-micro微服務(wù)JWT跨域認(rèn)證問(wèn)題
- 詳解go-micro微服務(wù)consul配置及注冊(cè)中心
- go-micro微服務(wù)domain層開(kāi)發(fā)示例詳解
- 微服務(wù)?Spring?Boot?整合?Redis?BitMap?實(shí)現(xiàn)?簽到與統(tǒng)計(jì)功能
- 一文帶你了解微服務(wù)架構(gòu)中的"發(fā)件箱模式"
- go?micro微服務(wù)框架項(xiàng)目搭建方法
- go?micro微服務(wù)proto開(kāi)發(fā)安裝及使用規(guī)則
- Mybatis與微服務(wù)注冊(cè)的詳細(xì)過(guò)程
- 簡(jiǎn)單介紹一下什么是microservice微服務(wù)
相關(guān)文章
Java多線程實(shí)現(xiàn)聊天客戶端和服務(wù)器
這篇文章主要為大家詳細(xì)介紹了Java多線程聊天客戶端和服務(wù)器實(shí)現(xiàn)代碼,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2016-10-10
Spring Security獲取用戶認(rèn)證信息的實(shí)現(xiàn)流程
Spring Security是一個(gè)能夠?yàn)榛赟pring的企業(yè)應(yīng)用系統(tǒng)提供聲明式的安全訪問(wèn)控制解決方案的安全框架。它提供了一組可以在Spring應(yīng)用上下文中配置的Bean,充分利用了Spring IoC,DI和AOP功能,為應(yīng)用系統(tǒng)提供聲明式的安全訪問(wèn)控制功能2022-12-12
Java文件處理之使用itextpdf實(shí)現(xiàn)excel轉(zhuǎn)pdf
在文件處理中,經(jīng)常有文件類型轉(zhuǎn)換的使用場(chǎng)景,本文主要介紹了如何使用poi以及itextpdf完成excel轉(zhuǎn)pdf的操作,需要的小伙伴可以參考一下2024-02-02
mybatis-flex實(shí)現(xiàn)多數(shù)據(jù)源操作
MyBaits-Flex內(nèi)置了功能完善的多數(shù)據(jù)源支持,本文主要介紹了mybatis-flex實(shí)現(xiàn)多數(shù)據(jù)源操作,具有一定的參考價(jià)值,感興趣的可以了解一下2024-06-06
Alibaba?Fastjson之超好用的JOSN解析庫(kù)
這篇文章主要介紹了Alibaba?Fastjson之超好用的JOSN解析庫(kù),具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2022-10-10
Spring整合Quartz Job以及Spring Task的實(shí)現(xiàn)方法
下面小編就為大家分享一篇Spring整合Quartz Job以及Spring Task的實(shí)現(xiàn)方法,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2017-12-12
Map 使用 Lambda 的 forEach 實(shí)現(xiàn)跳出循環(huán)操作
這篇文章主要介紹了Map 使用 Lambda 的 forEach 實(shí)現(xiàn)跳出循環(huán)操作,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2020-09-09
解決打開(kāi)的idea項(xiàng)目maven不生效問(wèn)題
這篇文章主要給大家介紹了關(guān)于如何解決打開(kāi)的idea項(xiàng)目maven不生效問(wèn)題,最近在配置maven時(shí),發(fā)現(xiàn)無(wú)論配置幾遍,IDEA中的maven配置總會(huì)還原成默認(rèn)的,所以這里給大家分享下解決辦法,需要的朋友可以參考下2023-07-07
java實(shí)現(xiàn)簡(jiǎn)單的學(xué)生信息管理系統(tǒng)代碼實(shí)例
這篇文章主要介紹了java實(shí)現(xiàn)簡(jiǎn)單的學(xué)生信息管理系統(tǒng),文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2019-04-04
Java根據(jù)坐標(biāo)經(jīng)緯度計(jì)算兩點(diǎn)距離5種方法及校驗(yàn)經(jīng)緯度是否在圓/多邊形區(qū)域內(nèi)的算法推薦
在項(xiàng)目開(kāi)發(fā)過(guò)程中需要根據(jù)兩地經(jīng)緯度坐標(biāo)計(jì)算兩地間距離,下面這篇文章主要給大家介紹了關(guān)于Java根據(jù)坐標(biāo)經(jīng)緯度計(jì)算兩點(diǎn)距離5種方法以及校驗(yàn)經(jīng)緯度是否在圓/多邊形區(qū)域內(nèi)的算法推薦,需要的朋友可以參考下2023-12-12

