package com.test05.myTest; class Fruit { public String toString() { return "Fruit"; } } class Apple extends Fruit { public String toString(){ return "Apple"; } } class Person { public String toString(){ return "Person"; } } class ClassName<T> {//主類,文件名ClassName.java void show_1(T t){ System.out.println("show_1 "+ t.toString()); } <E> void show_2(E e){ System.out.println("show_2 "+e.toString()); } <T> void show_3(T t){ System.out.println("show_3 "+t.toString()); } public static void main(String[] args) { ClassName<Fruit> o = new ClassName<Fruit>(); Fruit f = new Fruit(); Apple a = new Apple(); Person p = new Person(); System.out.println("show_1 演示________________________"); o.show_1( f ); o.show_1( a ); // o.show_1( p ); 樓主把這行代碼去掉注釋看一下,是不能編譯通過的。因為在 // ClassName<Fruit>中已經限定了全局的T為Fruit,所以不能再加入Person; System.out.println("show_2 演示________________________"); o.show_2( f ); o.show_2( a ); o.show_2( p ); System.out.println("show_3 演示________________________"); o.show_3( f ); o.show_3( a ); o.show_3( p ); } }
輸出:
show_1 演示________________________
show_1 Fruit
show_1 Apple
show_2 演示________________________
show_2 Fruit
show_2 Apple
show_2 Person
show_3 演示________________________
show_3 Fruit
show_3 Apple
show_3 Person
/* 而show_2 和show_3方法其實是完完全全等效的。意思就是說ClassName<T>中一旦
T被指定為Fruit后那么show_1沒有前綴<T> 的話,該方法中只能是show_1 (Fruit對象)
而你要是有前綴<T>或<E>的話,那么你就是告訴編譯器對它說:這是我新指定的一個類型,
跟ClassName<T>類對象中的T沒有半毛錢的關系。也就是說這個show_3中的T和show_2中的
E是一個效果,也就是你可以把show_3同等程度地理解為<E> void show_3(E e){~~~~~}
從上面我說的看,那就是 這個方法返回值前也加個<T>的話,這個T就代表該方法自己獨有的某個類,而不去和類中限定的T產生沖突,你直接換成<E>會更容易理解的。*/
<T> 你可以理解為一個類型的聲明,否則你的返回值和函數參數中突然出現了一個"T",編譯器知道這是什么東西,肯定會報錯,所以要從編譯器的角度來理解這個問題.
來自:https://www.cnblogs.com/winkey4986/p/3158760.html