泛型型协变逆变_Java泛型类型简介:协变和逆变

泛型型协变逆变

by Fabian Terh

由Fabian Terh

Java泛型类型简介:协变和逆变 (An introduction to generic types in Java: covariance and contravariance)

种类 (Types)

Java is a statically typed language, which means you must first declare a variable and its type before using it.

Java是一种静态类型的语言,这意味着您必须先声明一个变量及其类型,然后才能使用它。

For example: int myInteger = 42;

例如: int myInteger = 42;

Enter generic types.

输入通用类型。

通用类型 (Generic types)

Definition: “A generic type is a generic class or interface that is parameterized over types.”

定义 :“ 泛型类型是通过类型进行参数化的泛型类或接口。”

Essentially, generic types allow you to write a general, generic class (or method) that works with different types, allowing for code re-use.

本质上,泛型类型允许您编写可与不同类型一起使用的泛型通用类(或方法),从而可以重复使用代码。

Rather than specifying obj to be of an int type, or a String type, or any other type, you define the Box class to accept a type parameter <;T>. Then, you can use T to represent that generic type in any part within your class.

您可以将Box类定义为接受类型参数< ; T>,而不是将obj指定为int类型, String类型或任何其他类型。 然后,CA n用t代表在类中的任何一部分泛型类型。

Now, enter covariance and contravariance.

现在,输入协方差和自变量。

协方差和自变量 (Covariance and contravariance)

定义 (Definition)

Variance refers to how subtyping between more complex types relates to subtyping between their components (source).

差异是指更复杂类型之间的子类型如何与其组件( 源 )之间的子类型相关。

An easy-to-remember (and extremely informal) definition of covariance and contravariance is:

协方差和协方差的一个易于记忆(非常非正式)的定义是:

  • Covariance: accept subtypes

    协方差:接受子类型
  • Contravariance: accept supertypes

    矛盾:接受超类型

数组 (Arrays)

In Java, arrays are covariant, which has 2 implications.

在Java中, 数组是covariant ,有2个含义。

Firstly, an array of type T[] may contain elements of type T and its subtypes.

首先,类型为T[]的数组可以包含类型T及其子类型的元素。

Number[] nums = new Number[5];nums[0] = new Integer(1); // Oknums[1] = new Double(2.0); // Ok

Secondly, an array of type S[] is a subtype of T[] if S is a subtype of T.

其次,类型的数组S[]是的子类型T[]如果S是的子类型T

Integer[] intArr = new Integer[5];Number[] numArr = intArr; // Ok

However, it’s important to remember that: (1) numArr is a reference of reference type Number[] to the “actual object” intArr of “actual type” Integer[].

但是,重要的是要记住:(1) numArr是引用类型Number[]对“ actual type” Integer[]的“ actual object” intArr

Therefore, the following line will compile just fine, but will produce a runtime ArrayStoreException (because of heap pollution):

因此,以下行将编译得很好,但将生成运行时ArrayStoreException (由于堆污染):

numArr[0] = 1.23; // Not ok

It produces a runtime exception, because Java knows at runtime that the “actual object” intArr is actually an array of Integer.

它产生一个运行时异常,因为Java在运行时知道“实际对象” intArr实际上是Integer的数组。

泛型 (Generics)

With generic types, Java has no way of knowing at runtime the type information of the type parameters, due to type erasure. Therefore, it cannot protect against heap pollution at runtime.

对于泛型类型,由于类型擦除,Java无法在运行时知道类型参数的类型信息。 因此,它无法在运行时防止堆污染。

As such, generics are invariant.

因此,泛型是不变的。

ArrayList<Integer> intArrList = new ArrayList<>();ArrayList<Number> numArrList = intArrList; // Not okArrayList<Integer> anotherIntArrList = intArrList; // Ok

The type parameters must match exactly, to protect against heap pollution.

类型参数必须完全匹配,以防止堆污染。

But enter wildcards.

