JAVA 泛型的一些教程
泛型与反射 http://ray-yui.iteye.com/blog/1933127
面向对象篇:http://ray-yui.iteye.com/blog/1926984
数据类型篇:http://ray-yui.iteye.com/blog/1927251
字符串篇:http://ray-yui.iteye.com/blog/1927647
数组与集合(1):http://ray-yui.iteye.com/blog/1928170
数组与集合(2):http://ray-yui.iteye.com/blog/1930155
枚举与注解:http://ray-yui.iteye.com/blog/1931408
泛型与发射:http://ray-yui.iteye.com/blog/1933127
异常:http://ray-yui.iteye.com/blog/1938946
杂:http://ray-yui.iteye.com/blog/1942591
Java泛型拾遗 http://my.oschina.net/scipio/blog/298717
1、泛型编程
实际使用的类型在代码中只是以参数形式出现的占位符(称为形式类型参数),在具体实例化时,用实际类型替代其中的类型占位符(参数化类型),这种方式被称为泛型编程。
可以阻止向一个集合类的对象中添加不正确类型的对象,也可以提升抽象层度,减少不同类型的重复代码。
2、参数化类型
(1)不带通配符的类型
比如ObjectHolder<String>
(2)带通配符的类型
有界通配符:T extends Comparable<T> & Serializable,多个用&连接,没有指定上界时默认是Object,没有下界直说。
无界通配符:ObjectHolder<?>
3、形式参数类型
不能用来创建对象和数组、不能作为父类、不能使用instanceof表达式、不能使用其类型字面量、不能出现在异常处理中、不能出现在静态上下文中。
new T(),new T[],class MyClass extends T,instanceof T,T.class,catch(T),static T等这些都是不允许的。
4、类型擦除
泛型是在编译器这个层次实现的,在编译过程中会被擦除(包括泛型类型,泛型方法声明时的形式类型参数,以及实际类型参数),字节码不知道泛型,但为了反射API的需要,保留了相关信息,但这些信息在字节码执行时不被使用。
类型擦除后如下:
同一泛型类型的所有实例化形式在运行时的表示形式是相同的,比如List<String>和List<Integer>对虚拟机来说是相同的,都是List接口,无法通过List<String>.class形式来获取参数化类型的类对象的字面量,只能使用List.class,另外运行时不存在List<String>类型,只有List类型。
5、通配符
通配符的含义是一组类型的集合。用?表示,分有界和无界两种。
无界通配符用?表示,有界通配符如? extends/super
通配符一般做引用类型,比如返回Class<?>
6、数组允许协变
数组是由Java虚拟机根据类型创建出来的,如果一个数组的元素类型是另一个数组元素类型的子类型,则这个数组的类型也是对应数组类型的子类型。
比如String[]是Object[]的子类型。其中的原因是数组的元素类型信息在运行时仍然是被保留的。
变参通过数组传递:
7、类型系统
ArrayList<Number>是List<Number>的子类型
ArrayList<Number>与ArrayList<Integer>不存在父子类型关系
List<?>是所有List泛型类的实例化形式的父类型,比如List<String>,List<? extends Number>,List<? super Integer>等
? extends Integer 是 ? extends Number的子类型
List<? extends Integer>是List<? extends Number>的子类型
? super Number是? super Integer的子类型
List<? super Number>是List<? super Integer>的子类型
一个泛型的所有实例化形式是其对应的原始类型的子类型,比如List<String>和List<? extends Number>是List的子类型
8、覆写和重载
(1)覆写
包括方法类型签名(方法名称、参数类型)、返回值类型、异常抛出的类型。
A、要求方法参数类型是相同的或者父类类型擦除之后与子类参数类型是相同的。
B、子类型中的返回值类型必须可以替代父类型中对应方法的返回值类型。
C、子类型的方法声明中不能抛出父类型中对应方法没有声明的受检异常。
(2)重载
重载只考虑方法的类型签名,不考虑返回值类型和声明的受检异常。
9、类型推断和<>操作符
可以通过两种方式类判断所使用的实际类型:显示指定类型、编译器根据上下文信息进行推断
(1)显示指定类型
(2)类型推断
一般不需要显示指定类型,编译器可根据上下文进行推断
A、根据方法调用时的实际参数的静态类型进行推断(优先级较高)
B、当方法调用的结果被赋值给另外一个变量时,可以根据该变量的静态类型进行推断
10、泛型与反射API
JDK5对Java字节码的格式进行了修改,把泛型相关的信息添加到字节码中,不过字节码中的泛型相关内容只是提供相关信息,不会对字节码的运行造成影响,可以使用反射API获取。
全面总结Java泛型 http://www.cnblogs.com/rickie4you/p/3488551.html
本文对Java泛型进行了全面的总结。文章内容包括普通泛型、通配符、受限泛型、泛型接口、泛型方法、返回泛型类型实例等等。
虽然Scala创始人Martin Odersky说当年正是因为Java泛型的丑陋,所以才想到要创建一个新的语言,不过这仍然不妨碍我们学习Java泛型。毕竟即使听说Java泛型不好用,但好不好用还是得会用了才知道。下面是一些有关Java泛型的总结:
普通泛型
———————————————————-
通配符
受限泛型
———————————————————-
-
1234567891011121314151617181920212223242526Java代码class Info< T>{private T var ; // 定义泛型变量public void setVar(T var){this.var = var ;}public T getVar(){return this.var ;}public String toString(){ // 直接打印return this.var.toString() ;}};public class GenericsDemo21{public static void main(String args[]){Info< String> i1 = new Info< String>() ; // 声明String的泛型对象Info< Object> i2 = new Info< Object>() ; // 声明Object的泛型对象i1.setVar("hello") ;i2.setVar(new Object()) ;fun(i1) ;fun(i2) ;}public static void fun(Info< ? super String> temp){ // 只能接收String或Object类型的泛型System.out.print(temp + "、") ;}};
Java泛型无法向上转型
Java泛型接口
———————————————————-
-
1234567891011121314151617181920212223Java代码interface Info< T>{ // 在接口上定义泛型public T getVar() ; // 定义抽象方法,抽象方法的返回值就是泛型类型}class InfoImpl implements Info< String>{ // 定义泛型接口的子类private String var ; // 定义属性public InfoImpl(String var){ // 通过构造方法设置属性内容this.setVar(var) ;}public void setVar(String var){this.var = var ;}public String getVar(){return this.var ;}};public class GenericsDemo25{public static void main(String arsg[]){Info i = null; // 声明接口对象i = new InfoImpl("汤姆") ; // 通过子类实例化对象System.out.println("内容:" + i.getVar()) ;}};
Java泛型方法
通过泛型方法返回泛型类型实例
使用泛型统一传入的参数类型
Java泛型数组
Java泛型的嵌套设置
JAVA 泛型接口和泛型方法 http://wwwiteye.iteye.com/blog/1849917
接口:如果一个类或接口上有一个或多个类型变量,那它就是泛型。类型变量由尖括号界定,放在类或接口名的后面
方法:如果方法和构造器上声明了一个或多个类型变量,它们也可以泛型化。