11.1 为什么要有泛型
-
标签,中药店, 每个抽屉外面贴着标签
-
ArrayList.add( Object类的对象 ),所有的类都能往里放:缺点
-
所谓泛型, 就是允许在定义类、 接口时通过一个标识表示类中某个属性的类型或者是某个方法的返回值及参数类型。 这个类型参数将在使用时(例如,继承或实现这个接口, 用这个类型声明变量、 创建对象时) 确定(即传入实际的类型参数, 也称为类型实参)
-
从JDK1.5以后, Java引入了“参数化类型( Parameterized type) ” 的概念,允许我们在创建集合时再指定集合元素的类型, 如:
List<String>
, 这表明该List只能保存字符串类型的对象 -
没有泛型存在的问题:
List list = new ArrayList(); //需求存放学生编号 list.add(1); list.add(2); list.add(3); //问题一:类型不安全 list.add("String"); for (Object obj:list ) { //问题二:强转时可能出现ClassCastException int stuID = (int) obj; System.out.println(stuID); }
-
引入泛型:在集合中使用泛型
-
ArrayList为例:
ArrayList<Integer> integers = new ArrayList<Integer>(); //需求存放学生编号 integers.add(1); integers.add(2); integers.add(3); //integers.add("String"); //1.编译时就会进行类型检查: //Required type:Integer //Provided:String //foreach遍历: for (Integer obj:integers ) { //2.避免了强转操作 int stuID = obj; System.out.println(stuID); } //iterator遍历: Iterator<Integer> iterator = integers.iterator(); while(iterator.hasNext()){ System.out.println(iterator.next()); }
-
HashMap为例:
HashMap<Integer, Integer> hashMap = new HashMap<>(); hashMap.put(1,2); hashMap.put(3,4); //取到Entry Set<Map.Entry<Integer, Integer>> set = hashMap.entrySet(); Iterator<Map.Entry<Integer, Integer>> iterator = set.iterator(); while (iterator.hasNext()){ Map.Entry<Integer, Integer> next = iterator.next(); System.out.println(next.getKey()+":"+next.getValue()); }
-
总结:
-
集合接口或集合类在jdk5时,都可以修改为带泛型的结构
-
在实例化集合类时,可以指明具体的泛型类型
-
在集合或接口中凡是定义类或接口时,内部结构使用到的类的泛型位置都指定为实例化时指定的泛型类型(将T替换成实例化时指定的类型)
比如:
add(E e)
→ \to →add(Integer e)
-
注意泛型的类型必须是类:不能使用基本数据类型,需要用包装类替换
-
如果实例化时,没有指明泛型的类型,默认类型为java.lang.util类型
List list = new ArrayList();
-
-
11.2 自定义泛型结构
-
如何自定义泛型结构:泛型类、泛型接口;泛型方法
-
自定义泛型类泛型接口:
-
泛型类的定义和使用:
public class Order<T> { String orderName; int orderID; T orderT;//类中所有的T都会被,实例化时指定的泛型类型替换 public Order(String orderName, int orderID, T orderT) { this.orderName = orderName; this.orderID = orderID; this.orderT = orderT; } public Order() { } public T getOrderT() { return orderT; } public void setOrderT(T orderT) { this.orderT = orderT; } } public class Test { public static void main(String[] args) { //如果定义了泛型类,但实例化时没有指明泛型,则默认Object Order order = new Order(); order.setOrderT("abc"); order.setOrderT(123); //建议实例化时,添加泛型 Order<String> objectOrder = new Order<String>("ab",001,"order:AA"); //这里要求参数列表中的orderT为String类型 //objectOrder.setOrderT(123);//Required type:String Provided:int objectOrder.setOrderT("order:aaa"); } }
-
继承时的泛型类:情况一
class subOrder extends Order<String>{ //subOrder不再是一个泛型类了,父类给他已经确定了 } public class Test { public static void main(String[] args) { subOrder subOrder = new subOrder(); //subOrder.setOrderT(123);//Required type:String Provided:int subOrder.setOrderT("ccc"); //由于子类在继承带泛型的父类时,指明了泛型类型。则实例化子类对象时,不再需要指明泛型 } }
-
继承时的泛型类:情况二
class subOrder2<T> extends Order<T>{ //subOrder仍然是一个泛型类 } public class Test { public static void main(String[] args) { subOrder2<String> subOrder2 = new subOrder2<>(); } }
-
注意:
-
泛型类可能有多个参数,此时应将多个参数一起放在尖括号内。如<E1,E2,E3>
-
泛型类的构造器如下:
public GenericClass(){}
错误的:
public GenericClass<E>(){}
-
实例化后,操作原来泛型位置的结构必须与指定的泛型类型一致。
-
泛型不同的引用不能相互赋值。
尽管在编译时
ArrayList<String>
和ArrayList<Integer>
是两种类型,但在运行时只有一个ArrayList被加载到JVM中 -
泛型如果不指定,将被擦除,泛型对应的类型均按照Object处理,但不等价于Object。 经验: 泛型要使用一路都用。要不用,一路都不要用
-
如果泛型结构是一个接口或抽象类,则不可创建泛型类的对象
-
jdk1.7,泛型的简化操作: ArrayList flist = new ArrayList<>()
-
泛型的指定中不能使用基本数据类型,可以使用包装类替换
-
在类/接口上声明的泛型,在本类或本接口中即代表某种类型,可以作为非静态属性的类型、非静态方法的参数类型、非静态方法的返回值类型。但在静态方法中不能使用类的泛型(静态结构的创建早于对象的创建,而泛型类型是在实例化是才被指定)
-
异常类不能是泛型的
-
如果想要new一个T类型的数组:不能使用new T[10]
但是可以:
T[ ] elements = (T])new Object[10];
T实际还是一个变量 -
父类有泛型,子类可以选择保留泛型也可以选择指定泛型类型、还可以增加自己的泛型
class Father<T1, T2> {}
-
子类不保留父类的泛型:
-
没有类型擦除,或者重新定义
class Son1 extends Father {} class Son<A, B> extends Father{} //定义自己的A、B
-
具体类型
//只有父类指定的具体的泛型类型 class Son2 extends Father<Integer, String> {} //不仅有父类指定的具体的泛型类型,还有子类新添加的 class Son2<A, B> extends Father<Integer, String> {}
-
-
子类保留父类的泛型:泛型子类
-
全部保留
class Son3<T1, T2> extends Father<T1, T2> {} class Son3<T1, T2, A, B> extends Father<T1, T2> {}
-
部分保留
class Son4<T2> extends Father<Integer, T2> {} class Son4<T2, A, B> extends Father<Integer, T2> {}
-
-
-
-
-
自定义泛型方法:
-
方法也可以被泛型化,不管此时定义在其中的类是不是泛型类
-
泛型方法中可以定义泛型参数,此时参数的类型就是传入数据的类型,且于类的泛型参数没有任何关系
-
泛型方法的格式:
[访问权限] <泛型> 返回类型 方法名([泛型标识 参数名称]) 抛出的异常
-
示例:
public class Order<T> { String orderName; int orderID; T orderT; //定义泛型方法 public <E> List<E> arrayToArrayList(E[] arr){ List<E> eList = new ArrayList<>(); for (E e:arr ) { eList.add(e); } return eList; } } public class Test { public static void main(String[] args) { Order<Object> order1 = new Order<>(); Integer[] arr = {1, 2, 3}; List<Integer> list = order1.arrayToArrayList(arr); System.out.println(list); } }
-
泛型方法可以声明为静态的
原因:泛型参数是在调用方法时确定的,并非在实例化类时确定
public static <E> List<E> arrayToArrayList(E[] arr){}
-
-
泛型类的使用情形:
//DAO : data access object public class DAO<T>{ //增 public void add(T t){ } //删 public void remove(int index){ } //改 public void update(int index,T t){ } } public class student {} public class studentDAO extends DAO<student>{} public class Test { public static void main(String[] args) { studentDAO studentDAO = new studentDAO(); studentDAO.add(new student()); studentDAO.remove(2); studentDAO.update(1,new student()); } }
之前写的web底层代码:
如果使用泛型的话可以抽取出表的共性操作的DAO
-
泛型方法的使用情形:
public class DAO<T>{ //增 public void add(T t){ } //泛型方法: //举例:获取表中一共有多少条记录?获取最大员工入职时间? //各种诉求的返回值类型不确定,所以定义成泛型方法 public <E> E getValue(){ return null; } }
-
11.3 泛型在继承上的体现
-
类A是类B的父类,G<A>和G<B> 二者不具备子父类关系,二者是并列关系
示例:
public class Test { public static void main(String[] args) { Object obj = null; String str = null; //str可以赋值给obj,多态的体现 obj = str; List<Object> list1 = null; List<String> list2 = null; //编译错误 //list1 = list2; show(list1); //编译错误 //show(list2) } public static <E> void show(List<Object> List1){ } }
-
类A是类B的父类,A<G>和B<G> 二者具备子父类关系
示例:
public class Test { public static void main(String[] args) { List<String> list1 = null; ArrayList<String> list2 = null; list1 = list2;//允许 } }
11.4 通配符的使用
-
上面例子中G<A>和G<B>不具有子父类关系了,所以就需要重载很多的方法
public class Test { public static void main(String[] args) { List<Object> list1 = null; List<String> list2 = null; show(list1); show1(list2); } public static <E> void show(List<Object> List){ } public static <E> void show1(List<String> list){ } }
-
同配符:?
-
类A是类B的父类,G<A>和G<B> 二者不具备子父类关系,二者共同的父类是 :G<?>
public class Test { public static void main(String[] args) { List<Object> list1 = null; List<String> list2 = null; List<?> list = null; list = list1; list = list2; } public static <E> void show(List<?> list){ Iterator<?> iterator = list.iterator(); while (iterator.hasNext()){ Object next = iterator.next(); System.out.println(next); } } }
-
读取和写入时的注意点:
-
添加(写入):对于 List<?> 不能向其内部添加数据
List<String> list1 = new ArrayList<>(); list1.add("AA"); list1.add("BB"); List<?> list = null; list = list1; //对于List<?>不能想起内部添加数据 //list.add("CC");
-
获取(读取):允许读取数据,读取的数据类型为Object
List<String> list1 = new ArrayList<>(); list1.add("AA"); list1.add("BB"); List<?> list = null; list = list1; Object o = list.get(0); System.out.println(o);
-
-
有限制条件的通配符的使用:
-
条件:
- <? extends Person>
继承于Person的类,包括Person
- <? super Person>
被Person继承的类,包括Person
- <? extends Person>
-
示例:
public class Test { public static void main(String[] args) { List<? extends Person> list1 = null; List<? super Person> list2 = null; List<student> list3 = null; List<Person> list4 = null; List<Object> list5 = null; list1 = list3; list1 = list4; //extends相当于表示范围 <= //list1 = list5;编译不通过 //list2 = list3;编译不通过 //super相当于表示范围 >= list2= list4; list2 = list5; } }
-
读数据时:使用范围内最大的数据类型接收
-
写入数据时:写入比条件中的类相等或者更小的类型
-
11.5 泛型应用举例
//泛型嵌套
public static void main(String[] args) {
HashMap<String, ArrayList<Citizen>> map = new HashMap<String, ArrayList<Citizen>>();
ArrayList<Citizen> list = new ArrayList<Citizen>();
list.add(new Citizen("刘恺威"));
list.add(new Citizen("杨幂"));
list.add(new Citizen("小糯米"));
map.put("刘恺威", list);
Set<Entry<String, ArrayList<Citizen>>> entrySet = map.entrySet();
Iterator<Entry<String, ArrayList<Citizen>>> iterator = entrySet.iterator();
while (iterator.hasNext()) {
Entry<String, ArrayList<Citizen>> entry = iterator.next();
String key = entry.getKey();
ArrayList<Citizen> value = entry.getValue();
System.out.println("户主: " + key);
System.out.println("家庭成员: " + value);
}
}