java泛型类的继承规则

首先看一看java泛型类的使用:

/**
 * 一个泛型方法:使程序更加安全
 * 并且能被更多的使用
 * @author 丁**
 *
 * @param <T>
 */
class Pair<T>{
    private T first;
    private T second;

    //实例化类型变量
    public static<T> Pair<T> makePair(Class<T> cl){
        try {
            return new Pair<>(cl.newInstance(), cl.newInstance());
        } catch (Exception e) {
            return null;
        }
    }

    public Pair(){ first = null; second = null; }
    public Pair(T first, T second){ this.first = first; this.second = second;}

    public T getFirst() {
        return first;
    }
    public void setFirst(T first) {
        this.first = first;
    }
    public T getSecond() {
        return second;
    }
    public void setSecond(T second) {
        this.second = second;
    }

}
class Father {
    private int age = 40;
    public void getName(){
        System.out.println(age);
    }
}

class Son extends Father{
    private int age = 12;
    public void getName() {
        System.out.println(age);
    }
}

在普通类中:Father aa = new Son();父类是可以用来指向子类的

但是在泛型类中却不是如此:

Pair<Son> bb = new Pair<>();
Pair<Father> cc = bb;//error

1:虽然Son是Father的子类,但是Pair<T>之间没有继承关系:

List<Object> a1 = new ArrayList<>();
List<String> a2 = new ArrayList<>();
//设a1可以等于a2
//a1 = a2;
//a1.add(1111);因为是对a1进行操作,所以可以添加Object,a1.add(Object ad);
//a2.get(0);报错,因为a1添加Object进入了a2的空间中,但是a2是String类型,所以报错

2:可以将参数化类型转换成一个原始类型:

List<String> a2 = new ArrayList<>();
List a3 = a2;//可以通过编译,但是后面使用方法时可能会产生类型错误!
//这时候a3对象时原始类型,所以add(Object obj);
a3.add(123);//是对a3进行操作,但是最终结果保存到了a2中,将一个Integer装入String中
显然是错误的;

3:泛型类可以扩展或实现其他的泛型类:

//泛型接口
interface List1<E>{
}

//实现了泛型接口的泛型类
class List2<T, E> implements List1<E>{

}

//泛型类
class List3<T>{

}

//继承了其他泛型类的泛型类
class List4<T, E> extends List3<E>{

}
List1<Father> b1 = new List2<Son, Father>();//因为List2实现了List1
List3<Father> b2 = new List4<Son, Father>();//List4继承了List3,所以List3是父类,可以指向子类对象。

虽然这样也完成了泛型类的继承,实现了和普通类一样的多态,但是使用起来并不是特别好,就这样java引入了通配符概念:

通配符上限:

/*
通配符的上限:Pair<? extends Father> c2
extends是关键字
Pair<T>代表的是某个唯一(具体的泛型类)的泛型类:比如Pair<Son>,Pair<Father>
但是Pair<? extends Father>不是具体的泛型类,它所指的是参数类型为Father的子类的所有泛型类(包括Father)
*/
Pair<? extends Father> c3;
Pair<Father> c1 = new Pair<>();
Pair<Son> c2 = new Pair<>();
//c2 = c1;error
c3 = c1;
c3 = c2;

需要注意的是:

/*
* 使用通配符的上限的问题:
* ? extends Father getFirst();
* void setFirst(? extends Father);
* 当c2= c1时:
* c2.setFirst(Father father);时,会将Father对象添加到Son对象内存中,这是不好的
* 所以使用extends上限时,不能使用setFirst(? extends Father),add(? extends Father)* 等方法。
* 但可以使用getFirst();方法
*/c2.setFirst( new Father(); );//error

通配符下限:

/*
* 通配符的下限:
* ? super Son
* 表示的不是某个具体的泛型类,而是表示参数类型为Son的父类的所有可能的泛型类(包括* * Son)* 和通配符上限一样,通配符的超类型限定不能使用getFirst()方法,但可以使用setFirst(XX)方法
*/
Pair<Son> c1 = new Pair<>();
Pair<? super Son> c3;
c3 = c1;

//另一种超类型限定的写法Pair<T extends Comparable<? super T>> c4;

 无限定通配符:

//无限定通配符
Pair<Son> c7 = new Pair<>();
Pair<String> c5 = new Pair<>();
//c7 = c5;error,因为他们不是同一种类型
Pair<?> c6 = new Pair<>();
c6 = c5;//Pair<?>是所有的Pair泛型类的父类,Pair<?> c6 = new Pair<xx>();

Pair<?>和Pair的本质不同在于:可以用任意的Object对象调用原始的Pair类的setObject()方法;

通配符的捕获:

//交换First,Second变量值
    public static void swap(Pair<?> p){
        ? t = p.getFirst();//error,因为通配符(?)不是类型变量,所以不能直接将?写入代码中,利用通配符的捕获来解决这个问题。
        p.setFirst(p.getSecond());
        p.setSecond(t);
    }
//交换First,Second变量值
    public static void swap(Pair<?> p){
        swapHelper(p);//在调用下面的方法时,类型参数就被捕获了。
    }

    //利用通配符的捕获来解决该问题
    public static <T> void swapHelper(Pair<T> p){
        T t = p.getFirst();//T是具体的某个类型。
        p.setFirst(p.getSecond());
        p.setSecond(t);
    }

