自拍偷在线精品自拍偷,亚洲欧美中文日韩v在线观看不卡

看完后讓你成為武松,手把手教你打死Java中的紙老虎

開(kāi)發(fā) 后端
泛型,其實(shí)算是Java當(dāng)中比較難的語(yǔ)法了,很多人一開(kāi)始都對(duì)其一知半解,也很害怕閱讀帶泛型的源碼,雖然看起來(lái)語(yǔ)法很難,但當(dāng)你理解后會(huì)覺(jué)得很簡(jiǎn)單,其實(shí)只是一個(gè)紙老虎罷了。

泛型,其實(shí)算是Java當(dāng)中比較難的語(yǔ)法了,很多人一開(kāi)始都對(duì)其一知半解,也很害怕閱讀帶泛型的源碼,雖然看起來(lái)語(yǔ)法很難,但當(dāng)你理解后會(huì)覺(jué)得很簡(jiǎn)單,其實(shí)只是一個(gè)紙老虎罷了。下面,我將會(huì)用非常簡(jiǎn)單易懂的方式帶你去理解它,相信你在認(rèn)真看完后會(huì)有非常大的收獲,從此不會(huì)再畏懼它!

[[422931]]

一. 泛型的定義

這里大家可以不必去看網(wǎng)上的有些定義,因?yàn)橄鄬?duì)于比較學(xué)術(shù)化,只需記住泛型可以在程序設(shè)計(jì)中指定某種類(lèi)型,讓程序的設(shè)計(jì)更加規(guī)范化即可

二. 為什么要用到泛型

了解到了泛型是什么后,那我們來(lái)討論討論為什么要用泛型這個(gè)語(yǔ)法,這個(gè)語(yǔ)法到底是干什么的?別急,下面,我先給大家舉一個(gè)例子: 

  1. class Stack { 
  2.     public Object[] objects; 
  3.     public int top
  4.  
  5.     public Stack() { 
  6.         this.objects =new Object[10]; 
  7.     } 
  8.  
  9.     public void push(Object obj) { 
  10.         objects[this.top++] = obj; 
  11.     } 
  12.  
  13.     public Object get() { 
  14.         return objects[this.top-1]; 
  15.     } 

大家可以看看這是在干什么呢?這是我們自己寫(xiě)了一個(gè)棧,然后將棧里的數(shù)組類(lèi)型設(shè)置成Object類(lèi)型,這樣的話(huà)這個(gè)棧里任意類(lèi)型的數(shù)據(jù)都可以存放了(Object類(lèi)是任何類(lèi)的父類(lèi),不管插入什么類(lèi)型的數(shù)據(jù),都可以發(fā)生向上轉(zhuǎn)型)

下面,我們來(lái)測(cè)試一下: 

  1. public class Test { 
  2.     public static void main(String[] args) { 
  3.         Stack stack=new Stack(); 
  4.         stack.push(1); 
  5.         stack.push(2); 
  6.         stack.push("123"); 
  7.         String str=(String)stack.get(); 
  8.     } 

可以看到,我們可以向自己寫(xiě)的棧里放入整形以及字符串等等任何類(lèi)型的數(shù)據(jù),但注意一下取出數(shù)據(jù)的時(shí)候要進(jìn)行強(qiáng)制類(lèi)型轉(zhuǎn)換

以上這樣寫(xiě),可以向棧里存放任何類(lèi)型的數(shù)據(jù),比較通用,其優(yōu)點(diǎn)也可以變成缺點(diǎn),正因?yàn)樘ㄓ昧耍勾a的規(guī)范性降低,看起來(lái)比較凌亂,這時(shí)候,我們可以考慮使用泛型,這樣可以在類(lèi)中或者Java集合中存放特定的數(shù)據(jù)(使用Java集合時(shí),一般都要用到泛型,而自定義的類(lèi)型中可以使用泛型也可以不使用)

三. 泛型的寫(xiě)法

以自定義的類(lèi)型為例,寫(xiě)法為在類(lèi)名后面加上尖括號(hào),里面寫(xiě)上一個(gè)字母(注意,此處寫(xiě)任何字母都可以,只起到一個(gè)標(biāo)記這個(gè)類(lèi)為泛型類(lèi)的作用)

  1. class Stack 

而在new對(duì)象時(shí),以棧里只能存放整形為例,前面的尖括號(hào)必須寫(xiě)基本數(shù)據(jù)類(lèi)型對(duì)應(yīng)的包裝類(lèi),而后面的尖括號(hào)可以不用寫(xiě),示例如下:

  1. Stack stack = new Stack<>(); 

補(bǔ)一下Java中的基本數(shù)據(jù)類(lèi)型與對(duì)應(yīng)的包裝類(lèi):

因此,我們前面寫(xiě)的自定義的??梢詫?xiě)成以下形式(以存放整形為例): 

  1. class Stack<T> { 
  2.     public T[] objects; 
  3.     public int top
  4.  
  5.     public Stack() { 
  6.         this.objects = (T[])new Object[10]; 
  7.     } 
  8.  
  9.     public void push(T obj) { 
  10.         objects[this.top++] = obj; 
  11.     } 
  12.  
  13.     public T get() { 
  14.         return objects[this.top-1]; 
  15.     } 
  16.  Stack<Integer> stack = new Stack<>(); 
  17.         stack.push(1); 
  18.         stack.push(2); 
  19.         int ret = stack.get(); 
  20.         System.out.println(ret); 

特別注意此處:public Stack() { this.objects = (T[])new Object[10]; }

這里不能寫(xiě)成this.objects=new T[10];

原因:

  1. 不能new泛型類(lèi)型的數(shù)組
  2. 也可理解為泛型是先檢查后編譯的,如果new泛型類(lèi)型的數(shù)組的話(huà),編譯器檢查時(shí)并不知道T是什么類(lèi)型的,因此會(huì)報(bào)錯(cuò)。而編譯的時(shí)候才會(huì)進(jìn)行擦除機(jī)制,都會(huì)將其轉(zhuǎn)換為Object類(lèi)型
  3. 正因?yàn)橛羞@個(gè)擦除機(jī)制,這里才能進(jìn)行數(shù)組整體強(qiáng)制類(lèi)型轉(zhuǎn)換(一般數(shù)組不能整體進(jìn)行強(qiáng)制類(lèi)型轉(zhuǎn)換),因?yàn)榉盒椭皇窃诰幾g的時(shí)候起作用,而實(shí)際運(yùn)行時(shí)都會(huì)被擦除成Object類(lèi)型,即實(shí)際運(yùn)行時(shí)是沒(méi)有泛型這個(gè)概念的,也即實(shí)際運(yùn)行時(shí)類(lèi)型都是一樣的,所以T本質(zhì)上是object類(lèi)型的,所以此代碼等價(jià)于不進(jìn)行強(qiáng)制類(lèi)型轉(zhuǎn)換!!!
  4. 而直接指定泛型的代碼(不是T) 比如:Stack和Stack都是在運(yùn)行時(shí)直接把尖括號(hào)里的類(lèi)型擦掉了,可以看到直接打印的結(jié)果(并沒(méi)有打印出類(lèi)型):

