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);
}
注意:通配符的捕获只有在许多限制的情况下才是合法的,编译器必须能够确信通配符表达的是单个,确定的类型。
java泛型类的继承规则的更多相关文章
- JAVA封装、继承
封装 1.概念: 将类的某些信息隐藏在类的内部,不允许外部程序访问,而是通过该类提供的方法来实现对隐藏信息的操作和访问. 2.好处: a.只能通过规定的方法访问数据 b.隐藏类的实例细节,方便修改和实 ...
- Java中的继承
我们在以前的学习中,我们会了C#中的继承,今天我们来了解了解Java中的继承,其实都大同小异啦! 1.语法 修饰符 SubClass extends SuperClass(){ //类定义部分 } e ...
- Java面向对象的继承
继承也是面向对象的又一重要特性,继承是类于类的一种关系,通俗来说狗属于动物类,那么狗这个类就继承了动物类 java中的继承是单继承的,一个类只能继承与一个父类 子类继承父类之后,子类就拥有了父类的所有 ...
- JAVA中的继承和覆盖
java里面的继承是子类继承父类的一些方法和属性(除private属性和方法之外):对于父类的私有属性和方法子类是没有继承的.可是要想子类也能訪问到父类的私有属性,必须给私有属性以外界訪问的方法接口. ...
- java类的继承,多态,抽象类与接口
知识点梳理: 1,怎样定义自己的类. MyStarFrame,MyStarPanel 类中定义: (1)属性(数据),变量. (2)方法(函数),行为. (3)构造方法(特征,作用,何时被调用 ...
- java类的继承的一些细节
类的继承是java面向对象体系的一个重要方面(封装.继承.多态),对于java类的继承,需要注意如下细节. 1.构造函数. 如果一个类没有任何构造函数,系统会默认分配一个无参的构造函数给它,这个构造函 ...
- java方法的继承,覆盖与重载
java中的继承使用extends关键字,在子类继承了父类之后将会获得父类的全部属性与方法(父类的构造器除外).如果在定义java类时没有显示定义她的父类,那么这个类默认将扩展java.lang.Ob ...
- Java 类设计----Java类的继承
Java类的继承 为描述和处理个人信息,定义类Person: public class Person { public String name; public inat age; public Dat ...
- Java中的继承、封装、多态的理解
Java中的继承.封装.多态 继承的理解: 1.继承是面向对象的三大特征之一,也是实现代码复用的重要手段.Java的继承具有单继承的特点,每个子类只有一个直接父类. 2.Java的继承通过extend ...
随机推荐
- 从零开始学习前端开发 — 12、CSS3弹性布局
一.分栏布局 1.设置栏数column-count:数值; 2.设置每栏的宽度column-width:数值+单位; 注:当设置了column-width,column-count会失效,二者设置其一 ...
- dedecms_插件
../dede/adbaoming.php../dede/baoming_edit.php../dede/templets/baoming_main.htm
- 把要写的内容做个list,半年过去了
先说点题外话,来国家气象局的第三周,今天出去和丁书记吃了个饭,基本在聊大学时候的人和事,好像都没怎么变,只是不联系的人愈发不会再联系. 抛开学校的课程不谈,半年做了三件事:CUMT校园导航 / Ope ...
- asm文件开头的assume意义
body, table{font-family: Consolas; font-size: 13.5pt} table{border-collapse: collapse; border: solid ...
- 2018年web前端学习路线图
前端的的技术一直在变化,更新和变革,现在基本是三驾马车(vue,angualr,react)主导整个前端框架,但是无论对于新人或者有经验的程序员,这些知识在必须掌握 前端必会技能 上图罗列了整个前端的 ...
- maven插件本地化安装
mvn install:install-file -Dfile="D:\maven\repository\com\tc\itfarm-api\1.0.0-SNAPSHOT\itfarm-ap ...
- libev学习笔记
转 libev的使用--结合Socket编程 作者:cxy450019566 之前自己学过一些libev编程的基础,这次写压测刚好用上了,才算真正动手写了些东西,在这里做一些总结.写这篇文章是为了用浅 ...
- tomcat监控(二)
标签: linux 笔者Q:972581034 交流群:605799367.有任何疑问可与笔者或加群交流 这里介绍二种监控Tomcat的方法 使用windows版本的jdk监控 使用zabbix监控 ...
- 浅谈TreeMap以及在java中的使用
treemap结构是红黑树 1.先介绍一下平衡二叉树 其特点是一棵空树或它的左右两个子树的高度差的绝对值不超过1,并且左右两个子树都是一棵平衡二叉树.也就是说该二叉树的任何一个子节点,其左右子树的高度 ...
- Hyperledger Fabric Endorsement policies——背书策略
背书策略 背书策略用于指导peer如何确定交易是否得到了的认可.当一个peer接收到一个事务时,它会调用与事务的Chaincode相关联的VSCC(验证系统链代码),作为事务验证流程的一部分,以确定交 ...