美文网首页
理解java的泛型

理解java的泛型

作者: 蒙多的菜刀 | 来源:发表于2018-06-07 16:13 被阅读0次

★可以把java的泛型理解为编译期的安全保障动作,在编译期保证了类型的检查,保证了输入输出时的类型正确。
★比方说,一个面具舞会,这个面具舞会的宗旨就是戴上面具,不问出身,不问来历,尽情的玩,但是那是进去参加之后的事啊,你要参加这个舞会,得凭邀请函吧,比如说这个邀请函只有大咖才能拥有,人凭着邀请函参加舞会,保安就会检查你的身份,一看你确实是符合的,身份证押着,进去玩吧。如果你没有邀请函,说明你不能进去,这个舞会的宗旨跟你毛关系没有了。进去参加的人大家都戴上了面具,就不知道你是哪里来的啊,你家住哪啊,这在泛型里叫类型擦除,你在舞会里就只是个人,玩完了,出去了,你得拿回你的身份证,好了,恢复身份,回家吧。
★在有泛型之前,我们可以用如下代码描述舞会:


import java.util.ArrayList;
import java.util.Random;

public class GenericMain {

    public static void main(String[] args) {
        MaskedBall maskedBall = new MaskedBall();
        //
        maskedBall.join(new ScienceBigShot());
        maskedBall.join(new GovernmtBigShot());
        //返回一个舞会的人要显式转型
        ((MaskPlayer)(maskedBall.aPlay())).play();;
    }

}

//面具舞会类
class MaskedBall extends ArrayList{
    private Random rd = new Random(24);
    public void join(Object mp){
        //检查身份,只有是有邀请函的(MaskPlayer)的人才能进去
        if(MaskPlayer.class.isInstance(mp)){
            this.add(mp);
            ((MaskPlayer)mp).play();
        }
        else{
            throw new RuntimeException();
        }
    }
    //随机返回一名舞者
    public Object aPlay(){
        return this.get(rd.nextInt(this.size()));
    }
    
}

//参加舞会的人
class MaskPlayer{
    private String name;
    
    public MaskPlayer(){
        this.name = "无名";
    }
    
    
    public String getName() {
        return name;
    }
    
    public void play(){
        System.out.println("忘记烦恼,尽情玩");
    }
}

class ScienceBigShot extends MaskPlayer{
    private String source;
    public ScienceBigShot(){
        this.source = "科技大咖";
    }
    public String getSource() {
        return source;
    }
    public void play(){
        System.out.println(this.source+"(无名)忘记烦恼,尽情玩");
    }
}

class GovernmtBigShot extends MaskPlayer{
    private String source;
    public GovernmtBigShot(){
        this.source = "政府大咖";
    }
    public String getSource() {
        return source;
    }
    public void play(){
        System.out.println(this.source+"(无名)忘记烦恼,尽情玩");
    }

}

★这样相当于进舞会,出舞会都需要人在那检查身份,这多不智能啊,如果有个什么系统拥有判断身份的能力就好了,泛型就给了编译期这样的能力。泛型使程序在边界处进行类型检查和恢复,在入口处(输入)进行类型检查,随后类型就被擦除了,在出口处进行身份恢复,一般都是加个显示类型转换,java1.5以后这些入口出口的工作都是我们在做的,1.5之后这些工作教给编译期来做,下面是加入了泛型的舞会代码。

import java.util.ArrayList;
import java.util.Random;

public class GenericMain {

    public static void main(String[] args) {
        MaskedBall maskedBall = new MaskedBall();
        //
        maskedBall.join(new ScienceBigShot());
        maskedBall.join(new GovernmtBigShot());
        //返回一个舞会的人要显式转型
        maskedBall.aPlay().play();
    }

}

//面具舞会类
class MaskedBall<T extends MaskPlayer> extends ArrayList<T>{
    private Random rd = new Random(24);
    public void join(T mp){
        //检查身份,只有是有邀请函的(MaskPlayer)的人才能进去,现在编译器自己就会帮我们进行类型检查
        this.add(mp);
        mp.play();
    }
    //随机返回一名舞者
    public T aPlay(){
        return this.get(rd.nextInt(this.size()));
    }
    
}

