1. 为什么要用泛型
2. 集合中泛型的使用
package com.atguigu.java;
import org.junit.Test;
import java.util.*;
/**
*
* 泛型的使用
* 1.jdk 5.0新增的特性
*
* 2.在集合中使用泛型:
* 总结:
* ① 集合接口或集合类在jdk5.0时都修改为带泛型的结构。
* ② 在实例化集合类时,可以指明具体的泛型类型
* ③ 指明完以后,在集合类或接口中凡是定义类或接口时,内部结构(比如:方法、构造器、属性等)使用到类的泛型的位置,都指定为实例化的泛型类型。
* 比如:add(E e) --->实例化以后:add(Integer e)
* ④ 注意点:泛型的类型必须是类,不能是基本数据类型。需要用到基本数据类型的位置,拿包装类替换
* ⑤ 如果实例化时,没有指明泛型的类型。默认类型为java.lang.Object类型。
*
* 3.如何自定义泛型结构:泛型类、泛型接口;泛型方法。见 GenericTest1.java
*
* @author shkstart
* @create 2019 上午 9:59
*/
public class GenericTest {
//在集合中使用泛型之前的情况:
@Test
public void test1(){
ArrayList list = new ArrayList();
//需求:存放学生的成绩
list.add(78);
list.add(76);
list.add(89);
list.add(88);
//问题一:类型不安全
// list.add("Tom");
for(Object score : list){
//问题二:强转时,可能出现ClassCastException
int stuScore = (Integer) score;
System.out.println(stuScore);
}
}
//在集合中使用泛型的情况:以ArrayList为例
@Test
public void test2(){
ArrayList<Integer> list = new ArrayList<Integer>();
list.add(78);
list.add(87);
list.add(99);
list.add(65);
//编译时,就会进行类型检查,保证数据的安全
// list.add("Tom");
//方式一:
// for(Integer score : list){
// //避免了强转操作
// int stuScore = score;
//
// System.out.println(stuScore);
//
// }
//方式二:
Iterator<Integer> iterator = list.iterator();
while(iterator.hasNext()){
int stuScore = iterator.next();
System.out.println(stuScore);
}
}
//在集合中使用泛型的情况:以HashMap为例
@Test
public void test3(){
// Map<String,Integer> map = new HashMap<String,Integer>();
//jdk7新特性:类型推断
Map<String,Integer> map = new HashMap<>();
map.put("Tom",87);
map.put("Jerry",87);
map.put("Jack",67);
// map.put(123,"ABC");
//泛型的嵌套
Set<Map.Entry<String,Integer>> entry = map.entrySet();
Iterator<Map.Entry<String, Integer>> iterator = entry.iterator();
while(iterator.hasNext()){
Map.Entry<String, Integer> e = iterator.next();
String key = e.getKey();
Integer value = e.getValue();
System.out.println(key + "----" + value);
}
}
}
3.自定义泛型类的使用
public class HTGenericTest {
public static void main(String[] args){
//如果定义了泛型类,实例化没有指明类的泛型,则认为此泛型类型为Object类型
//要求:如果大家定义了类是带泛型的,建议在实例化时要指明类的泛型。
order o = new order();
o.setT("123");
System.out.println(o.getT());//123
o.setT(123456);
System.out.println(o.getT());//123456
//使用泛型类
order<String> o1 = new order<>("456","tom", 12);
System.out.println(o1.getT());//456
//泛型的继承用法一
suborder1 sub1 = new suborder1("htt","hantong",19);
//由于子类在继承带泛型的父类时,指明了泛型类型。则实例化子类对象时,不再需要指明泛型。
System.out.println(sub1.getT());
//泛型的继承用法二
suborder2<String> sub2 = new suborder2<>("wxinxin","hantong",29);
System.out.println(sub2.getT());
}
}
class order<T>{
//参数T代表一个类
String orderName;
int orderId;
T t;//order的一个属性
//即只能声明一个T的对象t,但是不能new。因为一旦new的话就要在内存中创建该对象,但是并不知道这个对象具体是什么
// T t = new T();不能用这种方法来造对象
// 也不能用t来调用方法和属性
//构造器不用写成public order<T>()的形式
public order(){
}
public order(T t,String orderName,int orderId){
this.t = t;
this.orderName = orderName;
this.orderId = orderId;
}
public T getT(){
return t;
}
public void setT(T t){
this.t = t;
}
}
//方式一:继承时指明继承的泛型的具体类型
class suborder1<String> extends order<String>{
public suborder1(String s,String name,int orderId){
super(s, (java.lang.String) name,orderId);
}
}
//方式二:继承时不指明继承的泛型的具体类型
class suborder2<T> extends order<T>{
public suborder2(T t,String name,int orderId){
super(t, name,orderId);
}
}
4. 泛型方法
4.泛型在继承上的体现
/*
1. 泛型在继承方面的体现
虽然类A是类B的父类,但是G<A> 和G<B>二者不具备子父类关系,二者是并列关系。
补充:类A是类B的父类,A<G> 是 B<G> 的父类
*/
@Test
public void test1(){
Object obj = null;
String str = null;
obj = str;
Object[] arr1 = null;
String[] arr2 = null;
arr1 = arr2;
//编译不通过
// Date date = new Date();
// str = date;
List<Object> list1 = null;
List<String> list2 = new ArrayList<String>();
//此时的list1和list2的类型不具有子父类关系
//编译不通过
// list1 = list2;
/*
反证法:
假设list1 = list2;
list1.add(123);导致混入非String的数据。出错。
*/
show(list1);
show1(list2);
}
public void show1(List<String> list){
}
public void show(List<Object> list){
}
- 注意:如果 show(list2); 就会报错,因为 List<String> 不能赋值给 List<Object>
5. 通配符的使用