泛型
基本介绍
-
泛型又称参数化类型,是Jdk5.0出现的新特性,解决数据类型的安全性问题
-
在类声明或实例化时只要指定好需要的具体的类型即可。
-
Java泛型可以保证如果程序在编译时没有发出警告,运行时就不会产生ClassCastException异常。同时,代码更加简洁、健壮
-
泛型的作用是:可以在类声明时通过一个标识表示类中某个属性的类型,或者是某个方法的返回值的类型,或者是参数类型。[有点难,举例Generic03.java]
基本语法
- interface 接口{}和class类<K,V>{}
- List strList = new ArrayList0;
- Iterator iterator = customers.iterator();
说明:
- 其中,T,K,V不代表值,而是表示类型。
- 任意字母都可以。常用T表示,是Type的缩写
实例对比
package generic;
import java.util.ArrayList;
public class exerciese0q {
public static void main(String[] args) {
ArrayList arrayList = new ArrayList();
//需求
// 请编写程序,在ArrayList中,添加3个Dog对象
// Dog对象含有name和 age,并输出name和 age(要求使用getXxx())
// 如果程序员不小心在里面添加了一个猫
//使用传统的方法来解决,在for循环时就会报错
arrayList.add(new cat("啦啦啦",1));
arrayList.add(new Dog("旺财",10));
arrayList.add(new Dog("发财",1));
arrayList.add(new Dog("小黄",5));
// 遍历
for (Object o : arrayList) {
// 向下转型Object ->Dog
Dog dog = (Dog)o;
System.out.println(dog.getName() + "-" + dog.getAge());
}
// 使用传统方法的问题分析
// 1.不能对加入到集合ArrayList中的数据类型进行约束(不安全)
// 2.遍历的时候,需要进行类型转换,如果集合中的数据量较大,对效率有影响
//泛型解决
//1.当我们 ArrayList<Dog>表示存放到 ArrayList 集合中的元素是Dog类型(细节后面说...)
//2。如果编译器发现添加的类型,不满足要求,就会报错
// 3.在遍历的时候,可以直接取出Dog 类型而不是 0bject
ArrayList<Dog> arrayList1 = new ArrayList<Dog>();
arrayList1.add(new Dog("旺财",10));
arrayList1.add(new Dog("发财",1));
arrayList1.add(new Dog("小黄",5));//假如我们的程序员,不小心,添加了一只猫l/arrayList.add(new Cat("招财猫",8));
for (Dog dog : arrayList1) {
System.out.println(dog. getName() + "-" + dog. getAge();
}
}
}
class Dog {
private String name;private int age;
public Dog(String name,int age) {
this.name = name;
this.age = age;
}
public String getName() { return name; }
public void setName(String name) { this.name = name; }
public int getAge() { return age; }
public void setAge(int age) { this.age = age; }
}
class cat{
private String name;private int age;
public cat(String name,int age) {
this.name = name;
this.age = age;
}
public String getName() { return name; }
public void setName(String name) { this.name = name; }
public int getAge() { return age; }
public void setAge(int age) { this.age = age; }
}
使用细节
-
interface List0{}, public class HashSet{}…等等
说明:T,E只能是引用类型
看看下面语句是否正确?:
List list = new ArrayList0; List list2 = new ArrayList0;//错误
-
在指定泛型具体类型后,可以传入该类型或者其子类类型
-
泛型使用形式
List list1 = new ArrayList ();
List list2 =new ArrayList<>();
-
ArrayList arrayList = new TArrayList();//等价 ArrayList arrayList = new ArrayList<>();
自定义泛型类
语法
class类名<T,R…>{
成员
}
注意细节
- 普通成员可以使用泛型(属性、方法)
- 使用泛型的数组,不能初始化
- 静态方法中不能使用类的泛型
- 泛型类的类型,是在创建对象时确定的(因为创建对象时,需要指定确定类型)
- 如果在创建对象时,没有指定类型,默认为Object
代码案例
//1.Tiger后面泛型,所以我们把Tiger就称为自定义泛型类
//2.T,R,M泛型的标识符,一般是单个大写字母
//3.泛型标识符可以有多个。
//4.普通成员可以使用泛型〔属性、方法)
//5.使用泛型的数组,不能初始化
class Tiger<T,R,M> {
String name;
R r;//属性使用到泛型
M m;
T t;
//因为数组在new不能确定T的类型,就无法在内存开空间
T[] ts;
public Tiger(String name,R r,M m,T t){//构造器使用泛型
this.name = name;
this.r = r;
this.m = m;
this.t = t;
}
// public static void m1(M m){
//静态方法中不能使用类的泛型
// }
public String getName() { return name; }
public void setName(String name) { this.name = name; }
public R getR() { return r; }
public void setR(R r) { this.r = r; }
public T getT() { return t; }
public void setT(T t) { this.t = t; }
public T[] getTs() { return ts; }
public void setTs(T[] ts) { this.ts = ts; }
}
自定义泛型接口
基本语法
interface接口名<T,R…>{}
注意细节
- 接口中,静态成员也不能使用泛型(这个和泛型类规定一样)
- 泛型接口的类型,在继承接口或者实现接口时确定
- 没有指定类型,默认为Object
代码案例
package generic;
interface IUsb<U,R> {
int n = 10;
//U name;不能这样使用
//普通方法中,可以使用接口泛型R get(U u);
void hi(R r);
void run(R r1,R r2,U u1, U u2);
//在jdk8中,可以在接口中,使用默认方法,也是可以使用泛型
default R method(U u) {
return null;
}
}
//在继承接口指定泛型接口的类型
interface IA extends IUsb<String,Double> {
}
//当我们去实现IA接口时,因为IA在继承IUsU 接口时,指定了U为String R为Doublel
class AA implements IA {
@Override
public void hi(Double aDouble) { }
@Override
public void run(Double r1, Double r2, String u1, String u2) { }
@Override
public Double method(String s) { return null; }
}
//没有指定类型,默认为0bject
//class cc implements TUsb {//等价class CC implements IUsb<Object,0bjects {
自定义泛型方法
基本语法
修饰符<T,R…>返回类型方法名(参数列表){
}
注意细节
- 泛型方法,可以定义在普通类中,也可以定义在泛型类中
- 当泛型方法被调用时,类型会确定
- public void eat(Ee)0,修饰符后没有<T,R…> eat方法不是泛型方法,而是使用了泛型
泛型的继承和通配符
- 泛型不具备继承性
List list = new ArrayList0; /对吗?//不对 - <?>∶支持任意泛型类型
- <? extends A>:支持A类以及A类的子类,规定了泛型的上限
- <? super A>:支持A类以及A类的父类,不限于直接父类,规定了泛型的下限
章节练习题
package generic;
import org.junit.jupiter.api.Test;
import java.util.*;
public class work04 {
public static void main(String[] args) {
// 定义个泛型类 DAO<T>,在其中定义一个Map成员变量,
// Map的键为String类型,值为T类型。分别创建以下方法:
// (1) public void save(String id,T entity):保存T类型的对象
// 到 Map成员变量中
// (2) public T get(String id):从map中获取id对应的对象
// (3) public void update(String id,T entity):
// 替换map中key为id的内容,改为entity对象
// (4) public List<T> list():返回map中存放的所有T对象
// (5) public void delete(String id):删除指定id对象
// 定义一个 User 类:
// 该类包含:private成员变量(int类型)id, age;(String类型) name
// 创建 DAO类的对象,分别调用其 save.get、update、list、
// delete方法来操作User对象,使用Junit单元测试类进行测试
}
@Test
public void testList(){
//说明/这里我们给T指定类型User
DAO<User> dao = new DAO<>();
dao.save("001", new User(1, 10, "jack"));
dao.save("002", new User(2, 18, "king"));
dao.save("003", new User(3, 38, "smith"));
List<User> list = dao.list();
System.out.println("list=" + list);
}
}
class DAO<T>{
Map<String ,T> map=new HashMap();
public void save(String id,T entity){
map.put(id,entity);
System.out.println("添加成功!");
}
public T get(String id){
return map.get(id);
}
public void update(String id,T entity){
map.replace(id,entity);
System.out.println("更新成功!");
}
public List<T> list(){
ArrayList<T> ts = new ArrayList<>();
Set<String> keys = map.keySet();
for (Object o :keys) {
ts.add(map.get(o));
}
return ts;
}
public void delete(String id){
map.remove(id);
}
}
class User{
private int id;
private int age;
private String name;
public User(int id, int age, String name) {
this.id = id;
this.age = age;
this.name = name;
}
public int getId() { return id; }
public void setId(int id) { this.id = id; }
public int getAge() { return age; }
public void setAge(int age) { this.age = age; }
public String getName() { return name; }
public void setName(String name) { this.name = name; }
}