//参加舞会的人
class MaskPlayer{
    private String name;
    
    public MaskPlayer(){
        this.name = "无名";
    }
    
    
    public String getName() {
        return name;
    }
    
    public void play(){
        System.out.println("忘记烦恼,尽情玩");
    }
}

class ScienceBigShot extends MaskPlayer{
    private String source;
    public ScienceBigShot(){
        this.source = "科技大咖";
    }
    public String getSource() {
        return source;
    }
    public void play(){
        System.out.println(this.source+"(无名)忘记烦恼,尽情玩");
    }
}

class GovernmtBigShot extends MaskPlayer{
    private String source;
    public GovernmtBigShot(){
        this.source = "政府大咖";
    }
    public String getSource() {
        return source;
    }
    public void play(){
        System.out.println(this.source+"(无名)忘记烦恼,尽情玩");
    }

}

★这样我们就会省去很多代码,程序还不容易出错。

★泛型的类型擦除指的是java编译之后字节码文件中并没有存储具体的类型参数,比如List<String>,List<Integer>,字节码文件中只能看到List,String和Integer的信息都看不到,这就是被擦除了,它们都被替换为Object。

★T,?,类型擦除后被替换为Object。
★? extends x上界限定,不可输入,可输出,类型擦除后被替换为x。
★? super x下界限定,可输入x以及x的子类,不可输出,类型擦除后被替换为Object。

❤数组可以协变,集合不可以协变。
比如: Integer是Number的子类,则Integer[]是Number[]的子类
,但是List<Integer>不是List<Number>的子类。
数组的协变本身就是一种设计的缺陷,比如
Number[] ns = new Integer[10];
ns[0] = new Float(0.0);//error
你总不能像一个类型是Integer数组的数组里放一个Float。
集合修复了这个缺陷。但是为了有时候的必要,设计了上下界限定符。

★没有泛型数组
数组在创建的时候必须知道内部元素的类型,而且一直都会记得这个类型信息,每次往数组里添加元素,都会做类型检查。
但因为Java泛型是用擦除(Erasure)实现的,运行时类型参数会被擦掉。
所以,像List<String>[] l = new ArrayList<String>[10]; 这样的代码,运行时编译期只能看到ArrayList,看不到具体的类型参数。所以不允许创建泛型数组。

相关文章

  • 泛型琐碎之泛型上下限

    泛型的命名规范 为了更好地去理解泛型,我们也需要去理解java泛型的命名规范。 为了与java关键字区别开来,ja...

  • Java 泛型

    java 泛型 很多朋友对java的泛型不是很理解,很多文章写的已不是很清楚,这篇博客对java泛型进行 一个总结...

  • java泛型理解及应用

    内容: java泛型理解及应用

  • Java泛型教程

    Java泛型教程导航 Java 泛型概述 Java泛型环境设置 Java泛型通用类 Java泛型类型参数命名约定 ...

  • 泛型

    java 泛型详解-绝对是对泛型方法讲解最详细的,没有之一Java泛型深入理解加泛型面试数组的协变性与范型的不可变性

  • 反射获取泛型类型

    参考:步步理解 JAVA 泛型编程(一)

  • Java 泛型擦除原理

    问:请比较深入的谈谈你对 Java 泛型擦除的理解和带来的问题认识? 答:Java 的泛型是伪泛型,因为在编译期间...

  • java 泛型

    很多朋友对Java的泛型不是很理解,很多文章写的已不是很清楚,这篇博客对java泛型进行 一个总结。 泛型的转换:...

  • JDK1.5后增加了泛型,那么为什么要有泛型呢?我们该如何自定义

    本篇主要讲解java泛型的理解、集合中简单使用泛型、自定义泛型结构(包括类、接口、方法)。 一、什么是泛型? 通俗...

  • [Java菜鸟系列] 「林彪」教你"通配符泛型

    J007- [Java菜鸟系列] 「林彪」教你"通配符泛型" 菜鸟:Java方法参数中的泛型通配符要怎么理解?老湿...

网友评论

      本文标题:理解java的泛型

      本文链接:https://www.haomeiwen.com/subject/bvbgsftx.html