此處注意多理解理解

四. 泛型的使用實(shí)例

1. 求最大值

以上就是泛型的一個(gè)重要知識(shí)點(diǎn)了,但光看是不夠的,還是得通過(guò)例子讓大家有一個(gè)更為深入的理解,比如,如何寫(xiě)一個(gè)泛型類(lèi)來(lái)求數(shù)組的最大值呢?

基本的框架大概是這樣的:(沒(méi)看懂的小可愛(ài)好好看看上面講的內(nèi)容哦) 

  1. class Algorithm<T extends Comparable<T>> { 
  2.     public T findMax(T[] array) { 
  3.         T max = array[0]; 
  4.         for (int i = 1; i < array.length; i++) { 
  5.             if(max < array[i]) { 
  6.                 max = array[i]; 
  7.             } 
  8.         } 
  9.         return max
  10.     } 

但是此代碼if(max < array[i])會(huì)報(bào)錯(cuò),為什么呢?因?yàn)閷?lái)給T傳的值一定是一個(gè)引用類(lèi)型,引用類(lèi)型不能直接比較大于或者小于的,是要用Comparable或Comparator接口里的方法比較的,因?yàn)榉盒驮诰幾g的時(shí)候會(huì)被擦除成Object類(lèi)型,但Object類(lèi)本身并沒(méi)有實(shí)現(xiàn)Comparable或Comparator接口,所以我們要控制其不要擦除到Object類(lèi),所以要給泛型指定一個(gè)邊界

