日韩成人免费在线_国产成人一二_精品国产免费人成电影在线观..._日本一区二区三区久久久久久久久不

當(dāng)前位置:首頁 > 科技  > 軟件

Java 泛型編程所說的類型擦除到底是什么?

來源: 責(zé)編: 時間:2024-06-07 17:17:23 188觀看
導(dǎo)讀大部分語言都支持泛型,泛型是一種語言機制,各種語言的實現(xiàn)機制都不太一樣,例如C++使用模板方式來實現(xiàn)泛型,而 Java 中用類型擦除機制來實現(xiàn)泛型。什么是泛型在 Java 中,不會泛型,寸步難行。泛型可能是一個 Java 初學(xué)者需要

大部分語言都支持泛型,泛型是一種語言機制,各種語言的實現(xiàn)機制都不太一樣,例如C++使用模板方式來實現(xiàn)泛型,而 Java 中用類型擦除機制來實現(xiàn)泛型。rc728資訊網(wǎng)——每日最新資訊28at.com

什么是泛型

在 Java 中,不會泛型,寸步難行。泛型可能是一個 Java 初學(xué)者需要攻克的第一個難點。隨便跟著一門教程或 任何一本《Java入門到精通》,前面關(guān)于變量、關(guān)鍵字、語法(if、while、for等等)這些基本上是一看就懂,而當(dāng)內(nèi)容來到泛型的時候,大部分人可能就突然感覺沒那么輕松了。rc728資訊網(wǎng)——每日最新資訊28at.com

如果沒有編程經(jīng)驗的話,可能需要練習(xí)一段時間才能完全掌握泛型編程概念和技巧,這么說吧,有些人寫了好幾年代碼,碰到泛型的時候可能還是不太熟練。rc728資訊網(wǎng)——每日最新資訊28at.com

說到Java泛型,最明顯的標(biāo)志就是 <> 。rc728資訊網(wǎng)——每日最新資訊28at.com

泛型是什么呢?通俗的說就是一個類型是沒有固定類型的,即可以是Integer 也可以是 Long,還可能是你自定義的類。rc728資訊網(wǎng)——每日最新資訊28at.com

泛型使類型(類和接口)能夠在定義類、接口和方法時成為參數(shù)。與方法聲明中使用的更熟悉的形式參數(shù)非常相似,類型參數(shù)為您提供了一種通過不同輸入重復(fù)使用相同代碼的方法。區(qū)別在于形式參數(shù)的輸入是值,而類型參數(shù)的輸入是類型。rc728資訊網(wǎng)——每日最新資訊28at.com

例如在類定義中使用泛型,最常見的 ArrayListrc728資訊網(wǎng)——每日最新資訊28at.com

public class ArrayList<E> extends AbstractList<E>        implements List<E>, RandomAccess, Cloneable, java.io.Serializable{//... code}

例如在方法參數(shù)中使用泛型,來一個復(fù)雜的例子rc728資訊網(wǎng)——每日最新資訊28at.com

public static <T extends Number & Comparable<T>, U extends List<T>, R extends T> R complexMethod(U list, T element) {}

在這個例子中,有兩個傳入?yún)?shù) U list, T element,而這兩個參數(shù)需要在方法的返回類型前用<>做出說明,也就是 <T extends Number & Comparable<T>, U extends List<T>, R extends T>這一部分。rc728資訊網(wǎng)——每日最新資訊28at.com

返回值也是一個泛型 R。rc728資訊網(wǎng)——每日最新資訊28at.com

為什么是 T、U、R

經(jīng)常看到泛型類型用 T、U、R,還有K、V 這樣的符號表示。我們肯定知道不用T也完全沒問題,用 X 也可以。rc728資訊網(wǎng)——每日最新資訊28at.com

之所以這么統(tǒng)一是因為這是官方比較推薦的寫法,推薦的規(guī)則如下:rc728資訊網(wǎng)——每日最新資訊28at.com

  • E - 表示一個元素,例如集合元素、數(shù)組元素
  • K - 表示一個 Key,鍵值對經(jīng)常用到,與之對應(yīng)的是 V
  • V - 表示一個 Value,鍵值對經(jīng)常用
  • N - 表示 Number(數(shù)字類型)
  • T - 這個見得最多,表示一個類型 Type,不管是基礎(chǔ)類型還是自定義的類

