Java随笔-泛型

概述

泛型,即“参数化类型”,就是将类型由原来的具体类型改成类似于方法中变量参数那样,在使用的时候再传入具体的类型。这种在使用时候将操作的数据类型作为参数,常用在类,方法,接口中。

使用

public class ArrayList<E> extends AbstractList<E>
        implements List<E>, RandomAccess, Cloneable, java.io.Serializable

泛型在容器中使用的最为频繁。

   List<String> listId = new ArrayList<>();
   List<Integer> skuidList = new ArrayList<>();

泛型的主要目的之一就是指定容器需要存储数据的类型。容器可以持有多种类型的对象,在使用的时候通常一个容器只会存储一种类型的对象,所以泛型在容器中使用最频繁。
通常为了更好的扩展,又有所约束,会在公共的一些工具或基类中使用泛型,好处如下:

  1. 多种数据类型可以执行相同的代码。
  2. 无需强制类型转换。
public class GenericTest2<K,V,E> {
    private K key;
    private V value;
    private E other;

    public GenericTest2(K key, V value, E other) {
        this.key = key;
        this.value = value;
        this.other = other;
    }

    public void show(){
        System.out.println("键:" + key + ",值:" + value + "," + other + "是多余的");
    }
}

使用继承可以实现长度更多的数据类型。

public class GenericTest3<A,B,C,D> extends GenericTest2<A,B,C> {
    private D d;
    
    public GenericTest3(A key, B value, C other,D d) {
        super(key, value, other);
        this.d = d;
    }
}

方法

    public <T> T genericMethod(T t) {
        return t;
    }

泛型方法就是在调用的时候指定具体类型,泛型方法可以在普通类中使用,也可以在泛型类中使用。在泛型类中使用是,如果使用了泛型类声明的类型,此时泛型方法就是不同的方法而不是泛型方法。

public class GenericTest<T> {
    private T data;

    public T getData() {
        return data;
    }
 }

此时返回的是泛型类中声明的类型,而不是方法自己声明的类型。
泛型方法的在修饰符与返回类型之间的类型是必不可少的,只有声明的这个,才能表示这是一个泛型方法。
在这里插入图片描述
泛型方法也可以有多个泛型。

  public <K,V> void genericMethod2(K k) {
  }

接口

public interface Generator<T> {

    void show(T t);
}

实现类:

public class GeneratorImpl<T> implements Generator<T>{
    @Override
    public void show(T t) {
        System.out.println("泛型:" + t);
    }
}

要点

限定泛型类型。

    /**
     * 取大值
     * @param a
     * @param b
     * @param <T>
     * @return
     */
    public static <T extends Comparable> T max(T a, T b) {
        return a.compareTo(b) > 0 ? a : b;
    }

如果传入的参数没有实现Comparable,则编译报错。限定可以是多个。

    public static <T extends Comparable & Serializable> T max(T a, T b) {
        return a.compareTo(b) > 0 ? a : b;
    }

这里的Comparable & Serializable都是接口,如果是类的话就需要放在第一个位置。

public class GenericClass {
}
    public static <T extends GenericClass & Comparable & Serializable> T max(T a, T b) {
        return a.compareTo(b) > 0 ? a : b;
    }

实用类的时候只能使用一个,而且必须放在限定列表的第一个。Java中类只能单继承,而接口是可以多实现的。

通配符。

在这里插入图片描述

  • ? extends X,表示参数类型必须是X的子类。
    public static void main(String[] args) {

        GenericTest<Father> father = new GenericTest<>();
        GenericTest<Son> son = new GenericTest<>();
        print(son);
    }

    static void print(GenericTest<? extends Father> father){
        System.out.println(father);
    }

使用? extends X时可以访问X和X的子类,但是不能set方法,因为set的时候编译器不知道具体的数据类型会报错,但是可以是用get方法,get之前类型已确定,也不能卸任非null的数据。

  • ? super X,表示参数类型必须是X的父类。
    在这里插入图片描述
  • 如果没有限制,使用通配符没有意义。
ArrayList<?> list1 = new ArrayList<>();

约束不能使用基本数据类型。

  ArrayList list = new ArrayList<int>();

这行代码编译器会报异常:

Type argument cannot be of primitive type

此时使用包装类就没事。

  ArrayList list = new ArrayList<Integer>();
  • 类型判断只能使用原始类型,不能使用泛型。
    在这里插入图片描述

    在这里插入图片描述
    即使指定了泛型中的类型,也是没有办法判断的,编译器会直接报错的。

静态域中泛型失效。

静态域中肯定不能使用泛型,静态域是随着类的加载而加载,而泛型是创建对象的时候才确定,加载静态域的时候对象还没创建呢,所以使用泛型无效。
在这里插入图片描述

  • 不能创建泛型类数组对象。
    在这里插入图片描述
    但是只声明是可以的。
 GenericTest<String>[] strArray;

但是声明完后不能使用,声明没有意义,浪费空间。

不能捕获泛型类实例。

在这里插入图片描述
在这里插入图片描述
但是catch Throwable或Exception肯定就没问题。

   public <E extends Throwable> void test(E e){
        try {

        } catch (Throwable e1){

        }
    }

泛型中使用父类和子类创建的对象之间没有任何关系。

父类:

public class Father {
}

子类:

public class Son extends Father {
}

泛型:
在这里插入图片描述
但是对泛型类是可以扩展的。
扩展:

public class GenericTest01<T> extends GenericTest<T> {
}

使用:

 GenericTest<Father> father = new GenericTest01<>();

就像ArrayList。