但是输入通配符。

通配符,协方差和逆方差 (Wildcards, covariance, and contravariance)

With wildcards, it’s possible for generics to support covariance and contravariance.

使用通配符,泛型有可能支持协方差和协方差。

Tweaking the previous example, we get this, which works!

调整前面的示例,我们得到了这个,它有效!

ArrayList<Integer> intArrList = new ArrayList<>();ArrayList<? super Integer> numArrList = intArrList; // Ok

The question mark “?” refers to a wildcard which represents an unknown type. It can be lower-bounded, which restricts the unknown type to be a specific type or its supertype.

问号“?” 表示代表未知类型的通配符。 它可以是下界的,这将未知类型限制为特定类型或其超类型。

Therefore, in line 2, ? super Integer translates to “any type that is an Integer type or its supertype”.

因此,在第2行, ? super Integer ? super Integer转换为“任何为Integer类型或其超类型的类型”。

You could also upper-bound the wildcard, which restricts the unknown type to be a specific type or its subtype, by using ? extends Integer.

您还可以使用? extends Integer来限制通配符的范围,通配符将未知类型限制为特定类型或其子类型? extends Integer ? extends Integer

只读和只写 (Read-only and write-only)

Covariance and contravariance produce some interesting outcomes. Covariant types are read-only, while contravariant types are write-only.

协方差和自变量产生一些有趣的结果。 协变类型是只读的,而协变类型是只写的。

Remember that covariant types accept subtypes, so ArrayList<? extends Number> can contain any object that is either of a Number type or its subtype.

还记得协变类型接受子类型ArrayList<? extends Numb ,所以ArrayList<? extends Numb ArrayList<? extends Numb ER>可以包含任何对象,或者是of a数字类型或子类型。

In this example, line 9 works, because we can be certain that whatever we get from the ArrayList can be upcasted to a Number type (because if it extends Number, by definition, it is a Number).

在此示例中,第9行有效,因为我们可以确定从ArrayList中获得的任何内容都可以转换为Number类型(因为如果它扩展Number ,顾名思义,它就是 Number )。

But nums.add() doesn’t work, because we cannot be sure of the “actual type” of the object. All we know is that it must be a Number or its subtypes (e.g. Integer, Double, Long, etc.).

但是nums.add()不起作用,因为我们不能确定对象的“实际类型”。 我们所知道的是它必须是一个Number或其子类型(例如Integer,Double,Long等)。

With contravariance, the converse is true.

在相反的情况下,反之亦然。

Line 9 works, because we can be certain that whatever the “actual type” of the object is, it must be Integer or its supertype, and thus accept an Integer object.

第9行行得通,因为我们可以确定,无论对象的“实际类型”是什么,它都必须是Integer或它的超类型,因此可以接受Integer对象。

But line 10 doesn’t work, because we cannot be sure that we will get an Integer. For instance, nums could be referencing an ArrayList of Objects.

但是第10行不起作用,因为我们无法确定是否会获得Integer 。 例如, nums可以引用Objects的ArrayList。

应用领域 (Applications)

Therefore, since covariant types are read-only and contravariant types are write-only (loosely speaking), we can derive the following rule of thumb: “Producer extends, consumer super”.

因此,由于协变量类型是只读的,而协变量类型是只写的(松散地说),我们可以得出以下经验法则: “生产者扩展,消费者超级”

A producer-like object that produces objects of type T can be of type parameter <? extends T>, while a consumer-like object that consumes objects of type T can be of type parameter <? super T>.

产生类型T对象的类似生产者的对象可以是参数<? extends <? extends T>,而消费类对象消费 T型可以是参数meter <? 超级T>。

翻译自: https://www.freecodecamp.org/news/understanding-java-generic-types-covariance-and-contravariance-88f4c19763d2/

泛型型协变逆变

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/393103.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

安卓系统换成linux系统软件,将旧安卓手机打造成“简易linux”机器,并部署AdGuardHome...

