泛型

2020-09-07  本文已影响0人  我是许仙

本质

参数化类型, 指定类型防止出现参数转化异常

泛型的擦除

只在编译阶段有效,编译之后jvm会采取去泛型化的措施。可以通过反射跳过泛型的检查。

public class Erase {

    public static void main(String[] args) throws Exception {
        List<String> stringList = new ArrayList<>();
        stringList.add("1");
        stringList.add("2");
        //直接添加会失败 编译不通过提示错误
        //stringList.add(new Object());

        //通过反射跳过编译添加object对象在list集合
        Class clas = stringList.getClass();
        Method method = clas.getDeclaredMethod("add", Object.class);
        method.invoke(stringList, new Object());

        System.out.println(stringList);
    }

}

通配符

无边界

public class NoBorder {

    //无边界通配符 可以任意传
    public static  void  test(List<?> list) {
        System.out.println(list.size());
    }

    public static void main(String[] args) {
        List<String> stringList = new ArrayList<>();
        test(stringList);
    }
}

上边界

public class UpBorder {
    //上边界通配符 必须是Number的子类
    public static void test(List<? extends  Number> list) {
        System.out.println(list.size());
    }

    public static void main(String[] args) {
        List<Integer> numberList = new ArrayList<>();
        test(numberList);
    }
}

下边界

public class DownBorder {

    //下边界通配符  必须是 number或者 number - object
    public static void test(List<? super Number> list) {
        System.out.println(list.size());
    }

    public static void main(String[] args) {
        List<Object> objects = new ArrayList<>();
        test(objects);
    }
}

规则

必须先声明在使用,泛型的声明是通过一对<>实现的,约定泛型可以使用单个大写字母来表示。比如说常见T E等字母并没有任何的意义只不过是泛型的声明罢了。

<span style='color:red'>错误,无声明</span>

public class Demo {
    //定义一个方法传入一个泛型参数,返回一个泛型的对象。因为这里没有声明报错
    public  H getName(H h) {
        return null;
    }
}

正确的写法

public <H> H getName(H h) {
    return null;
}

使用

泛型类
//声明
public class DemoClass<T> {
    private T getName;

    public T getGetName() {
        return getName;
    }

    public void setGetName(T getName) {
        this.getName = getName;
    }

    public static void main(String[] args) {
        //new的时候用<>指定泛型的类型
        DemoClass<String> demoClass = new DemoClass<>();
        demoClass.setGetName("Test");
    }
}
泛型方法
public  class GenericMethod <K,V> {

    //使用类中定义的泛型
    public K getName(K k,V v) {
        return (K) null;
    }
    //使用方法中定义的泛型
    public <T> T getName(K k) {
        return (T) k;
    }
    
    //静态方法无法使用类中定义的泛型,只能使用方法定义的泛型
    public static <K,V> K getName1(V v) {
        return (K) v;
    }
    
}
泛型接口
public interface GenericInterface<K,V> {

    public K getName(V v);
}
//实现了泛型接口的类 需要指定泛型的参数
public class GenericInterfaceImpl implements GenericInterface<String,Integer> {
    @Override
    public String getName(Integer integer) {
        return null;
    }
}
上一篇 下一篇

猜你喜欢

热点阅读