众神进入瓦尔哈拉_一时冲动:“通往瓦尔哈拉之路的冒险”

众神进入瓦尔哈拉

通过所有有关Java 9和Project Jigsaw的讨论,我们不应忽视Java的另一重大变化。 希望在第10版或第11版中, Valhalla项目能够实现并介绍价值类型和专业化。

那么,这是什么一回事,项目进展如何,面临什么挑战? 几天前,Oracle的Java语言架构师和Valhalla的项目负责人Brian Goetz在JVM Language Summit 2015上的一次演讲中回答了这些问题。

我们来看一下。

总览

这篇文章将介绍Goetz演讲“冒险之路到Valhalla”的四个部分中的三个。

他以序言开头,我为那些尚不了解Valhalla项目的人提供了一些补充说明。 Goetz继续展示这两个原型,其中第一个原型于去年公开发布,而第二个原型仅在两周前发布。 由于这篇文章已经足够长了,因此我将不讨论他关于未来实验的最后一部分。 如果您觉得这个话题很有趣,那么一定要看整个演讲!

全文中的所有引用均来自幻灯片或逐字记录。

谈话

这里是谈话:

(顺便说一句,JVMLS团队在几个小时内就将所有讨论都在线上获得了赞誉!)

如果您可以节省50分钟,那就去看吧! 然后,无需阅读这篇文章。

要点

序幕

Valhalla项目涉及的两个主要主题是价值类型和通用专业化 。

值类型

前者将允许用户定义具有相同属性(如不变性,相等性而不是标识)的“类似int”的类型,以及由此产生的性能优势。 它们之前是Java 8的基于值的类 。

(除非另有说明,否则当本文章的其余部分讨论基元时,将包括值类型。)

通用专业化

随着每个人都声明自己的原始类型,泛型无法在它们之上工作的事实(即,没有ArrayList<int> )引起的问题变得令人无法忍受。 从概念的角度来看,虽然必须对原语进行装箱,但它具有显着的性能成本。

首先,存储对象而不是基元会花费额外的内存(例如,用于对象标头)。 然后,更糟的是,装箱会破坏缓存的局部性 。 当CPU缓存一个Integer数组时,它仅获取指向实际值的指针。 提取这些是额外的随机内存访问。 当CPU主要在等待高速缓存未命中时,这种额外级别的间接开销会付出巨大的代价,并可能破坏并行化。

因此,Valhalla项目的另一个目标是扩大参数多态性的范围,以使泛型能够覆盖基元。 为了获得成功,JVM应该使用基元而不是用于通用类中的通用字段,参数和返回值的框。

由于可能的实现方式,这称为通用专业化

因此,泛型需要与值类型很好地配合使用,并且原语可以伴随而来。

泛型的现状

由于擦除,类型变量被擦除到其边界,即ArrayList<Integer>实际上变为ArrayList<Object> (或者仅仅是ArrayList )。 这样的界限必须是所有可能实例化的超类型。 但是Java在基本类型和引用类型之上没有类型。

另外,JVM字节码指令通常是正交的,即沿同一行分割。 一个aloadastore只能移动引用。 专业变体具有用于原语,如iloadistoreint 。 没有字节码可以同时移动引用和int

因此,类型系统和字节码指令集都无法完成生成原语的任务。 十多年前开发出仿制药时,这已广为人知,作为妥协,决定只是不允许这样做。

今天的问题来自昨天的解决方案...

兼容性!

当然,Valhalla项目下发生的所有事情都必须向后兼容。 这有几种形式:

  • 二进制兼容性:现有字节码(即已编译的类文件)必须继续表示同一意思。 这样可以确保依赖项继续工作而无需重新编译。
  • 源代码兼容性:源文件必须继续具有完全相同的含义,因此重新编译它们一定不能“仅仅因为语言已更改”而更改任何内容。
  • 迁移可竞争性:来自不同Java版本的编译类必须协同工作,以允许一次迁移一个依赖项。

另一个要求是不要使JVM模仿太多的Java语言。 这样做将迫使其他JVM语言处理Java语言的语义。

