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

Java generic中通配符的幾點(diǎn)理解

開發(fā) 后端
結(jié)合Java本身的一些面向?qū)ο蟮奶匦?,我們很容易理解這么一個(gè)置換原則。但是在泛型的編程中,我們考慮到子類型關(guān)系的時(shí)候,更是容易把一種關(guān)系給弄混淆,并錯(cuò)誤的采用置換原則。

置換原則

結(jié)合Java本身的一些面向?qū)ο蟮奶匦?,我們很容易理解這么一個(gè)置換原則:

一個(gè)指定類型的變量可以被賦值為該類型的任何子類;一個(gè)指定某種類型參數(shù)的方法可以通過傳入該類型的子類來進(jìn)行調(diào)用。

總的來說,就是說我們使用的任何類型變量都可以用該類型的子類型來替換。

泛型中一種錯(cuò)誤的繼承關(guān)系

在泛型的編程中,我們考慮到子類型關(guān)系的時(shí)候,容易把一種關(guān)系給弄混淆,并錯(cuò)誤的采用置換原則。

比如說:

  1. List<Integer> ints = new ArrayList<Integer>();  
  2. ints.add(1);  
  3. ints.add(2);  
  4. List<Number> nums = ints; // compile error 

在這段代碼中,我們看到類型參數(shù)Integer是Number的子類型,就容易想當(dāng)然的認(rèn)為L(zhǎng)ist<Integer>也是List<Number>的子類。實(shí)際上并不是。所以才會(huì)導(dǎo)致類型不匹配,產(chǎn)生編譯時(shí)錯(cuò)誤。

有點(diǎn)時(shí)候,我們覺得,這樣的轉(zhuǎn)換看似不能用到一個(gè)好處,就是利用對(duì)象之間繼承的關(guān)系。要是我們能有一個(gè)列表,它既能處理某種類型的數(shù)據(jù),還能處理該類型的所有子類型的數(shù)據(jù),這樣豈不是既能用到泛型的好處又可以用到對(duì)象關(guān)系的好處么?于是在這里就引出了通配符(wildcard)。

通配符(Wildcard)

在Java類庫(kù)中Collection接口定義中有一個(gè)用到通配符的方法:

  1. interface Collection<E> {  
  2. ...  
  3. public boolean addAll(Collection<? extends E> c);  
  4. ...  

在addAll方法的描述里,可以接受Collection類型的參數(shù)。其中Collection中的類型參數(shù)可以為任何繼承E的子類型。

因此,我們可以在實(shí)際代碼中這么使用:

  1. List<Number> nums = new ArrayList<Number>();  
  2. List<Integer> ints = Arrays.asList(12);  
  3. List<Double> dbls = Arrays.asList(2.783.14);  
  4. nums.addAll(ints);  
  5. nums.addAll(dbls); 

在代碼中我們可以看到,List<Integer>和List<Double>都是Collection<? extends Number>類型的子類。所以上面的方法中可以將Integer和Double兩種類型的List傳入到方法中。

通配符使用限制1:

使用通配符的泛型數(shù)據(jù)類型比較有意思,既然前面我們可以將其作為方法聲明的參數(shù),那么是否可以將它作為一個(gè)變量類型來直接創(chuàng)建變量呢?

看如下代碼:

  1. List<? extends Number> nums = new ArrayList<Integer>(); //compile error 

實(shí)際上上面這段代碼是編譯通不過的。

通配符使用限制2:

既然不能用來直接創(chuàng)建變量對(duì)象,那么再看下面這段代碼:

  1. List<Integer> ints = new ArrayList<Integer>();  
  2. ints.add(1);  
  3. ints.add(2);  
  4. List<? extends Number> nums = ints;  
  5. nums.add(3.14); // compile error 

這段代碼的第5行會(huì)導(dǎo)致編譯錯(cuò)誤。在第4行代碼中,我們將ints賦值給nums,表面上nums聲明為一個(gè)List<Integer>的父類型,所以第4行編譯正常。為什么第5行代碼會(huì)出錯(cuò)呢?表面上看來,既然nums類型可以接受繼承自Number的所有參數(shù),那加一個(gè)Double類型的數(shù)據(jù)應(yīng)該是沒問題的。實(shí)際上我們?cè)倏紤]一下這樣會(huì)帶來的問題:

nums本來引用的是一個(gè)繼承自該類型的List<Integer>,如果我們?cè)试S加入Double類型的數(shù)據(jù)的話,那么ints這個(gè)Integer的List里面就包含了Double的數(shù)據(jù),當(dāng)我們使用ints的時(shí)候,和我們所期望的只包含Integer類型的數(shù)據(jù)不符合。

 

因此,這段代碼也說明了一個(gè)問題,就是在? extends E這種通配符引用的數(shù)據(jù)類型中,如果向其中增加數(shù)據(jù)操作的話會(huì)有問題。所以向其中增加數(shù)據(jù)是不允許的。但是我們可以從其中來讀取數(shù)據(jù)。

總結(jié):

1:通配符修飾的泛型不能用來直接創(chuàng)建變量對(duì)象。

2:通配符修飾相當(dāng)于聲明了一種變量,它可以作為參數(shù)在方法中傳遞。這么做帶來的好處就是我們可以將應(yīng)用于包含某些數(shù)據(jù)類型的列表的方法也應(yīng)用到包含其子類型的列表中。相當(dāng)于可以在列表中用到一些面向?qū)ο蟮奶匦浴?/p>

原文鏈接:http://shmilyaw-hotmail-com.iteye.com/blog/1405904

【編輯推薦】

  1. Java中泛型創(chuàng)建數(shù)組的總結(jié)
  2. Java Socket超時(shí)淺析
  3. Java Socket重要參數(shù)講解
  4. 對(duì)于Java類加載過程中的順序問題探究
  5. 深入理解Java對(duì)象序列化
責(zé)任編輯:林師授 來源: frank-liu的博客
相關(guān)推薦

2012-03-21 10:30:58

Java

2023-03-04 21:05:53

JAVA泛型通配符

2009-06-22 13:46:00

java程序員考試

2015-11-02 17:20:00

Java弱引用

2021-02-25 10:20:26

Java接口代碼

2018-07-09 15:11:14

Java逃逸JVM

2009-04-10 09:43:00

Java輸出流異常

2017-09-23 15:28:32

JavaOptional方法

2009-07-20 15:39:32

ASP.NET緩存

2011-09-28 11:14:35

Zshellzshbash shell

2010-12-07 11:22:02

提薪

2009-03-17 16:22:13

Java泛型接口

2014-11-11 15:25:30

PHPWeb

2009-06-29 18:26:46

HibernateHQL查詢

2010-06-01 13:32:15

Visual Stud

2021-06-18 08:25:42

Java泛型通配符

2018-03-07 12:50:59

Apache優(yōu)化建議

2012-03-07 09:02:29

代碼復(fù)用

2009-08-13 09:01:00

ASP.NET開發(fā)Web標(biāo)準(zhǔn)

2012-12-25 09:38:41

JavaScript設(shè)計(jì)模式
點(diǎn)贊
收藏

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