具體寫(xiě)法如下: 

  1. class Algorithm<T extends Comparable<T>> { 
  2.     public T findMax(T[] array) { 
  3.         T max = array[0]; 
  4.         for (int i = 1; i < array.length; i++) { 
  5.             //max < array[i] 
  6.             if(max.compareTo(array[i]) < 0) { 
  7.                 max = array[i]; 
  8.             } 
  9.         } 
  10.         return max
  11.     } 
  12.  
  13. class Algorithm<T extends Comparable<T>> 

注意,extends叫做上界,此代碼代表的意思為T(mén)這個(gè)泛型類(lèi)會(huì)擦除到實(shí)現(xiàn)了Comparable接口的地方,換句話(huà)說(shuō),這個(gè)T類(lèi)型一定是實(shí)現(xiàn)了Comparable接口的

同理:這個(gè)代碼public class MyArrayList { ... }代表E為Number的子類(lèi)或Number本身

下面讓我們來(lái)用一下: 

  1. Algorithm<Integer> algorithm1 = new Algorithm<>(); 
  2.        Integer[] integers = {1,2,13,4,5}; 
  3.        Integer ret = algorithm1.findMax(integers); 
  4.        System.out.println(ret); 

運(yùn)行結(jié)果如下:

[[422934]]

成功了!

2. 優(yōu)化

經(jīng)過(guò)上面的努力,我們已經(jīng)寫(xiě)出了一個(gè)泛型類(lèi)來(lái)求一個(gè)數(shù)組的最大值了,但是,上面的例子是一個(gè)整形數(shù)組,那么我們能不能在數(shù)組里存放別的類(lèi)型去比較呢?答案是可以的,但是我們還得去new一個(gè)對(duì)象,例如:Algorithm algorithm2 = new Algorithm<>();這樣很麻煩。但是我們可以將求最大值的方法設(shè)置成靜態(tài)的class Algorithm2 ,因?yàn)槭庆o態(tài)的方法,不需要new對(duì)象,所以就沒(méi)有在new對(duì)象時(shí)指定泛型的過(guò)程了,所以沒(méi)必要給方法后加尖括號(hào),但是去掉之后,代碼又會(huì)被錯(cuò):

