1 public <T> Test<String,T> setCacheObject(String key,T value){ 2 return null; 3 }
- 前面的
T
的聲明,跟類后面的<T>
沒有關系。 - 方法前面的
<T>
是給這個方法級別指定泛型
直接上例子了
1 package com.system.net.generic; 2 3 class Fruit { 4 public String toString() { 5 return "Fruit"; 6 } 7 } 8 9 class Apple extends Fruit { 10 public String toString() { 11 return "Apple"; 12 } 13 } 14 15 class Person { 16 public String toString() { 17 return "Person"; 18 } 19 } 20 21 class ClassName<T> { 22 void show_1(T t) { 23 System.out.println("show_1 " + t.toString()); 24 } 25 26 <E> void show_2(E e) { 27 System.out.println("show_2 " + e.toString()); 28 } 29 30 <T> void show_3(T t) { 31 System.out.println("show_3 " + t.toString()); 32 } 33 34 public static void main(String[] args) { 35 ClassName<Fruit> o = new ClassName<Fruit>(); 36 Fruit f = new Fruit(); 37 Apple a = new Apple(); 38 Person p = new Person(); 39 System.out.println("show_1 演示________________________"); 40 o.show_1(f); 41 o.show_1(a); 42 // o.show_1( p ); 是不能編譯通過的。因為在 ClassName<Fruit>中已經限定了全局的T為Fruit,所以不能再加入Person; 43 System.out.println("show_2 演示________________________"); 44 o.show_2(f); 45 o.show_2(a); 46 o.<Person>show_2(p); 47 System.out.println("show_3 演示________________________"); 48 o.show_3(f); 49 o.show_3(a); 50 o.show_3(p); 51 } 52 }
-
show_2 和 show_3 方法其實是完完全全等效的。意思就是說ClassName中一旦
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產生沖突,你直接換成會更容易理解的。如例子中的:
1 o.<Person>show_2(p);