欧美亚洲中文,在线国自产视频,欧洲一区在线观看视频,亚洲综合中文字幕在线观看

      1. <dfn id="rfwes"></dfn>
          <object id="rfwes"></object>
        1. 站長(zhǎng)資訊網(wǎng)
          最全最豐富的資訊網(wǎng)站

          一起來(lái)分析Java泛型和泛型的通配符

          本篇文章給大家?guī)?lái)了關(guān)于java的相關(guān)知識(shí),其中主要介紹了關(guān)于泛型以及泛型的通配符相關(guān)問(wèn)題,因?yàn)榉盒偷闹С质蔷幾g器支持,字節(jié)碼加載到虛擬機(jī)的時(shí)候泛型信息已經(jīng)被擦除,所以泛型不支持一些運(yùn)行時(shí)特性,下面一起來(lái)看一下,希望對(duì)大家有幫助。

          一起來(lái)分析Java泛型和泛型的通配符

          程序員必備接口測(cè)試調(diào)試工具:立即使用
          Apipost = Postman + Swagger + Mock + Jmeter
          Api設(shè)計(jì)、調(diào)試、文檔、自動(dòng)化測(cè)試工具
          后端、前端、測(cè)試,同時(shí)在線協(xié)作,內(nèi)容實(shí)時(shí)同步

          推薦學(xué)習(xí):《java視頻教程》

          泛型不是運(yùn)行時(shí)特性

          我們這里依然說(shuō)的是Open JDK

          因?yàn)榉盒偷闹С质蔷幾g器支持,字節(jié)碼加載到虛擬機(jī)的時(shí)候泛型信息已經(jīng)被擦除,所以泛型不支持一些運(yùn)行時(shí)特性。所以要注意有些寫法將編譯不過(guò),比如new。

          如下,類Plate<T>是帶泛型的類,如下演示,

          new Plate(...) new Plate(...) class Plate {     T item;     public Plate(T t) {         new T();//是錯(cuò)誤的,因?yàn)門是一個(gè)不被虛擬機(jī)所識(shí)別的類型,最終會(huì)被編譯器擦除轉(zhuǎn)為Object類給到虛擬機(jī)         item = t;     }     public void set(T t) {         item = t;     }     public T get() {         return item;     } }
          登錄后復(fù)制

          泛型T不能被new,因?yàn)門是一個(gè)不被虛擬機(jī)所識(shí)別的類型。

          泛型通配符

          存在三種形式的用通配符的泛型變量表達(dá),分別是:

          • <? extends A>: C<? extends A> c,c中的元素類型都是A或者A的子類

          • <? super B>:C<? super B> c,c中的元素類型是B或者B的父類

          • <?>:C<?> c,c中的元素類型不確定

          具體是什么意思以及怎么使用,我們一起來(lái)看看吧~

          上界通配符

          在面向?qū)ο缶幊填I(lǐng)域,我們認(rèn)為基類base在最上層。從繼承樹(shù)的角度來(lái)看,Object類處于最上層。

          所以我們將這樣的表達(dá)<? extends T>稱為上界通配符。

          <? extends T>表示T或繼承T類型的任意泛型類型。

          先看下面這個(gè)例子.

          Sping Webmvc中的RequestBodyAdvice

          public interface RequestBodyAdvice {    /**     * Invoked first to determine if this interceptor applies.     * @param methodParameter the method parameter     * @param targetType the target type, not necessarily the same as the method     * parameter type, e.g. for {@code HttpEntity}.     * @param converterType the selected converter type     * @return whether this interceptor should be invoked or not     */    boolean supports(MethodParameter methodParameter, Type targetType,          Class> converterType);    ... }
          登錄后復(fù)制

          在ping Webmvc中,RequestBodyAdvice用來(lái)處理http請(qǐng)求的body,supports用來(lái)判斷是否支持某種參數(shù)類型到HttpMessage請(qǐng)求的轉(zhuǎn)換。

          HttpMessageConverter是一個(gè)接口,比如支持Body為Json格式的JsonViewRequestBodyAdvice類,實(shí)現(xiàn)如下:

          @Override public boolean supports(MethodParameter methodParameter, Type targetType,       Class> converterType) {    return (AbstractJackson2HttpMessageConverter.class.isAssignableFrom(converterType) &&          methodParameter.getParameterAnnotation(JsonView.class) != null); }
          登錄后復(fù)制

          使用AbstractJackson2HttpMessageConverter來(lái)處理JsonView,Jackson2庫(kù)是流行的Java JSON解析庫(kù)之一,也是Springboot自帶的HttpMessageConverter.

          不同的使用方可以自己定義不同類型的Advice,便使得能支持非常多的參數(shù)類型比如xml,那么sping-webmvc的功能也就更加靈活通用了,可以將很多Type通過(guò)不同的HttpMessageConverter翻譯為不同的HttpInputMessage請(qǐng)求。如下所示,

          @Override public HttpInputMessage beforeBodyRead(HttpInputMessage request, MethodParameter parameter,       Type targetType, Class> converterType) throws IOException {    for (RequestBodyAdvice advice : getMatchingAdvice(parameter, RequestBodyAdvice.class)) {       if (advice.supports(parameter, targetType, converterType)) {          request = advice.beforeBodyRead(request, parameter, targetType, converterType);       }    }    return request; }
          登錄后復(fù)制

          通過(guò)getMatchingAdvice(parameter, RequestBodyAdvice.class)獲得匹配的advice列表,遍歷這個(gè)列表解析支持parameter的Advice得到HttpInputMessage類型的請(qǐng)求。

          上界通配符的表達(dá)無(wú)法再set

          使用上屆通配符的表達(dá)方式無(wú)法再設(shè)置泛型字段,其實(shí)意思就是上界通配符不能改變已經(jīng)設(shè)置的泛型類型,我們一起來(lái)看下這個(gè)demo。

              @Test     void genericTest() {                 Plate p = new Plate(new Apple());         p.set(new Apple());//可以set           Apple apple = p.get();                    Plate q = new Plate(new Apple());                 Fruit fruit = q.get();                 q.set(new Fruit());//將編譯錯(cuò)誤     }
          登錄后復(fù)制

          Plate<? extends Fruit>這種表達(dá)方式意味著java編譯期只知道容器里面存放的是Fruit和它的派生類,具體是什么類型不知道,可能是Fruit、Apple或者其他子類, 編譯器在p賦值以后,盤子里面沒(méi)有標(biāo)記為“Apple",只是標(biāo)記了一個(gè)占位符“CAP#1”(可以通過(guò)javap反編譯字節(jié)碼來(lái)嚴(yán)重),來(lái)表示捕獲一個(gè)Fruit或者Fruit的子類。

          但是不管是不是通配符的寫法,泛型終究指的是一種具體的類型,而且被編譯器使用了特殊的“CAP#1”,所以我們無(wú)法再重新設(shè)置這個(gè)字段了,否則就會(huì)出現(xiàn)類型不一致的編譯錯(cuò)誤了。

          但這個(gè)特點(diǎn)對(duì)于用法來(lái)說(shuō)并沒(méi)有妨礙,框架使用上界通配符范型達(dá)到靈活擴(kuò)展的目的。

          下界通配符

          接下來(lái)我們一起看下下界通配符,<? super T>表示T或T父類的任意類型,下界的類型是T。

          語(yǔ)言陷阱

          我們?cè)诶斫馍先菀椎羧胍粋€(gè)陷阱,以為只可以設(shè)置Fruit或Fruit的基類。實(shí)際上Fruit和Fruit的子類才可以設(shè)置進(jìn)去,讓我們寫一個(gè)單元測(cè)試來(lái)看看。

          @Test void genericSuperTest() {     Plate p = new Plate(new Fruit());     p.set(new Apple()); //ok,存取的時(shí)候可以存任意可以轉(zhuǎn)為T的類或T     p.set(new Object()); //not ok,無(wú)法 set Object     Object object = p.get();//ok     Fruit object = p.get();//not ok,super Fruit不是Fruit的子類 }
          登錄后復(fù)制

          存取的時(shí)候可以存可以轉(zhuǎn)為T的類或T,也就是可以設(shè)置Fruit或Fruit子類的類。

          但是使用的時(shí)候必須使用object來(lái)引用。

          spring-kafka的異步回調(diào)

          現(xiàn)在,讓我們看實(shí)際的一個(gè)例子。

          SettableListenableFuture是spring 并發(fā)框架的一個(gè)類,繼承自Future<T>,我們知道Future表示異步執(zhí)行的結(jié)果,T表示返回結(jié)果的類型。ListenableFuture可以支持設(shè)置回調(diào)函數(shù),如果成功了怎么處理,如果異常又如何處理。

          在spring-kafka包里使用了SettableListenableFuture來(lái)設(shè)置異步回調(diào)的結(jié)果,kafka客戶端調(diào)用 doSend發(fā)送消息到kafka隊(duì)列之后,我們可以異步的判斷是否發(fā)送成功。

          public class SettableListenableFuture implements ListenableFuture {   ...    @Override    public void addCallback(ListenableFutureCallback callback) {       this.settableTask.addCallback(callback);    }    @Override    public void addCallback(SuccessCallback successCallback, FailureCallback failureCallback) {       this.settableTask.addCallback(successCallback, failureCallback);    }  ...
          登錄后復(fù)制

          SettableListenableFuture有重載的addCallback函數(shù),支持添加ListenableFutureCallback<? super T> callback和SuccessCallback<? super T> successCallback;當(dāng)調(diào)用的異步方法成功結(jié)束的時(shí)候使用notifySuccess來(lái)觸發(fā)onSuccess的執(zhí)行,這個(gè)時(shí)候?qū)?shí)際異步執(zhí)行的結(jié)果變成參數(shù)給callback調(diào)用。

          private void notifySuccess(SuccessCallback callback) {    try {       callback.onSuccess((T) this.result);    }    catch (Throwable ex) {       // Ignore    } }
          登錄后復(fù)制

          SuccessCallback是一個(gè)函數(shù)式接口,從設(shè)計(jì)模式的角度來(lái)看是一個(gè)消費(fèi)者,消費(fèi)<T>類型的result。ListenableFutureCallback同理。

          public interface SuccessCallback {    /**     * Called when the {@link ListenableFuture} completes with success.     * 

          Note that Exceptions raised by this method are ignored. * @param result the result */ void onSuccess(@Nullable T result); }

          登錄后復(fù)制

          為什么要用notifySuccess(SuccessCallback<? super T> callback)呢?

          這是因?yàn)閟uper能支持的范圍

          贊(0)
          分享到: 更多 (0)
          網(wǎng)站地圖   滬ICP備18035694號(hào)-2    滬公網(wǎng)安備31011702889846號(hào)