百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 技术文章 > 正文

[JAVA冷知识]什么是逆变与协变?数组是否支持协变&逆变?泛型呢?

haoteby 2024-12-22 18:19 9 浏览

写在前面


  • 和小伙伴分享一些java小知识点,主要围绕下面几点:
  • 什么是逆变(contravariant)&协变(covariant)?
  • 数组支持协变&逆变吗?
  • 泛型支持协变&逆变吗?
  • 部分内容参考 《编写高质量代码(改善Java程序的151个建议)》
  • 博文理解有误的地方小伙伴留言私信一起讨论

生活不能等待别人来安排,要自己去争取和奋斗;而不论其结果是喜是悲,但可以慰藉的是,你总不枉在这世界上活了一场。有了这样的认识,你就会珍重生活,而不会玩世不恭;同时,也会给人自身注入一种强大的内在力量。 ——路遥《平凡的世界》


关于协变逆变到底是什么意思,其实很好理解,用一句话描述:(小伙伴们看到下面的话,会不会想到这不就是多态吗,哈,今天我们只看协变和逆变,关于多态的一些内容,如强制多态包含多态重载多态等之后有机会和小伙伴们分享)

协变 即指窄类型替换宽类型逆变宽类型覆盖窄类型

这里的窄类型子类(派生类),这里的宽类型父类(基类,超类),那这里的替换覆盖又是什么意思,这里就要说到OO(面相对象)六大设计原则之一的LSP(里氏代换原则 Liskov Substitution Principle),里氏代换原则中说,任何基类可以出现的地方,子类一定可以出现。LSP是继承复用的基石,只有当派生类可以替换掉基类,且软件单位的功能不受到影响时,基类才能真正被复用,而派生类也能够在基类的基础上增加新的行为

下面们结合代码就数组和泛型的协变和逆变进行分析

我们来看一段代码

package com.liruilong;

import java.util.Arrays;

/**
 * @Project_name: workspack
 * @Package: com.liruilong
 * @Description:
 * @Author: 1224965096@qq.com
 * @WeChat_Official_Accounts: 山河已无恙
 * @blog: https://liruilong.blog.csdn.net/
 * @Date: 2022/2/11  1:18
 */
public class CovariantDemo {

    public static void main(String[] args) {
        Number [] numbers = {1,1L,3d,2.0F};
        Arrays.stream(numbers).forEach(System.out::print);
    }
}

Number类是所有基本类型封装类的父类,同理基本类型封装类为Number类的子类,关于自动装箱和自动拆箱是java在JDK1.5的时候引入的新特性,我们这里不多讲,上面的代码可以正常编译,并且输出下面的内容,这里,数组里的基本类型装箱为封装类放到了堆中,这些封装类可以出现在Number类定义的数组中,说明子类可以替换了父类,即数组是满足协变的。

113.02.0
Process finished with exit code 0

既然数组支持协变,那么逆变呢?我们来看看

package com.liruilong;

import java.util.Arrays;

/**
 * @Project_name: workspack
 * @Package: com.liruilong
 * @Description:
 * @Author: 1224965096@qq.com
 * @WeChat_Official_Accounts: 山河已无恙
 * @blog: https://liruilong.blog.csdn.net/
 * @Date: 2022/2/11  1:18
 */
public class CovariantDemo {

    public static void main(String[] args) {
        Number [] numbers = {new Object()};
        Arrays.stream(numbers).forEach(System.out::print);
    }
}

这里我们把数组元素换成Object类,即所有类的父类,希望是可以通过父类来覆盖代替子类,但是直接编译报错,说明数组不支持直接逆变

Error:(17, 30) java: 不兼容的类型: java.lang.Object无法转换为java.lang.Number

数组不支持直接逆变,那么是否可以接见的实现逆变的,这里我么就要用到多态里的一种,强制多态,即强制类型转化试试

package com.liruilong;

import java.util.Arrays;

/**
 * @Project_name: workspack
 * @Package: com.liruilong
 * @Description:
 * @Author: 1224965096@qq.com
 * @WeChat_Official_Accounts: 山河已无恙
 * @blog: https://liruilong.blog.csdn.net/
 * @Date: 2022/2/11  1:18
 */