我們可以這樣修改: 

  1. class Algorithm2 { 
  2.     public static<T extends Comparable<T>> T findMax(T[] array) { 
  3.         T max = array[0]; 
  4.         for (int i = 1; i < array.length; i++) { 
  5.             if(max.compareTo(array[i]) < 0) { 
  6.                 max = array[i]; 
  7.             } 
  8.         } 
  9.         return max
  10.     } 

此方法public static> T findMax(T[] array){}叫做泛型方法

下面繼續(xù)帶大家來(lái)用一下: 

  1. public static void main(String[] args) { 
  2.         Integer[] integers = {1,2,13,4,5}; 
  3.         //會(huì)根據(jù)形參的類(lèi)型推導(dǎo)出整個(gè)泛型的類(lèi)型參數(shù) 
  4.         Integer ret = Algorithm2.findMax(integers); 
  5.         System.out.println(ret); 
  6.         Integer ret2 = Algorithm2.<Integer>findMax(integers); 
  7.         System.out.println(ret2); 
  8.     } 

注意,ret1寫(xiě)法和ret2寫(xiě)法是一樣的,都可以

打印結(jié)果如下:

五. 通配符

1. 基本寫(xiě)法

通配符也是泛型的一種,下面我們來(lái)寫(xiě)一個(gè)泛型方法來(lái)打印集合中的元素。 

  1. class Test { 
  2.  
  3.     public static<T> void print(ArrayList<T> list) { 
  4.  
  5.         for (T t : list) { 
  6.             System.out.println(t); 
  7.         } 
  8.     } 

這個(gè)寫(xiě)法很簡(jiǎn)單,上文都講過(guò)了,那么讓我們來(lái)試著用一下吧: 

  1. public static void main(String[] args) { 
  2.         ArrayList<Integer> list = new ArrayList<>(); 
  3.         list.add(1); 
  4.         list.add(2); 
  5.         list.add(3); 
  6.         Test.print(list); 
  7.     } 

打印的結(jié)果如下:

除了以上這種寫(xiě)法,我們還可以將其改成通配符的寫(xiě)法,先給大家上代碼: 

  1. //?代表通配符  擦除機(jī)制  Object 
  2.     public static void print2(ArrayList<?> list) { 
  3.         for (Object t : list) { 
  4.             System.out.println(t); 
  5.         } 
  6.     } 

此處for (Object t : list)必須這樣寫(xiě),因?yàn)橥ㄅ浞彩怯胁脸龣C(jī)制的,會(huì)在編譯器編程O(píng)bject類(lèi)型。

2. 上界

  • 語(yǔ)法:
  • 示例: 
  1. public static void printAll(MyArrayList list) {  
  2. ...  
  3.  
  4. xxxxxxxxxxbr public static void printAll(MyArrayList list) {br...br } 

代表可以傳入類(lèi)型實(shí)參是 Number 子類(lèi)的任意類(lèi)型的 MyArrayList所以以下調(diào)用都是正確的: 

  1. printAll(new MyArrayList());  
  2. printAll(new MyArrayList());  
  3. printAll(new MyArrayList()); 
  4.  
  5. xxxxxxxxxxbr printAll(new MyArrayList());brprintAll(new MyArrayList());brprintAll(new MyArrayList()); 

以下調(diào)用都是錯(cuò)誤的: 

  1. printAll(new MyArrayList());  
  2. printAll(new MyArrayList ()); 
  3.  
  4. xxxxxxxxxxbr printAll(new MyArrayList());brprintAll(new MyArrayList()); 

3. 下界

下界和上界的用法很類(lèi)似

  • 語(yǔ)法:
  • 示例: 
  1. public static void printAll(MyArrayList list) {  
  2. ...  
  3.  
  4. xxxxxxxxxxbr public static void printAll(MyArrayList list) {br...br} 

代表可以傳入類(lèi)型實(shí)參是 Integer 父類(lèi)的任意類(lèi)型的 MyArrayList所以以下調(diào)用是正確的: 

  1. printAll(new MyArrayList());  
  2. printAll(new MyArrayList());  
  3. printAll(new MyArrayList()); 
  4.  
  5. xxxxxxxxxxbr printAll(new MyArrayList());brprintAll(new MyArrayList());brprintAll(new MyArrayList()); 

以下調(diào)用是錯(cuò)誤的: 

  1. printAll(new MyArrayList());  
  2. printAll(new MyArrayList()); 
  3.  
  4. xxxxxxxxxxbr printAll(new MyArrayList());brprintAll(new MyArrayList()); 

六. 泛型的限制 

學(xué)習(xí)完后,我們應(yīng)該注意泛型使用過(guò)程中以下一些限制:

  • 泛型類(lèi)型參數(shù)不支持基本數(shù)據(jù)類(lèi)型

  • 無(wú)法實(shí)例化泛型類(lèi)型的對(duì)象

  • 無(wú)法使用泛型類(lèi)型聲明靜態(tài)的屬性

  • 無(wú)法使用 instanceof 判斷帶類(lèi)型參數(shù)的泛型類(lèi)型(因?yàn)楸徊脸龣C(jī)制擦除了)

  • 無(wú)法創(chuàng)建泛型類(lèi)數(shù)組

  • 無(wú)法 create、catch、throw 一個(gè)泛型類(lèi)異常(異常不支持泛型)

  • 泛型類(lèi)型不是形參一部分,無(wú)法重載

好啦,本次泛型知識(shí)點(diǎn)的分享就先告一段落了,整理不易,但如果能幫到大家很開(kāi)心了。也希望大家多理解理解,不論是剛開(kāi)始學(xué)習(xí)還是復(fù)習(xí),都值得仔細(xì)揣摩哦!一起加油吧!

 

 

責(zé)任編輯:未麗燕 來(lái)源: 今日頭條
相關(guān)推薦

2022-10-30 10:31:42

i2ccpuftrace

2021-07-14 09:00:00

JavaFX開(kāi)發(fā)應(yīng)用

2011-01-10 14:41:26

2011-05-03 15:59:00

黑盒打印機(jī)

2009-06-15 16:58:57

Java安裝Linux

2022-11-06 14:46:28

腳本windows文件

2021-01-20 10:18:48

MySQL服務(wù)器配置

2021-09-26 16:08:23

CC++clang_forma

2011-04-28 09:23:36

REST

2022-01-08 20:04:20

攔截系統(tǒng)調(diào)用

2022-03-14 14:47:21

HarmonyOS操作系統(tǒng)鴻蒙

2023-04-26 12:46:43

DockerSpringKubernetes

2022-12-07 08:42:35

2022-07-27 08:16:22

搜索引擎Lucene

2011-02-22 13:46:27

微軟SQL.NET

2021-12-28 08:38:26

Linux 中斷喚醒系統(tǒng)Linux 系統(tǒng)

2021-02-26 11:54:38

MyBatis 插件接口

2009-06-02 15:38:36

eclipse streclipse開(kāi)發(fā)steclipse str

2022-11-28 08:23:14

IDEAGradle配置

2021-11-12 14:50:32

Kubectl命令Linux
點(diǎn)贊
收藏

51CTO技術(shù)棧公眾號(hào)