泛型的作用

前面也說了,當(dāng)一個參數(shù)預(yù)期可能有多種類型的時候,就會用到泛型,那既然是類型不確定,那直接用 Object 不就行了嗎,何必費事兒呢?一會兒講到類型擦除的時候會發(fā)現(xiàn),本身類型擦除的核心就是把泛型類型轉(zhuǎn)為 Object。但是這是編譯器干的,為了給JVM看的。而作為開發(fā)者和編譯器,使用泛型還是有很大好處的。rc728資訊網(wǎng)——每日最新資訊28at.com

1、在編譯時提供更嚴(yán)格的類型檢查,如果代碼違反類型安全,編譯器可以及時發(fā)現(xiàn),而不是等到運行的時候拋出運行時異常。rc728資訊網(wǎng)——每日最新資訊28at.com

2、使程序員能夠?qū)崿F(xiàn)通用算法。通過使用泛型,程序員可以實現(xiàn)適用于不同類型集合的泛型算法,可以自定義,并且類型安全且更易于閱讀。rc728資訊網(wǎng)——每日最新資訊28at.com

例如下面這個方法,只接受Number 類型的參數(shù),用來比較兩數(shù)的大小。rc728資訊網(wǎng)——每日最新資訊28at.com

public static <T extends Number> Boolean compare(T first, T second) {        double firstValue = first.doubleValue();        double secondValue = second.doubleValue();        return firstValue > secondValue;    }

3、消除不必要的類型轉(zhuǎn)換。rc728資訊網(wǎng)——每日最新資訊28at.com

例如下面不用泛型的情況,每次取數(shù)據(jù)的時候都要轉(zhuǎn)換一下類型。rc728資訊網(wǎng)——每日最新資訊28at.com

List list = new ArrayList();list.add("hello");String s = (String) list.get(0);

而用了泛型后,就不用自己轉(zhuǎn)換了。rc728資訊網(wǎng)——每日最新資訊28at.com

List<String> list = new ArrayList<String>();list.add("hello");String s = list.get(0);

類型擦除

Java 中的泛型實現(xiàn)可以說就是用的類型擦除原理。通俗一點說,類型只在編譯期存在,在運行時就不在了,都變?yōu)榱?Object,一視同仁。rc728資訊網(wǎng)——每日最新資訊28at.com

在我們寫好代碼進行編譯時,編譯器會將泛型參數(shù)的類型進行替換,大部分情況下會將類型替換為 0bject 類型。這種行為模式用類型擦除來描述就非常形象。rc728資訊網(wǎng)——每日最新資訊28at.com

類型擦除原理

在類型擦除過程中,Java 編譯器會擦除所有類型參數(shù),如果類型參數(shù)有界,則用其第一個邊界替換每個參數(shù);如果類型參數(shù)無界,則用 Object 替換。rc728資訊網(wǎng)——每日最新資訊28at.com

在類型擦除過程中,編譯器會按照以下規(guī)則來處理泛型類型參數(shù):rc728資訊網(wǎng)——每日最新資訊28at.com

如果類型參數(shù)有界(bounded type),即使用了extends關(guān)鍵字限定了類型的上界,例如<T extends Number>,則編譯器會用該類型的第一個邊界來替換類型參數(shù)。rc728資訊網(wǎng)——每日最新資訊28at.com

例如下面這個例子,泛型 T 繼承了Number類型,又實現(xiàn)了 Displayable 接口(沒錯,泛型可以這樣定義)rc728資訊網(wǎng)——每日最新資訊28at.com

interface Displayable {    void display();}public class Result<T extends Number & Displayable> {    private T value;    public Result(T value) {        this.value = value;    }    public T getValue() {        return value;    }     public void show() {        value.display();    }}

在編譯器進行類型擦除后會變成下面這樣,因為 T 的上限是 Number,所以直接將 T 替換為 Number。rc728資訊網(wǎng)——每日最新資訊28at.com

public class Result {    private Number value;    public Result(Number value) {        this.value = value;    }    public Number getValue() {        return value;    }}

如果類型參數(shù)無界(unbounded type),即沒有限定類型的上界,例如<T>,則編譯器會用Object類型來替換類型參數(shù)。rc728資訊網(wǎng)——每日最新資訊28at.com