从原教程的安装Linux Deploy 完成后&#xff0c;在配置 Linux下载镜像的一些东西时有些许出入。首先&#xff0c;我是用的下载源地址是 http://mirrors.tuna.tsinghua.edu.cn/ubuntu-ports 清华的源挺好用的。 其他有出入的配置如图(记得把源地址改清华的&#xff0c;华中科大…

let与expr命令的用法与实战案例

let命令的用法 格式&#xff1a; let 赋值表达式 【注】let赋值表达式功能等同于&#xff1a;&#xff08;赋值表达式&#xff09; 例子&#xff1a;给自变量i加8 12345678[rootXCN ~]# i2 [rootXCN ~]# let ii8 [rootXCN ~]# echo $i 10[rootXCN ~]# ii8 #去掉let定义 [root…

在使用ToolBar + AppBarLayout,实现上划隐藏Toolbar功能,遇到了一个坑。

问题&#xff1a;Android5.0以下版本Toolbar不显示沉浸式状态栏&#xff0c;没有这个问题&#xff0c;但是5.0以上版本&#xff0c;就出现了莫名其妙的阴影问题&#xff0c;很是头疼。 分享一下我的解决方案&#xff1a; 在AppBarLayout中加一个属性&#xff1a; app:elevation…

leetcode1476. 子矩形查询

请你实现一个类 SubrectangleQueries &#xff0c;它的构造函数的参数是一个 rows x cols 的矩形&#xff08;这里用整数矩阵表示&#xff09;&#xff0c;并支持以下两种操作&#xff1a; updateSubrectangle(int row1, int col1, int row2, int col2, int newValue) 用 new…

msbuild构建步骤_如何按照以下步骤构建最终的AI聊天机器人

msbuild构建步骤by Paul Pinard保罗皮纳德(Paul Pinard) 如何按照以下步骤构建最终的AI聊天机器人 (How to build the ultimate AI chatbot by following these steps) 快速指南&#xff0c;可帮助您避免常见的陷阱 (A quick guide that helps you avoid common pitfalls) Bui…

第一章:最小可行区块链

概览区块数据结构区块哈希创世块创建区块保存区块链验证区块完整性选择最长链节点间通信操作节点架构运行测试小结概览 区块链的基础概念非常简单, 说白了就是一个维护着一个持续增长的有序数据记录列表的这么一个分布式数据库。在此章节中我们将实现一个简单的玩具版的区块链。…

Oracle Controlfile控制文件中记录的信息片段sections

初学Oracle的朋友肯定对Controlfile控制文件中到底记录了何种的信息记录而感到好奇&#xff0c;实际上我们可以通过一个视图v$controlfile_record_section来了解控制文件的信息片段&#xff1a; SQL> select type, record_size, records_total from v$controlfile_record_s…

linux 怎么禁止遍历目录,linux下遍历目录功能实现

/*编译:dir:dir.cgcc -o $ $<*/#include #include #include #include #include int do_search_dir(char *path);int do_check_dir(char *fullpath, char* truefullpath);void usage(char *apps);int count 0;intmain(int argc,char **argv){char fullpath[…

leetcode面试题 16.26. 计算器(栈)

给定一个包含正整数、加()、减(-)、乘(*)、除(/)的算数表达式(括号除外)&#xff0c;计算其结果。 表达式仅包含非负整数&#xff0c;&#xff0c; - &#xff0c;*&#xff0c;/ 四种运算符和空格 。 整数除法仅保留整数部分。 示例 1: 输入: “32*2” 输出: 7 代码 clas…

团队项目电梯会议视频

http://v.youku.com/v_show/id_XMjcyMjI3Mjk2NA.html?spma2hzp.8244740.userfeed.5!2~5~5~5!3~5~A转载于:https://www.cnblogs.com/jingxiaopu/p/6749776.html

arduino服务器_如何使用Arduino检查Web服务器的响应状态