public class CovariantDemo {
    class A {

    }
    class B extends A{
      
    }

    public static void main(String[] args) {
        A a = new CovariantDemo().new A();
        B [] bs = {(B) a};
        Arrays.stream(bs).forEach(System.out::print);
    }
}

类型转化报错。说明对于数组的逆变来讲,是不支持逆变的,将父类强制转化为子类报类型转化异常,java并没有对这方面做限制。

Exception in thread "main" java.lang.ClassCastException: com.liruilong.CovariantDemo$A cannot be cast to com.liruilong.CovariantDemo$B
 at com.liruilong.CovariantDemo.main(CovariantDemo.java:24)

Process finished with exit code 1

通过上面代码,我们可以知道数组支持协变,不支持逆变,那泛型呢?对于协变和逆变是否支持

泛型不支持协变也不支持逆变,即不能把一个父类对象赋值给一个子类类型变量,相反也是同理。

下面我们看看代码

package com.liruilong;

import java.util.ArrayList;
import java.util.List;

/**
 * @Project_name: workspack
 * @Package: com.liruilong
 * @Description:
 * @Author: 1224965096@qq.com
 * @WeChat_Official_Accounts: 山河已无恙
 * @blog: https://liruilong.blog.csdn.net/
 * @Date: 2022/2/11  1:18
 */
public class CovariantDemo {
    public static void main(String[] args) {
        List<Number> ln = new ArrayList<Integer>();
    }
}

java为了保证运行期安全性,必须保证泛型参数类型固定的,所以它不允许一个泛型参数可以同时包含两种类型,即使为父子关系也不行。所以直接编译报错,即泛型不支持协变也不支持逆变.

Error:(17, 27) java: 不兼容的类型: java.util.ArrayList<java.lang.Integer>无法转换为java.util.List<java.lang.Number>

但可以使用通配符(Wildcard)模拟协变逆变,通配符在编译期有效,在运行期必须为一个明确的类型

package com.liruilong;

import java.util.ArrayList;
import java.util.List;

/**
 * @Project_name: workspack
 * @Package: com.liruilong
 * @Description:
 * @Author: 1224965096@qq.com
 * @WeChat_Official_Accounts: 山河已无恙
 * @blog: https://liruilong.blog.csdn.net/
 * @Date: 2022/2/11  1:18
 */
public class CovariantDemo {
    public static void main(String[] args) {
        List< ? extends Number > list = new ArrayList<Integer>();
    }
}

Number的子类型都可以为泛型类型参数,即允许NUmber所有的子类作为泛型参数类型,在运行期为一个具体的值.编译没有报错


Process finished with exit code 0

逆变同样也是可以,即泛型可以通过superextends来模拟实现协变和逆变,但是本身是不存在协变和逆变的,这里主要利用了泛型在编译器有效

List< ? super Integer> li = new ArrayList<Number>();

关于协变逆变就和小伙伴分享到这里,嗯,还有协变逆变方法,这里要简单说明下

协变方法: 即子类的方法返回值的类型比父类方法要窄,即该方法为协变方法,也称多态,覆写,重写

//子类的doStuff()方法返回值的类型比父类方法要窄,即该方法为协变方法,也称多态。
    class A{
        public  Number doStuff(){
            return 0;
        }
    }
    class B extends A{
        @Override
        public  Integer doStuff(){
            return 0;
        }
    }

逆变方法:子类的方法返回值的类型比父类方法宽,此时为逆变方法。虽然子类扩大了父类的输入返回参数,但是这里已经是重载了。

//子类的doSutff方法返回值的类型比父类方法宽,此时为逆变方法,
    class C {
        public Integer doStuff(Integer i) {
            return 0;
        }
    }
    class D extends C {
        public Number doStuff(Number i) {
            return 0;
        }
    }

相关推荐

一日一技:用Python程序将十进制转换为二进制

用Python程序将十进制转换为二进制通过将数字连续除以2并以相反顺序打印其余部分,将十进制数转换为二进制。在下面的程序中,我们将学习使用递归函数将十进制数转换为二进制数,代码如下:...

十进制转化成二进制你会吗?#数学思维