原型模型1:使其工作

大约一年前,Goetz和他的同事展示了专业化的第一个实验性实现。

想法

在此原型中,编​​译器继续生成已擦除的类文件,但使用其他类型信息对其进行了扩充。

VM会忽略此信息,但将由专化器使用 ,这是类加载器的新组成部分。 后者将识别何时需要带有原始类型参数的类,并让专业化工具从已删除但已增强的类文件中即时生成它。

通过擦除,一个类的所有通用实例都使用相同的类文件。 相反,为每种原始类型创建一个新的类文件称为specialization

细节

在该原型中,使用“名称处理技术”描述了专门的类。 类名后面附加一个字符串,该字符串指示哪种类型参数专用于哪个原语。 例如, ArrayList${0=I}表示“用第一个类型变量int实例化的ArrayList ”。

在专业化期间,必须更改签名字节码。 为了正确地做到这一点,专门化者需要知道哪些Object出现(所有通用类型都被擦除了)必须专门化为哪种类型。 所需的签名信息已经大部分存在于类文件中,并且原型使用附加的类型元数据注释字节码。

从Goetz的8:44起 ,我们给出了几个如何进行的示例。 他还使用它们来指出这种实现必须要注意的一些细节,例如泛型方法的主题。

我知道那是很多快速的挥手。 关键是,这很简单,但是却有很多复杂性。

摘要

该实验表明,基于类文件元数据的即时专业化无需更改虚拟机即可工作。 这些都是重要的成就,但也有不利的弊端。

首先,它需要实现一组复杂的细节。

其次,也许是最重要的是,它具有有问题的类型系统特征。 在不更改VM的情况下,仍然没有intString公共超类型,因此也没有ArrayList<int>ArrayList<String>公共超类型。 这意味着无法声明“ ArrayList任何实例化”。

第三,这具有可怕的代码共享属性。 即使ArrayList<int>ArrayList<String>大部分代码是相同的,也可以在ArrayList${0=I}ArrayList复制它们。

死亡减少1000点。

原型模型2:抢救通配符

第二个非常新的原型解决了有问题的类型系统特征。

问题

当前,无界通配符表示“类的任何实例”,例如ArrayList<?>表示“任何ArrayList ”。 它们被大量使用,尤其是库开发人员。 在ArrayList<int>ArrayList<String>是不同类的系统中,通配符可能更为重要,因为它们弥合了它们之间的鸿沟并“表达了基本的ArrayList -ness”。

但是,如果我们假设ArrayList<?>ArrayList<int>的超类型,那么我们最终会遇到需要多个类继承的情况。 原因是ArrayList<T>扩展了AbstractList<T>所以我们也希望ArrayList<int>扩展AbstractList<int> 。 现在, ArrayList<int>将扩展ArrayList<?>AbstractList<int> (它们没有继承关系)。

在冒险途中,Valhalla的多重继承

(请注意,与当前泛型的区别在于擦除。在VM中, ArrayList<Integer>ArrayList<?>是同一类ArrayList,可以自由扩展AbstractList。)

根本原因是,虽然ArrayList<?>看起来像是“ any ArrayList ”,但实际上是ArrayList< ?。 扩展Object> ,即“引用类型上的任何ArrayList ”。

想法

原型引入了带有refvalany的通配符新层次结构:

  • ref包含所有引用类型并替换?
  • val包含所有原语和值类型(原型当前不支持该值,并且在谈话中未提及,但已在Valhalla邮件列表中宣布 )
  • any包含refval

专用类的多重继承将通过使用合成接口表示任意类型来解决。 ArrayList<int>将因此扩展AbstractList<int>并实现ArrayList<any>

细节

层次结构

ArrayList<ref> (即ArrayList<?> )将继续为已擦除类型。

为了表示ArrayList<any> ,编译器将创建一个接口ArrayList$any 。 它将由ArrayList生成的所有类(例如ArrayList<int>和已擦除的ArrayList )实现,并将扩展与超类相对应的所有综合接口,例如AbstractList$anyAbstractList<any>