public class ArrayList<E> extends AbstractList<E>
        implements List<E>, RandomAccess, Cloneable, java.io.Serializable
public abstract class AbstractList<E> extends AbstractCollection<E> implements List<E> {

Java中泛型是伪泛型。

Java中泛型在编译后类型会自动擦除类型,是伪泛型,并非真正的泛型,编译后会进行替换和强制转型。所以:

        ArrayList<Integer> list  = new ArrayList<>();
        ArrayList<String> list0  = new ArrayList<>();
        

编译后是一样的。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Java泛型Java 5引入的新特性,可以提高代码的可读性和安全性,降低代码的耦合度。泛型是将类型参数化,实现代码的通用性。 一、泛型的基本语法 在声明类、接口、方法时可以使用泛型泛型的声明方式为在类名、接口名、方法名后面加上尖括号<>,括号中可以声明一个或多个类型参数,多个类型参数之间用逗号隔开。例如: ```java public class GenericClass<T> { private T data; public T getData() { return data; } public void setData(T data) { this.data = data; } } public interface GenericInterface<T> { T getData(); void setData(T data); } public <T> void genericMethod(T data) { System.out.println(data); } ``` 其中,`GenericClass`是一个泛型类,`GenericInterface`是一个泛型接口,`genericMethod`是一个泛型方法。在这些声明中,`<T>`就是类型参数,可以用任何字母代替。 二、泛型的使用 1. 泛型类的使用 在使用泛型类时,需要在类名后面加上尖括号<>,并在括号中指定具体的类型参数。例如: ```java GenericClass<String> gc = new GenericClass<>(); gc.setData("Hello World"); String data = gc.getData(); ``` 在这个例子中,`GenericClass`被声明为一个泛型类,`<String>`指定了具体的类型参数,即`data`字段的类型为`String`,`gc`对象被创建时没有指定类型参数,因为编译器可以根据上下文自动推断出类型参数为`String`。 2. 泛型接口的使用 在使用泛型接口时,也需要在接口名后面加上尖括号<>,并在括号中指定具体的类型参数。例如: ```java GenericInterface<String> gi = new GenericInterface<String>() { private String data; @Override public String getData() { return data; } @Override public void setData(String data) { this.data = data; } }; gi.setData("Hello World"); String data = gi.getData(); ``` 在这个例子中,`GenericInterface`被声明为一个泛型接口,`<String>`指定了具体的类型参数,匿名内部类实现了该接口,并使用`String`作为类型参数。 3. 泛型方法的使用 在使用泛型方法时,需要在方法名前面加上尖括号<>,并在括号中指定具体的类型参数。例如: ```java genericMethod("Hello World"); ``` 在这个例子中,`genericMethod`被声明为一个泛型方法,`<T>`指定了类型参数,`T data`表示一个类型为`T`的参数,调用时可以传入任何类型的参数。 三、泛型的通配符 有时候,我们不知道泛型的具体类型,可以使用通配符`?`。通配符可以作为类型参数出现在方法的参数类型或返回类型中,但不能用于声明泛型类或泛型接口。例如: ```java public void printList(List<?> list) { for (Object obj : list) { System.out.print(obj + " "); } } ``` 在这个例子中,`printList`方法的参数类型为`List<?>`,表示可以接受任何类型的`List`,无论是`List<String>`还是`List<Integer>`都可以。在方法内部,使用`Object`类型来遍历`List`中的元素。 四、泛型的继承 泛型类和泛型接口可以继承或实现其他泛型类或泛型接口,可以使用子类或实现类的类型参数来替换父类或接口的类型参数。例如: ```java public class SubGenericClass<T> extends GenericClass<T> {} public class SubGenericInterface<T> implements GenericInterface<T> { private T data; @Override public T getData() { return data; } @Override public void setData(T data) { this.data = data; } } ``` 在这个例子中,`SubGenericClass`继承了`GenericClass`,并使用了相同的类型参数`T`,`SubGenericInterface`实现了`GenericInterface`,也使用了相同的类型参数`T`。 五、泛型的限定 有时候,我们需要对泛型的类型参数进行限定,使其只能是某个类或接口的子类或实现类。可以使用`extends`关键字来限定类型参数的上限,或使用`super`关键字来限定类型参数的下限。例如: ```java public class GenericClass<T extends Number> { private T data; public T getData() { return data; } public void setData(T data) { this.data = data; } } public interface GenericInterface<T extends Comparable<T>> { T getData(); void setData(T data); } ``` 在这个例子中,`GenericClass`的类型参数`T`被限定为`Number`的子类,`GenericInterface`的类型参数`T`被限定为实现了`Comparable`接口的类。 六、泛型的擦除 在Java中,泛型信息只存在于代码编译阶段,在编译后的字节码中会被擦除。在运行时,无法获取泛型的具体类型。例如: ```java public void genericMethod(List<String> list) { System.out.println(list.getClass()); } ``` 在这个例子中,`list`的类型为`List<String>`,但是在运行时,`getClass`返回的类型为`java.util.ArrayList`,因为泛型信息已经被擦除了。 七、泛型的类型推断 在Java 7中,引入了钻石操作符<>,可以使用它来省略类型参数的声明。例如: ```java List<String> list = new ArrayList<>(); ``` 在这个例子中,`ArrayList`的类型参数可以被编译器自动推断为`String`。 八、总结 Java泛型是一个强大的特性,可以提高代码的可读性和安全性,降低代码的耦合度。在使用泛型时,需要注意它的基本语法、使用方法、通配符、继承、限定、擦除和类型推断等问题。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值