六年级奥赛起跑线:抽屉原理揭秘。同学们好,我是你们的奥耀老师。今天一起来学习奥赛起跑线第三讲二进制计数法。例一:把十进制五十三化成二进制数是多少?首先十进制就是满十进一,二进制就是满二进一。二进制每个...

二进制、十进制、八进制和十六进制,它们之间是如何转换的?

在学习进制时总会遇到多种进制转换的时候,学会它们之间的转换方法也是必须的,这里分享一下几种进制之间转换的方法,也分享两个好用的转换工具,使用它们能够大幅度的提升你的办公和学习效率,感兴趣的小伙伴记得点...

c语言-2进制转10进制_c语言 二进制转十进制

#include<stdio.h>intmain(){charch;inta=0;...

二进制、八进制、十进制和十六进制数制转换

一、数制1、什么是数制数制是计数进位的简称。也就是由低位向高位进位计数的方法。2、常用数制计算机中常用的数制有二进制、八进制、十进制和十六进制。...

二进制、十进制、八进制、十六进制间的相互转换函数

二进制、十进制、八进制、十六进制间的相互转换函数1、输入任意一个十进制的整数,将其分别转换为二进制、八进制、十六进制。2、程序代码如下:#include<iostream>usingna...

二进制、八进制、十进制和十六进制等常用数制及其相互转换

从大学开始系统的接触计算机专业,到现在已经过去十几年了,今天整理一下基础的进制转换,希望给还在上高中的表妹一个入门的引导,早日熟悉这个行业。一、二进制、八进制、十进制和十六进制是如何定义的?二进制是B...

二进制如何转换成十进制?_二进制如何转换成十进制例子图解

随着社会的发展,电器维修由继电器时代逐渐被PLC,变频器,触摸屏等工控时代所替代,特别是plc编程,其数据逻辑往往涉及到数制二进制,那么二进制到底是什么呢?它和十进制又有什么区别和联系呢?下面和朋友们...

二进制与十进制的相互转换_二进制和十进制之间转换

很多同学在刚开始接触计算机语言的时候,都会了解计算机的世界里面大多都是二进制来表达现实世界的任何事物的。当然现实世界的事务有很多很多,就拿最简单的数字,我们经常看到的数字大多都是十进制的形式,例如:我...

十进制如何转换为二进制,二进制如何转换为十进制

用十进制除以2,除的断的,商用0表示;除不断的,商用1表示余0时结束假如十进制用X表示,用十进制除以2,即x/2除以2后为整数的(除的断的),商用0表示;除以2除不断的,商用1表示除完后的商0或1...

十进制数如何转换为二进制数_十进制数如何转换为二进制数举例说明

我们经常听到十进制数和二进制数,电脑中也经常使用二进制数来进行计算,但是很多人却不清楚十进制数和二进制数是怎样进行转换的,下面就来看看,十进制数转换为二进制数的方法。正整数转二进制...

二进制转化为十进制,你会做吗?一起来试试吧

今天孩子问把二进制表示的110101改写成十进制数怎么做呀?,“二进制”简单来说就是“满二进一”,只用0和1共两个数字表示,同理我们平常接触到的“十进制”是“满十进一”,只用0-9共十个数字表示。如果...

Mac终于能正常打游戏了!苹果正逐渐淘汰Rosetta转译

Mac玩家苦转译久矣!WWDC2025苹果正式宣判Rosetta死刑,原生游戏时代终于杀到。Metal4光追和AI插帧技术直接掀桌,连Steam都连夜扛着ARM架构投诚了。看到《赛博朋克2077》...

怎么把视频的声音提出来转为音频?音频提取,11款工具实测搞定

想把视频里的声音单独保存为音频文件(MP3/AAC/WAV/FLAC)用于配音、播客、听课或二次剪辑?本文挑出10款常用工具,给出实测可复现的操作步骤、优缺点和场景推荐。1)转换猫mp3转换器(操作门...

6个mp4格式转换器测评:转换速度与质量并存!

MP4视频格式具有兼容性强、视频画质高清、文件体积较小、支持多种编码等特点,适用于网络媒体传播。如果大家想要将非MP4格式的视频转换成MP4的视频格式的话,可以使用MP4格式转换器更换格式。本文分别从...