在通往Valhalla任何接口的道路上冒险

该接口将包含该类的所有方法的声明以及其字段的访问器。 因为仍然没有对象和基元的公共超类型,所以必须将其通用参数和返回类型装箱。

但是,仅当以ArrayList<any>的方式访问该类时,才需要采取这种绕行的方法,而直接访问的对象是例如ArrayList<int> 。 因此,装箱的性能成本仅由使用通配符的开发人员承担,而使用原始专业化的代码直接获得期望的改进性能。

它干净利落地工作。

你不应该相信我,事情变得复杂了。 但这是个好故事。 我们会继续前进。

从Goetz的26:33开始,提供一些示例来解释一些细节。

辅助功能

可访问性是VM需要更改的区域。 到目前为止,接口不能具有私有或包可见的方法。 (在Java 9中,可以使用私有默认方法,但这在这里无济于事,因为需要实现。)

一个联系在一起但更老的问题是,即使VM不允许这样做,外部类及其内部类也可以彼此访问私有成员,因为VM不允许这样做,因为它们都是不相关的类。 当前,这是通过生成桥接方法(即具有较高可见性的方法,而不是无法访问的私有成员)进行调用来解决的。

为专门的类创建更多的桥接方法将是可能的,但是很费力。 相反,可能的更改是创建类嵌套的概念。 它包含所有专用类和内部类,并且VM允许访问嵌套内部的私有成员。

这将使语言的解释与VM的解释保持一致,VM的语言将所有专业化知识和内部类作为一个单元,而VM的解释直到现在只看到一堆不相关的类。

数组

通用方法也可以采用或返回数组。 但是,尽管专业化可以将一个int到一个Object上,但是int[]Object[]并且将每个单独的int装箱是一个糟糕的想法。

Arrays 2.0可能在这里有所帮助。 因为讨论需要基本熟悉该提案,所以我不会详细介绍。 总而言之,看起来他们将解决问题。

摘要

语言的更改在概念上很简单。 在没有任何变化的情况下。 类型变量可以用任何修饰,如果需要将实例分配给通配符类型,则通配符也必须使用任何通配符。

对于跨越原始类型和引用类型(例如ArrayList<any> )的泛型类,使用通用的超类,所得的编程模型更加合理。 在谈到他的团队将Stream API移植到此原型的经验时,Goetz说:

真的很顺利。 这正是您想要的。 大约70%的代码刚刚消失了,因为所有手工专门化的原始东西都消失了,然后许多支持手工专门化的复杂机制也消失了,三年级的学生可以成为一个简单的图书馆写。 因此,我们认为这是一个非常成功的实验。

与现有代码的兼容性也很好。

不幸的是,第一个原型的不良代码共享属性仍然存在。 ArrayList<int>ArrayList<String>仍然是不同的类,它们非常相似,但不共享任何代码。 下一部分(我将不在本文中介绍)解决该问题,并提出解决该问题的可能方法。

反射

谈话非常密集,涉及很多领域。 我们已经看到,值类型的引入和期望的性能改进需要通用专业化,因此可以减少甚至防止装箱。

第一个原型通过在装入类时专门化类来实现这一目标,而无需更改JVM。 但是,存在一个问题,即类的所有实例都没有通用的超类型,因为基本类型和引用类型参数会产生完全不相关的类。 第二个原型引入了通配符refvalany并使用合成接口来表示any-types。

这一切都非常令人兴奋,我等不及要尝试一下! 不幸的是,我要去度假,所以我不能一会儿了。 愚蠢的现实生活……别走了,别毁了一切!

翻译自: https://www.javacodegeeks.com/2015/08/impulse-adventures-on-the-road-to-valhalla.html

众神进入瓦尔哈拉

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

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

相关文章

当电子工程师十余年,感慨万千!

点击蓝字关注我们因公众号更改推送规则&#xff0c;请点“在看”并加“星标”第一时间获取精彩技术分享来源于网络&#xff0c;侵删当电子工程师也一些年头了&#xff0c;不算有出息&#xff0c;环顾四周&#xff0c;也没有看见几个有出息的。回顾工程师生涯&#xff0c;感慨万…