例如下面方法,沒有指定類型上限類型。rc728資訊網(wǎng)——每日最新資訊28at.com

public static <T> int count(T[] anArray, T elem) {    int cnt = 0;    for (T e : anArray)        if (e.equals(elem))            ++cnt;        return cnt;}

經(jīng)過編譯器的擦除處理后,就變成下面這樣,都替換成了 Object。rc728資訊網(wǎng)——每日最新資訊28at.com

public static int count(Object[] anArray, Object elem) {    int cnt = 0;    for (Object e : anArray)        if (e.equals(elem))            ++cnt;        return cnt;}

橋接方法

來看一下下面這段代碼rc728資訊網(wǎng)——每日最新資訊28at.com

public class Node<T> {    public T data;    public Node(T data) { this.data = data; }    public void setData(T data) {        this.data = data;    }}public class SubNode extends Node<Integer> {    public SubNode(Integer data) { super(data); }    public void setData(Integer data) {        super.setData(data);    }     public static void main(String[] args) {        SubNode subNode = new SubNode(8);        Node node = subNode;        node.setData("Hello");        Integer x = subNode.data;    }}

這段代碼大家一看就知道肯定是有問題的,運行的時候會出現(xiàn) ClassCastException,但是編譯是可以通過的。rc728資訊網(wǎng)——每日最新資訊28at.com

而運行時出現(xiàn)錯誤的代碼是 node.setData("Hello");這一行,但是經(jīng)過前面對類型擦除的了解,Node 類的 setData 參數(shù)肯定被擦除成了 Object 類型了,既然是 Object,那Integer 和 String 都滿足啊,為啥還會報錯呢。rc728資訊網(wǎng)——每日最新資訊28at.com

這就要說到橋接了。rc728資訊網(wǎng)——每日最新資訊28at.com

當(dāng)編譯器對泛型擴展的類或接口進行編譯處理的時候,會根據(jù)實際的類型進行方法的橋接處理。什么意思呢,還是拿上面的 Node 和 SubNode 類說明。rc728資訊網(wǎng)——每日最新資訊28at.com

類型擦除后的代碼是下面這樣的,多了一個橋接方法。rc728資訊網(wǎng)——每日最新資訊28at.com

public class Node {    public Object data;    public Node(Object data) { this.data = data; }    public void setData(Object data) {        this.data = data;    }}public class SubNode extends Node {    public SubNode(Integer data) { super(data); } /** ** 橋接方法 **/ public void setData(Object data) {        setData((Integer) data);    }    public void setData(Integer data) {        super.setData(data);    }}

為什么需要這個橋接方法呢?rc728資訊網(wǎng)——每日最新資訊28at.com

Node 類的 setData 方法入?yún)⑹?Object 類型。rc728資訊網(wǎng)——每日最新資訊28at.com

public void setData(Object data) {    this.data = data;}

而 SubNode 的setData 方法入?yún)⑹?Integer。rc728資訊網(wǎng)——每日最新資訊28at.com

public void setData(Integer data) { super.setData(data);}

所以,SubNode 的 setData 方法并不會重寫父類 Node 的setData 方法,而想要重寫的話,就必須讓 SubNode 的setData 的入?yún)⒁彩?Object,這就是橋接方法的由來。rc728資訊網(wǎng)——每日最新資訊28at.com

public void setData(Object data) { setData((Integer) data);}

這樣一來重寫父類的方法,但是要把參數(shù)強轉(zhuǎn)成 Integer。rc728資訊網(wǎng)——每日最新資訊28at.com

前面說的 node.setData("Hello");這一行會報錯,那大家就知道為什么了吧,是因為把 Hello強轉(zhuǎn)為 Integer 的時候出現(xiàn)的錯誤。rc728資訊網(wǎng)——每日最新資訊28at.com

總結(jié)

正是類型擦除的機制幫助 Java 實現(xiàn)了泛型編程,讓我們作為開發(fā)者能夠更好的了解和控制我們正在使用類型的是什么,而不是 Object 滿天飛。rc728資訊網(wǎng)——每日最新資訊28at.com

本文鏈接:http://www.www897cc.com/showinfo-26-92736-0.htmlJava 泛型編程所說的類型擦除到底是什么?