arduino服务器by Harshita Arora通过Harshita Arora 如何使用Arduino检查Web服务器的响应状态 (How to use Arduino to check your web server’s response status) Last year, I created Crypto Price Tracker (an app which was acquired by Redwood City Ventures this yea…

leetcode486. 预测赢家(dp)

给定一个表示分数的非负整数数组。 玩家 1 从数组任意一端拿取一个分数&#xff0c;随后玩家 2 继续从剩余数组任意一端拿取分数&#xff0c;然后玩家 1 拿&#xff0c;…… 。每次一个玩家只能拿取一个分数&#xff0c;分数被拿取之后不再可取。直到没有剩余分数可取时游戏结束…

linux怎么看文件状态,linux查看文件类型-file、状态-stat

linux查看文件类型-file、状态-stat首页 计算机相关 linux命令 linux查看文件类型-file、状态-statfile 命令可以用来查看文件类型-i mime type-s 读取字符或块设备文件最好指定[root192 tmp]# file freeclsfreecls: UTF-8 Unicode text[root192 tmp]# file -i freeclsfreecls:…

Linux课程笔记 Crond介绍

1. 定时任务比较及cron语法 Linux的任务调度可以分为两类&#xff1a; 系统自身执行的任务用户执行的工作Linux系统下另外两种定时任务软件&#xff1a; at&#xff1a;适合仅执行一次的调度任务&#xff0c;需要启动一个名为atd的服务 anacron&#xff1a;这个命令主要用于非…

Python 学习日记第二篇 -- 列表,元组

一、列表 列表是一个可以包含所以数据类型的对象的位置有序集合&#xff0c;它是可以改变的。 1、列表的序列操作&#xff08;Python3&#xff09; 123456789101112131415161718192021222324>>> one_list [1,2,3,4]>>> two_list ["jonny","…

【Gamma】PhyLab 测试报告

PhyLab Gamma测试报告 测试中发现的bug Gamma阶段新Bug Bug可能原因部分错误码设置与原先抛异常的逻辑冲突原先代码中使用了一些特殊的办法处理异常Beta未发现Bug Bug可能原因控制台新建实验编号不能以0开头后端处理编号会将其前导0去除&#xff0c;以数字形式存储&#xff0c;…

如何使用Node.js,Express和MongoDB设置GraphQL服务器

by Leonardo Maldonado莱昂纳多马尔多纳多(Leonardo Maldonado) 如何使用Node.js&#xff0c;Express和MongoDB设置GraphQL服务器 (How to set up a GraphQL Server using Node.js, Express & MongoDB) 从GraphQL和MongoDB开始的最直接的方法。 (The most straightforward…

leetcode954. 二倍数对数组(treemap)

给定一个长度为偶数的整数数组 A&#xff0c;只有对 A 进行重组后可以满足 “对于每个 0 < i < len(A) / 2&#xff0c;都有 A[2 * i 1] 2 * A[2 * i]” 时&#xff0c;返回 true&#xff1b;否则&#xff0c;返回 false。 示例 1&#xff1a; 输入&#xff1a;[3,1,…

linux文件内容打印成二进制,如何在二进制文件中只打印可打印字符(相当于Linux下的字符串)?...

在Python3中&#xff0c;以二进制模式打开文件会得到bytes的结果。迭代一个bytes对象可以得到0到255(包括0到255)的整数&#xff0c;而不是字符。从^{} documentation&#xff1a;While bytes literals and representations are based on ASCII text, bytes objects actually b…

1098 均分纸牌

1098 均分纸牌 2002年NOIP全国联赛提高组 时间限制: 1 s 空间限制: 128000 KB 题目等级 : 黄金 Gold 题目描述 Description有 N 堆纸牌&#xff0c;编号分别为 1&#xff0c;2&#xff0c;…, N。每堆上有若干张&#xff0c;但纸牌总数必为 N 的倍数。可以在任一堆上取若于张纸…