canva画图 图片居中裁剪_css实现不定宽高的图片img居中裁剪_类似微信朋友圈图片效果...

需求如下&#xff1a;前端需要显示矩形的缩略图&#xff0c;接口返回的图片尺寸大小不一&#xff0c;宽高不相等&#xff0c;需要前端来处理并显示成正方形&#xff0c;类似微信朋友圈图片的效果&#xff0c;等比例正方形显示在列表中&#xff0c;让图片根据宽高来自适应显示在…

哈希策略_优化哈希策略的简介

哈希策略总览 用于哈希键的策略可以直接影响哈希集合&#xff08;例如HashMap或HashSet&#xff09;的性能。 内置的哈希函数被设计为通用的&#xff0c;并且可以在各种用例中很好地工作。 我们可以做得更好&#xff0c;特别是如果您对用例有一个很好的了解吗&#xff1f; 测…

面试大全 | C语言高级部分总结,2.6万字长文

点击蓝字关注我们因公众号更改推送规则&#xff0c;请点“在看”并加“星标”第一时间获取精彩技术分享来源于网络&#xff0c;侵删一、内存大话题1.0、内存就是程序的立足之地&#xff0c;体现内存重要性。1.1、内存理解&#xff1a;内存物理看是有很多个 Bank&#xff08;就是…

c#设计12星座速配软件_C#设计模式(12)——组合模式

阅读目录1.组合模式在软件开发中我们经常会遇到处理部分与整体的情况&#xff0c;如我们经常见到的树形菜单&#xff0c;一个菜单项的子节点可以指向具体的内容&#xff0c;也可以是子菜单。类似的情况还有文件夹&#xff0c;文件夹的下级可以是文件夹也可以是文件。举一个例子…

hibernate与jpa_将JPA Hibernate与OptaPlanner集成

hibernate与jpa我们一直在改进OptaPlanner与JEE其余部分的集成&#xff0c;因此更容易构建可以正常工作的最终用户应用程序。 让我们看一下改进的JPA Hibernate集成。 基础 JPA Hibernate和OptaPlanner都可以在POJO&#xff08;普通的旧Java对象&#xff09;上工作&#xff0c…

程序如何运行,编译、链接、装入?

点击蓝字关注我们因公众号更改推送规则&#xff0c;请点“在看”并加“星标”第一时间获取精彩技术分享来源于网络&#xff0c;侵删一、地址概念和程序如何运行在多道程序环境下&#xff0c;要使程序运行&#xff0c;必须先为之创建进程。而创建进程的第一件事&#xff0c;便是…

python举两种字符串引号的例子_python里的单引号和双引号的有什么作用

在Python当中表达字符串既可以使用单引号&#xff0c;也可以使用双引号&#xff0c;那两者有什么区别吗&#xff1f;python单引号和双引号的区别简单来说&#xff0c;在Python中使用单引号或双引号是没有区别的&#xff0c;都可以用来表示一个字符串。但是这两种通用的表达方式…

枚举对象注释_如何以及何时使用枚举和注释

枚举对象注释本文是我们名为“ 高级Java ”的学院课程的一部分。 本课程旨在帮助您最有效地使用Java。 它讨论了高级主题&#xff0c;包括对象创建&#xff0c;并发&#xff0c;序列化&#xff0c;反射等。 它将指导您完成Java掌握的旅程&#xff01; 在这里查看 &#xff01;…

background 互联网图片_cssbackground-image和layer-background-image的区别

layer-background-image语法&#xff1a;layer-background-image : none | url (url)参数&#xff1a;none :  无背景图url :  使用绝对或相对地址指定背景图像说明&#xff1a;设置或检索对象整个区域的背景图像。示例&#xff1a;code {position: absolute;top: 100px; lef…

纪事本 乱码_纪事日记–可自定义的数据存储

