Java中的泛型


一、泛型概述

  • 泛型:是JDK5中引入的特性,可以在编译阶段约束操作的数据类型,并进行检查
  • 格式:<数据类型>
  • 优点:
    • 统一数据类型
    • 把运行期间的问题提前到编译期间,避免了强制类型转化可能出现的异常,因为在编译阶段就能确定

注意:

  • 泛型只能支持引用数据类型,不能写基本数据类型
  • 指定泛型的具体类型后,传递数据时,可以传入该类类型或其子类类型
  • 如果不写泛型,类型默认是Object
 ArrayList<String> s=new ArrayList<>();

二、泛型类

  • 使用场景:当一个类,某个变量的数据类型不确定是,就可以定义带有泛型的类
  • 格式:修饰符 class 类名<类型>{ }
 public class ArrayList<E>{   //创建该类对象时,E就确定类型
        
    }
//E可理解为变量,但是不是用来记录数据的,而是记录数据的类型,可以写成T、E、K、V等

MyArrayList.java

import java.util.Arrays;
public class MyArrayList<E> {
    Object[] obj=new Object[10];
    int size;

    /*
    * E:表示不确定的类型,该类型在类名后面已经定义过
    * e:形参的名字,变量名
    * */
    public boolean add(E e){
        obj[size]=e;
        size++;
        return true;
    }

    public E get(int index){
        return (E)obj[index];
    }

    @Override
    public String toString() {
        return Arrays.toString(obj);
    }
}

Demo.java

public class Demo {
    public static void main(String[] args) {

        MyArrayList<String> list=new MyArrayList<>();
        list.add("aaa");
        list.add("bbb");

        System.out.println(list);
    }
}

三、泛型方法

  • 格式:修饰符 <类型> 返回值类型 方法名(类型 变量名){ }

  • 方法中形参类型不确定时候有两种方法

  1. 使用类名后面定义的泛型< E >
public class MyArrayList<E> {
 public boolean add(E e){
        obj[size]=e;
        size++;
        return true;
    }
 }

所有方法都能用

  1. 在方法申明上定义自己的泛型
public class MyArrayList {
 public <E> boolean add(E e){
        obj[size]=e;
        size++;
        return true;
    }
 }

只有本方法能用

四、泛型接口

  • 格式:修饰符 interface 接口名<类型>{ }
  • 泛型接口的两种使用方式:
  1. 实现类给出的具体类型
public class MyArrayList implements List<String> {

    @Override
    public int size() {
        return 0;
    }

    @Override
    public boolean isEmpty() {
        return false;
    }

    @Override
    public boolean contains(Object o) {
        return false;
    }

    @Override
    public Iterator<String> iterator() {
        return null;
    }

    @Override
    public Object[] toArray() {
        return new Object[0];
    }

    @Override
    public <T> T[] toArray(T[] a) {
        return null;
    }

    @Override
    public boolean add(String s) {
        return false;
    }

    @Override
    public boolean remove(Object o) {
        return false;
    }

    @Override
    public boolean containsAll(Collection<?> c) {
        return false;
    }

    @Override
    public boolean addAll(Collection<? extends String> c) {
        return false;
    }

    @Override
    public boolean addAll(int index, Collection<? extends String> c) {
        return false;
    }

    @Override
    public boolean removeAll(Collection<?> c) {
        return false;
    }

    @Override
    public boolean retainAll(Collection<?> c) {
        return false;
    }

    @Override
    public void clear() {

    }

    @Override
    public String get(int index) {
        return null;
    }

    @Override
    public String set(int index, String element) {
        return null;
    }

    @Override
    public void add(int index, String element) {

    }

    @Override
    public String remove(int index) {
        return null;
    }

    @Override
    public int indexOf(Object o) {
        return 0;
    }

    @Override
    public int lastIndexOf(Object o) {
        return 0;
    }

    @Override
    public ListIterator<String> listIterator() {
        return null;
    }

    @Override
    public ListIterator<String> listIterator(int index) {
        return null;
    }

    @Override
    public List<String> subList(int fromIndex, int toIndex) {
        return null;
    }
 }
  1. 实现类延续泛型,创建实现类对象时在确定类型
public class MyArrayList2<E> implements List<E>{
    @Override
    public int size() {
        return 0;
    }

    @Override
    public boolean isEmpty() {
        return false;
    }

    @Override
    public boolean contains(Object o) {
        return false;
    }

    @Override
    public Iterator<E> iterator() {
        return null;
    }

    @Override
    public Object[] toArray() {
        return new Object[0];
    }

    @Override
    public <T> T[] toArray(T[] a) {
        return null;
    }

    @Override
    public boolean add(E e) {
        return false;
    }

    @Override
    public boolean remove(Object o) {
        return false;
    }

