范型参数不仅仅可以用于类型的声明上,例如
package com.tom.lang.generics; import java.util.List; public class Generics<T> { private T value; public Generics(T value) { this.value = value; } public void copy(T obj, List<T> list) { list.add(obj); } }
这上面的范型定义中,当指定了T的具体类型,那么属性和方法中由T指代的类型都会确定下来。除此之外,也可以仅仅为方法指定范型参数,而方法所在类不必指定为范型类型,
范型方法
例如
package com.tom.lang.generics; import java.util.List; public class Generics { public void paste() { //T in copy method can't be here } public <T> T copy(T obj, List<T> list) { list.add(obj); return obj; } }
这里,仅仅把copy方法定义为范型方法,paste方法和Generics类都不是范型的。
范型方法的实际使用场景
这种在实际的开发中经常用到,比如要把字符串转换为某个类型的对象,这个类型就可以定义为范型的,如:
package com.tom.lang.generics; import com.google.gson.Gson; public class JsonUtil { public static <T> T json2Java(String str, Class<T> clazz) { return new Gson().fromJson(str,clazz); } public static void main(String[] args){ Integer i = json2Java("1", Integer.class); // String str = json2Java("1", Integer.class); //编译错 } }
还有常见的例子是在使用Spring的时候,通过ApplicationContext.getBean()获取对象时,通过传入一个Class类型,可以获得相应的bean,而无需进行类型转换
ApplicationContext cxt = null; cxt.getBean("jdbcTemplate",JdbcTemplate.class);
getBean的参数签名是:
<T> T getBean(String name, Class<T> requiredType) throws BeansException;
可见getBean使用了定义范型方法的做法,同时参数Class<T> requiredType可接受的类型是A.class等Class实例
类型限定的范型方法
同类型限定的范型类一样,可以为范型方法进行类型限定,语法是一样的
1.限定类型的上限SuperType(T是SuperType类型或者T是SuperType的子类),使用extends关键字
package com.tom.lang.generics; import java.util.ArrayList; import java.util.Date; import java.util.List; public class Generics { public static <T extends Date>T dateCopy(T obj, List<T> list) { return null; } public static void main(String[] args) { List<java.util.Date> list = new ArrayList<java.uti.Date>(); dateCopy(new java.util.Date(), list); List<java.sql.Date> list2 = new ArrayList<java.sql.Date>(); dateCopy(new java.sql.Date(new Date().getTime()), list2); }
2. 限定类型的上限ChildType(T是ChildType类型或者T是ChildType的父类),使用super关键字
public static <T super Date>T dateCopy(T obj, List<T> list) { return null; }