纪事本 乱码总览 使任何数据结构或算法尽可能快的方法是使代码完全执行您想要的操作&#xff0c;而无需执行其他操作。 建立一个可以做任何人想做的每件事的数据存储的问题是&#xff0c;它做得特别不好。 自定义数据存储在性能方面可以实现什么&#xff1f; 您可以支持&#…

datavideo切换台说明书_【新品发布】datavideo SE-650 高清四通道切换台

还在为音乐演唱会的拍摄而烦恼吗&#xff1f;或者为体育比赛的多机位发愁&#xff1f;或者为微课、优课、慕课制作而焦头烂额&#xff1f;大部分用户对多机位的恐惧来源于相关产品令人发指的复杂和专业性&#xff0c;面对满眼的键盘会有无从下手之感&#xff0c;很多学校和工作…

NSA:建议从 C/C++ 切换到内存安全语言

点击蓝字关注我们因公众号更改推送规则&#xff0c;请点“在看”并加“星标”第一时间获取精彩技术分享来源于网络&#xff0c;侵删美国国家安全局&#xff08;NSA&#xff09;发布了一份指南&#xff0c;旨在帮助软件开发商和运营商预防和缓解软件内存安全问题。其鼓励组织将编…

探索cqrs和事件源_编写基于事件的CQRS读取模型

探索cqrs和事件源关于事件源和CQRS的讨论似乎通常集中在CQRS上下文中的整体系统架构或领域驱动设计的各种形式。 但是&#xff0c;尽管也有一些有趣的考虑&#xff0c;但读取模型经常被忽略。 在本文中&#xff0c;我们将介绍通过使用事件流来填充视图模型的示例实现。 总览 …

shmmax单位_kernel.shmmax ,kernel.shmmni 和kernel.shmall

kernel.shmmax 2147483648//该参数定义了共享内存段的最大尺寸(以字节为单位)。其值应>sag_max_size初始化参数的大小,否则SAG由多个内存段构成,效率降低,还要不小于物理内存的一半,默认情况下在32位x86系统中,OracleSGA最大不能超过1.7GB.缺省为32M&#xff0c;对于oracle…

C# 11正式发布

点击蓝字关注我们因公众号更改推送规则&#xff0c;请点“在看”并加“星标”第一时间获取精彩技术分享来源于网络&#xff0c;侵删C# 11 现已发布。公告称&#xff0c;“随着每个版本的发布&#xff0c;社区的参与度越来越高&#xff0c;贡献了从建议、见解和错误报告到整个功…

java 自定义运算符_Java中集合的自定义运算符

java 自定义运算符总览 操作员重载有多种语言可用。 Java对String类型的运算符的支持对运算符的重载非常有限。 我们可以利用其他语言支持运算符的不同方式&#xff0c;但是我们可以在Java中实现一个使用Java已经使用的约定的实现。 获取&#xff0c;设置和放置操作 集合的运…

marker主题 ros_(五)ROS主题理解

参考网址&#xff1a;1&#xff0c;小海龟例子(1) 在新的终端打开roscore$ roscore&#xff0d;&#xff0d;&#xff0d;如果出错&#xff0c;请确定关闭所有ROS命令或者路径&#xff0c;重试。(2) 在新的终端打开运行小海龟界面$ rosrun turtlesimturtlesim_node得到结果&…

分享一个通用的嵌入式驱动层

点击蓝字关注我们因公众号更改推送规则&#xff0c;请点“在看”并加“星标”第一时间获取精彩技术分享来源于网络&#xff0c;侵删C 语言面向对象编程的最佳实践一、前言以STM32为例&#xff0c;打开网络上下载的例程或者是购买开发板自带的例程&#xff0c;都会发现应用层中会…

cks32和stm32_cks子,间谍,局部Mo子和短管

cks32和stm32本文是我们名为“ 用Mockito测试 ”的学院课程的一部分。 在本课程中&#xff0c;您将深入了解Mockito的魔力。 您将了解有关“模拟”&#xff0c;“间谍”和“部分模拟”的信息&#xff0c;以及它们相应的存根行为。 您还将看到使用测试双打和对象匹配器进行验证…