聲明:本網(wǎng)頁內(nèi)容旨在傳播知識,若有侵權(quán)等問題請及時與本網(wǎng)聯(lián)系,我們將在第一時間刪除處理。郵件:2376512515@qq.com

上一篇: Rathole:基于Rust開發(fā)的輕量級高性能反向代理,替代Frp和Ngrok!

下一篇: K9s:終端中的 Kubernetes 集群管理

標(biāo)簽:
  • 熱門焦點
  • 掘力計劃第 20 期:Flutter 混合開發(fā)的混亂之治

    在掘力計劃系列活動第20場,《Flutter 開發(fā)實戰(zhàn)詳解》作者,掘金優(yōu)秀作者,Github GSY 系列目負(fù)責(zé)人戀貓的小郭分享了Flutter 混合開發(fā)的混亂之治。Flutter 基于自研的 Skia 引擎
  • Golang 中的 io 包詳解:組合接口

    io.ReadWriter// ReadWriter is the interface that groups the basic Read and Write methods.type ReadWriter interface { Reader Writer}是對Reader和Writer接口的組合,
  • 讓我們一起聊聊文件的操作

    文件【1】文件是什么?文件是保存數(shù)據(jù)的地方,是數(shù)據(jù)源的一種,比如大家經(jīng)常使用的word文檔、txt文件、excel文件、jpg文件...都是文件。文件最主要的作用就是保存數(shù)據(jù),它既可以保
  • 三言兩語說透柯里化和反柯里化

    JavaScript中的柯里化(Currying)和反柯里化(Uncurrying)是兩種很有用的技術(shù),可以幫助我們寫出更加優(yōu)雅、泛用的函數(shù)。本文將首先介紹柯里化和反柯里化的概念、實現(xiàn)原理和應(yīng)用
  • Python異步IO編程的進程/線程通信實現(xiàn)

    這篇文章再講3種方式,同時講4中進程間通信的方式一、 Python 中線程間通信的實現(xiàn)方式共享變量共享變量是多個線程可以共同訪問的變量。在Python中,可以使用threading模塊中的L
  • 零售大模型“干中學(xué)”,攀爬數(shù)字化珠峰

    文/侯煜編輯/cc來源/華爾街科技眼對于絕大多數(shù)登山愛好者而言,攀爬珠穆朗瑪峰可謂終極目標(biāo)。攀登珠峰的商業(yè)路線有兩條,一是尼泊爾境內(nèi)的南坡路線,一是中國境內(nèi)的北坡路線。相
  • 小紅書1周漲粉49W+,我總結(jié)了小白可以用的N條漲粉筆記

    作者:黃河懂運營一條性教育視頻,被54萬人&ldquo;珍藏&rdquo;是什么體驗?最近,情感博主@公主是用鮮花做的,火了!僅僅憑借一條視頻,光小紅書就有超過128萬人,為她瘋狂點贊!更瘋狂的是,這
  • 華為Mate 60系列用上可變靈動島:正式版體驗將會更出色

    這段時間以來,關(guān)于華為新旗艦的爆料日漸密集。據(jù)此前多方爆料,今年華為將開始恢復(fù)一年雙旗艦戰(zhàn)略,除上半年推出的P60系列外,往年下半年的Mate系列也將
  • 電博會與軟博會實現(xiàn)"線下+云端"的雙線融合

    在本次“電博會”與“軟博會”雙展會利好條件的加持下,既可以發(fā)揮展會拉動人流、信息流、資金流實現(xiàn)快速交互流動的作用,繼而推動區(qū)域經(jīng)濟良性發(fā)展;又可以聚
Top 主站蜘蛛池模板: 民勤县| 博罗县| 桂东县| 石柱| 民县| 西盟| 大埔县| 清流县| 玉门市| 荆门市| 武定县| 武平县| 灵川县| 浦东新区| 临猗县| 易门县| 元氏县| 收藏| 南皮县| 南宁市| 社旗县| 彝良县| 弥渡县| 会同县| 梁山县| 北宁市| 苍南县| 麻城市| 怀仁县| 旌德县| 江油市| 三明市| 恩平市| 汶川县| 恭城| 广汉市| 建平县| 深圳市| 师宗县| 平谷区| 若尔盖县|