java基础——泛型

泛型

基本介绍

  1. 泛型又称参数化类型,是Jdk5.0出现的新特性,解决数据类型的安全性问题

  2. 在类声明或实例化时只要指定好需要的具体的类型即可。

  3. Java泛型可以保证如果程序在编译时没有发出警告,运行时就不会产生ClassCastException异常。同时,代码更加简洁、健壮

  4. 泛型的作用是:可以在类声明时通过一个标识表示类中某个属性的类型,或者是某个方法的返回值的类型,或者是参数类型。[有点难,举例Generic03.java]

    在这里插入图片描述

基本语法

  • interface 接口{}和class类<K,V>{}
  • List strList = new ArrayList0;
  • Iterator iterator = customers.iterator();

说明:

  1. 其中,T,K,V不代表值,而是表示类型。
  2. 任意字母都可以。常用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; }
}

使用细节

  1. interface List0{}, public class HashSet{}…等等

    说明:T,E只能是引用类型
    看看下面语句是否正确?:
    List list = new ArrayList0;

    ​ List list2 = new ArrayList0;//错误

  2. 在指定泛型具体类型后,可以传入该类型或者其子类类型

  3. 泛型使用形式

    ​ List list1 = new ArrayList ();

    ​ List list2 =new ArrayList<>();

  4. ArrayList arrayList = new TArrayList();//等价 ArrayList arrayList = new ArrayList<>();

自定义泛型类

语法

class类名<T,R…>{

​ 成员

}

注意细节

  1. 普通成员可以使用泛型(属性、方法)
  2. 使用泛型的数组,不能初始化
  3. 静态方法中不能使用类的泛型
  4. 泛型类的类型,是在创建对象时确定的(因为创建对象时,需要指定确定类型)
  5. 如果在创建对象时,没有指定类型,默认为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…>{}

注意细节

  1. 接口中,静态成员也不能使用泛型(这个和泛型类规定一样)
  2. 泛型接口的类型,在继承接口或者实现接口时确定
  3. 没有指定类型,默认为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…>返回类型方法名(参数列表){

}

注意细节

  1. 泛型方法,可以定义在普通类中,也可以定义在泛型类中
  2. 当泛型方法被调用时,类型会确定
  3. public void eat(Ee)0,修饰符后没有<T,R…> eat方法不是泛型方法,而是使用了泛型

泛型的继承和通配符

  1. 泛型不具备继承性
    List list = new ArrayList0; /对吗?//不对
  2. <?>∶支持任意泛型类型
  3. <? extends A>:支持A类以及A类的子类,规定了泛型的上限
  4. <? 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; }
}

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

XZY-SUNSHINE

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值