    @Override
    public boolean containsAll(Collection<?> c) {
        return false;
    }

    @Override
    public boolean addAll(Collection<? extends E> c) {
        return false;
    }

    @Override
    public boolean addAll(int index, Collection<? extends E> c) {
        return false;
    }

    @Override
    public boolean removeAll(Collection<?> c) {
        return false;
    }

    @Override
    public boolean retainAll(Collection<?> c) {
        return false;
    }

    @Override
    public void clear() {

    }

    @Override
    public E get(int index) {
        return null;
    }

    @Override
    public E set(int index, E element) {
        return null;
    }

    @Override
    public void add(int index, E element) {

    }

    @Override
    public E remove(int index) {
        return null;
    }

    @Override
    public int indexOf(Object o) {
        return 0;
    }

    @Override
    public int lastIndexOf(Object o) {
        return 0;
    }

    @Override
    public ListIterator<E> listIterator() {
        return null;
    }

    @Override
    public ListIterator<E> listIterator(int index) {
        return null;
    }

    @Override
    public List<E> subList(int fromIndex, int toIndex) {
        return null;
    }
}

五、泛型的继承和通配符

  • 泛型不具备继承性,但数据具有继承性
public class GenericsDemo {
    public static void main(String[] args) {
        ArrayList<Ye> list1=new ArrayList<>();
        ArrayList<Fu> list2=new ArrayList<>();
        ArrayList<Zi> list3=new ArrayList<>();
//        调用method方法
        method(list1);
        method(list2);  //报错
        method(list3);  //报错,因为泛型不具备继承性。

        //不报错,数据具有继承性
        list1.add(new Ye());
        list1.add(new Fu());
        list1.add(new Zi());
    }
    public static void method(ArrayList<Ye> list){

    }
}

class Ye{

}

class Fu extends Ye{

}

class Zi extends Fu{

}
  • 通配符

在泛型离写什么方法,它就只能传递什么数据类型,所以存在一个弊端:即它课接受任意数据类型如上例中的Ye Fu Zi 甚至Student对象,但是如果只希望传递Ye Fu Zi,此时就要用到通配符。

?:表示不确定的类型
? extends E :表示可以传递E或者E的所有子类类型
? super E :表示可以传递E或者E的所有父类类型
public class GenericsDemo3 {
   
    public static void main(String[] args) {
        ArrayList<Ye> list1=new ArrayList<>();
        ArrayList<Fu> list2=new ArrayList<>();
        ArrayList<Zi> list3=new ArrayList<>();      
        method(list1);
        method(list2);
        method(list3);

    }
    public static<E> void method(ArrayList<? extends Ye> list){

    }
}

class Ye{

}

class Fu extends Ye{

}

class Zi extends Fu{

}

六、没有泛型的时候,集合如何存储数据

  • 没有给集合指定类型,默认所有数据类型都是Object类型,此时可往集合里添加任意数据类型
  • 缺点:获取数据时,无法使用它的特有行为
Java泛型通配符用于解决类型安全问题,并提供了一种方法来引用类型,而不需要知道具体的类型参数。通配符主要有以下几种形式: 1. `?`:无界通配符,表示未知的类型,可以匹配任何对象的类型。 2. `<? extends 类或接口>`:表示具体的类型,但不是确定的类型。`extends`在这里的意思是“某种类型的子类”。这可以用来限定类型参数的上界,只允许该通配符引用某个类型或其子类型。 3. `<? super 类或接口>`:表示具体的类型,但不是确定的类型。`super`在这里的意思是“某种类型的父类”。这可以用来限定类型参数的下界,只允许该通配符引用某个类型或其父类型。 4. `? extends 类型1 & 类型2 & ... & 类型n>`:表示有限制的通配符,可以引用同时满足多个上限类型约束的类型。 使用泛型通配符的目的是为了增强代码的灵活性,同时保证类型安全。例如,如果有一个泛型方法需要接受任何类型的`List`作为参数,但不想限制这个`List`只能包含特定类型的对象,那么可以使用无界通配符`<?>`。 下面是一个简单的例子,来说明如何使用泛型通配符: ```java public class GenericWildcardExample { public static void processElements(List<?> elements) { for (Object element : elements) { // 处理元素,但不知道元素的实际类型 } } } ``` 在这个例子,`processElements`方法可以接受任何类型的`List`对象作为参数,但在方法内部,我们只能将元素当作`Object`类型来处理。 使用泛型通配符的注意事项: - 不能实例化泛型通配符。 - 不能使用带有泛型通配符的类型作为方法的参数类型,因为这会导致在调用时出现类型安全问题。 - 如果要向泛型集合添加元素,那么只能添加`Object`类型的元素,因为编译器不知道集合具体存储的是哪种类型的对象。
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值