注意:通配符的捕获只有在许多限制的情况下才是合法的,编译器必须能够确信通配符表达的是单个,确定的类型。

原文地址:https://www.cnblogs.com/dtx0/p/8466127.html

时间: 2024-08-29 20:27:23

java泛型类的继承规则的相关文章

Java面向对象之继承

Java 中的继承规则: 1.子类继承父类所有的成员变量和成员方法,但是不能继承父类的构造方法. 2.子类虽然继承了父类的成员变量,但是子类不能直接访问父类的私有变量,可以通过getter/setter()方法进行访问 3.子类对父类构造函数的调用规则: a.子类的构造方法首先必须调用父类的构造方法. b.如果没有显示指定,子类的构造方法会默认的调用父类中的无参构造方法, 1 public class Animal { 2 public Animal() { 3 System.out.print

Java泛型 泛型类型的继承规则

泛型类型的继承规则可能不能凭直觉的来想当然.下面举个例子来看看泛型类型的继承规则. 首先有几个辅助类: package generic; public class Person extends Animal { private String name; public Person(String name) { super(); this.name = name; } public String getName() { return name; } public void setName(Stri

Java中的继承、封装、多态的理解

Java中的继承.封装.多态 继承的理解: 1.继承是面向对象的三大特征之一,也是实现代码复用的重要手段.Java的继承具有单继承的特点,每个子类只有一个直接父类. 2.Java的继承通过extends关键字来实现,实现继承的类被称为子类,被继承的类称为父类(有的也称其为基类.超类),父类和子类的关系,是一种一般和特殊的关系.就像是水果和苹果的关系,苹果继承了水果,苹果是水果的子类,水果是苹果的父类,则苹果是一种特殊的水果. 3.Java使用extends作为继承的关键字,extends关键字在

Java面向对象的继承

继承也是面向对象的又一重要特性,继承是类于类的一种关系,通俗来说狗属于动物类,那么狗这个类就继承了动物类 java中的继承是单继承的,一个类只能继承与一个父类 子类继承父类之后,子类就拥有了父类的所有属性和方法,private的除外,优点就是可以提高代码的复用性,简单的继承实例如下: 1 public class Dog extends Animal { 2 //Dog类继承了Animal类中的所有非私有的属性和方法,可以直接使用了 3 } 使用继承还是挺简单的 方法重写 如果子类对继承父类的方

黑马程序员——Java基础——面向对象——继承、抽象类、接口、多态、包、内部类、异常等

第一讲 继承 1.继承: 当多个类拥有相同的功能时,那么这些类不需要每个都写这些相同的功能,只需要把相同功能抽到 一个单独的类中,继承这个类就能获得这些相同的功能: (1)继承的体系结构:就是对要描述的事物进行不断的向上抽取,就出现了体系结构. 要了解这个体系结构中最共性的内容,就看最顶层的类. 要使用这个体系的功能,就用最底层的类创建对象 (2)好处: A.提高代码复用性: B.继承的出现,让类与类之间产生关系,为多态的前提 (3)特点: A.只能单继承(准确的说是java对多继承进行优化,避

JAVA中的继承和覆盖

java里面的继承是子类继承父类的一些方法和属性(除private属性和方法之外):对于父类的私有属性和方法子类是没有继承的:但是要想子类也能访问到父类的私有属性,必须给私有属性以外界访问的方法接口. package com.PengRong; public class Demo { public static void main(String[] args) { /* Pupil stu1 = new Pupil(); stu1.pay(500f); stu1.ShowName(); Syst

java方法的继承,覆盖与重载

java中的继承使用extends关键字,在子类继承了父类之后将会获得父类的全部属性与方法(父类的构造器除外).如果在定义java类时没有显示定义她的父类,那么这个类默认将扩展java.lang.Object类.因此java.lang.Object是所有类的父类,要么是直接父类,要么就是间接父类. 方法的覆盖(Overridden Methods)    在类继承中,子类可以修改从父类继承来的行为,也就是说子类能创建一个与父类方法有不同功能的方法,但具有相同的:名称.返回类型.参数列表.如果在新

java容器类的继承结构

摘要: java容器类的继承结构 Java容器类库定义了两个不同概念的容器,Collection和Map Collection 一个独立元素的序列,这些元素都服从一条或多条规则.List必须按照插入的顺序保存元素.Set不能有重复元素.Queue按照排队规则来确定对象产生的顺序. (文中Jdk源码版本无特殊说明均为jdk1.8.0_101) public interface Collection<E> extends Iterable<E> { 可以看到,java定义了Collec

Java基础笔记-继承、抽象类、接口、多态、内部类、异常

继承:多个类具有相同的属性和行为时,将这些内容单独抽取到一个单独的类中,那么多个类无需再定义这些属性和行为,直接继承那个类即可 1)  多个类称为子类,单独的类称为父类或者超类 2)  子类可以继承父类中非私有的属性和方法 3)  通过extends关键字让类与类之间建立继承关系 4)  继承的出现提高了代码的复用性,让类与类之间产生了关系,为多态的出现提供了前提 5)  不能为了实现某类的功能,简化代码而继承,必须是类与类之间具有所属关系才可以继承,所属关系:is a 6)  特点:java只