站長資訊網
        最全最豐富的資訊網站

        一起來分析Java泛型和泛型的通配符

        本篇文章給大家帶來了關于java的相關知識,其中主要介紹了關于泛型以及泛型的通配符相關問題,因為泛型的支持是編譯器支持,字節碼加載到虛擬機的時候泛型信息已經被擦除,所以泛型不支持一些運行時特性,下面一起來看一下,希望對大家有幫助。

        一起來分析Java泛型和泛型的通配符

        程序員必備接口測試調試工具:立即使用
        Apipost = Postman + Swagger + Mock + Jmeter
        Api設計、調試、文檔、自動化測試工具
        后端、前端、測試,同時在線協作,內容實時同步

        推薦學習:《java視頻教程》

        泛型不是運行時特性

        我們這里依然說的是Open JDK

        因為泛型的支持是編譯器支持,字節碼加載到虛擬機的時候泛型信息已經被擦除,所以泛型不支持一些運行時特性。所以要注意有些寫法將編譯不過,比如new。

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

        new Plate(...) new Plate(...) class Plate {     T item;     public Plate(T t) {         new T();//是錯誤的,因為T是一個不被虛擬機所識別的類型,最終會被編譯器擦除轉為Object類給到虛擬機         item = t;     }     public void set(T t) {         item = t;     }     public T get() {         return item;     } }
        登錄后復制

        泛型T不能被new,因為T是一個不被虛擬機所識別的類型。

        泛型通配符

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

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

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

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

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

        上界通配符

        在面向對象編程領域,我們認為基類base在最上層。從繼承樹的角度來看,Object類處于最上層。

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

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

        先看下面這個例子.

        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);    ... }
        登錄后復制

        在ping Webmvc中,RequestBodyAdvice用來處理http請求的body,supports用來判斷是否支持某種參數類型到HttpMessage請求的轉換。

        HttpMessageConverter是一個接口,比如支持Body為Json格式的JsonViewRequestBodyAdvice類,實現如下:

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

        使用AbstractJackson2HttpMessageConverter來處理JsonView,Jackson2庫是流行的Java JSON解析庫之一,也是Springboot自帶的HttpMessageConverter.

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

        @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; }
        登錄后復制

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

        上界通配符的表達無法再set

        使用上屆通配符的表達方式無法再設置泛型字段,其實意思就是上界通配符不能改變已經設置的泛型類型,我們一起來看下這個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());//將編譯錯誤     }
        登錄后復制

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

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

        但這個特點對于用法來說并沒有妨礙,框架使用上界通配符范型達到靈活擴展的目的。

        下界通配符

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

        語言陷阱

        我們在理解上容易掉入一個陷阱,以為只可以設置Fruit或Fruit的基類。實際上Fruit和Fruit的子類才可以設置進去,讓我們寫一個單元測試來看看。

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

        存取的時候可以存可以轉為T的類或T,也就是可以設置Fruit或Fruit子類的類。

        但是使用的時候必須使用object來引用。

        spring-kafka的異步回調

        現在,讓我們看實際的一個例子。

        SettableListenableFuture是spring 并發框架的一個類,繼承自Future<T>,我們知道Future表示異步執行的結果,T表示返回結果的類型。ListenableFuture可以支持設置回調函數,如果成功了怎么處理,如果異常又如何處理。

        在spring-kafka包里使用了SettableListenableFuture來設置異步回調的結果,kafka客戶端調用 doSend發送消息到kafka隊列之后,我們可以異步的判斷是否發送成功。

        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);    }  ...
        登錄后復制

        SettableListenableFuture有重載的addCallback函數,支持添加ListenableFutureCallback<? super T> callback和SuccessCallback<? super T> successCallback;當調用的異步方法成功結束的時候使用notifySuccess來觸發onSuccess的執行,這個時候將實際異步執行的結果變成參數給callback調用。

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

        SuccessCallback是一個函數式接口,從設計模式的角度來看是一個消費者,消費<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); }

        登錄后復制

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

        這是因為super能支持的范圍

        贊(0)
        分享到: 更多 (0)
        網站地圖   滬ICP備18035694號-2    滬公網安備31011702889846號
        主站蜘蛛池模板: 亚洲国产精品久久久久婷婷软件| 精品国产午夜福利在线观看| AAA级久久久精品无码区| 少妇人妻偷人精品无码视频| 精品国产午夜肉伦伦影院| 国内精品91最新在线观看| 无码精品日韩中文字幕| 久久久这里有精品中文字幕| 91精品国产91久久久久久青草| 精品国产三级a在线观看| 亚洲精品无码av人在线观看 | 国产午夜精品一区二区三区小说| 国产精品久久久天天影视| 国产女人精品视频国产灰线| 亚洲色精品vr一区二区三区| 亚州日韩精品专区久久久| 国产女人18毛片水真多18精品 | 2021精品国产综合久久| 久久香综合精品久久伊人| 亚洲精品中文字幕乱码三区| 亚洲精品一级无码中文字幕 | 国产欧美日韩精品专区| 亚洲国产精品成| 精品国产福利第一区二区三区| 精品国际久久久久999波多野| 亚洲AV无码国产精品麻豆天美| 亚洲日韩欧美制服精品二区| 无码精品人妻一区二区三区免费| 久久精品国产精品亚洲艾草网美妙| 国产精品单位女同事在线| 国产精品 一区 在线| 国产精品成人免费观看 | 国产精品成人观看视频网站| 国产成人无码精品久久久免费 | 日本人精品video黑人| 四虎影视884a精品国产四虎| 先锋影音国产精品| 国产美女精品视频| 日本精品一区二区三区四区| 亚洲精品456播放| 亚洲精品乱码久久久久久按摩|