- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
根据this在旧的java版本中,有一个限制较少的invokespecial版本,称为invokenonvirtual。该指令将允许您调用实例方法而无需虚拟查找,如果我没猜错的话,这些方法可能属于当前类的不可分配类。使用invokespecial,这种情况发生了变化,因为它仅用于调用 super 方法、私有(private)方法或初始化方法。
我的问题是:java 8(或更高版本)有没有办法绕过这些结构限制 [4.9.2]的invokespecial指令并在没有虚拟查找的情况下调用不同类的方法(即与当前类不兼容赋值的类)
我可以使用 no-verify 标志来禁用验证过程,但是有更优雅的方法吗?
最佳答案
您正在混淆两个不同的问题。 invokenonvirtual
和 inokespecial
是相同的,并且一直如此。它们只是同一操作码的两个不同名称。
您链接的问题正在谈论一个单独的功能ACC_SUPER
,它有自己漫长而复杂的历史。
基本上,在 Java 的早期版本中, super 调用的编译被破坏了。编译父类(super class)调用时,它会在编译时将 invokespecial
插入父类(super class)方法。如果稍后更改类层次结构,即使在层次结构的中间点插入新的重写,它仍会尝试调用它被编译要调用的方法。
请注意,这仍然不允许您调用不相关的类中的方法 - 该问题仅与编译后在同一继承更改中添加的同一方法的不同重写有关。
Java 作者意识到他们的错误后,他们更新了 JVM 对 invokespecial
的处理,以正确处理 super 调用。现在,无论指令中指定哪个方法,它都会在链接/运行时遍历父类(super class)层次结构并调用适当的方法。
但是,他们担心在旧版本 Java 下编译的代码依赖于损坏的行为,因此为了向后兼容,他们添加了一个新的类文件标志 ACC_SUPER
。如果在类上设置了该标志,则它具有新的(正确的)行为,如果没有,则它使用旧的行为。由于旧的类文件是在该标志存在之前编译的,因此它们不会设置它。同时,编译器已更新为在所有新类上设置 ACC_SUPER ,每个人都很高兴...
截至 2011 年。事实证明,java.lang.Thread
有一个用户不应调用的安全敏感方法。为了防止人们从 Thread 的子类调用它,他们重写了它以引发异常。然而,someone realized黑客可以在没有 ACC_SUPER 标志的情况下定义 Thread 的子类,从而跳过安全检查并调用该方法的危险版本并突破 Java 沙箱。
不幸的是,修复此安全漏洞的唯一方法是完全删除 ACC_SUPER
功能 - 也就是说,将每个类视为设置了标志,无论实际上是否设置了该标志。它在 Java 7 update 13 中被匆忙修复,而在 Java 8 中,规范本身已更改为记录 ACC_SUPER
不再产生任何效果。
所以答案是否定的,在 7u13 之后的任何 Java 版本中,或者实现安全修复的任何更新中,都没有办法获得 super 调用的旧行为。
关于java - 具有旧语义的 Invokespecial : call instance method of other class,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/53807175/
我知道我的问题有点含糊,但我不知道如何描述它。我问过很多地方,但似乎没有人理解我为什么要这样做。但请耐心等待,我会解释为什么我想要这样的东西。 我使用 Liquid Templates 允许用户在我的
这个问题在这里已经有了答案: what is the difference between null != object and object!=null [duplicate] (2 个回答) 7年
当我在我的本地主机 Google App Engine 应用程序中将日志记录级别更改为 FINE 时,我开始在我的跟踪堆栈中看到这些: Apr 17, 2013 4:54:20 PM com.goog
Python 有内置函数 type : class type(object) With one argument, return the type of an object. The return v
我正在使用深度学习进行语义分割,我遇到了以下术语:语义分割、实例检测、对象检测 和对象分割. 它们有什么区别? 最佳答案 这些术语的某些用法对用户而言是主观的或依赖于上下文,但据我所知对这些术语的合理
我面临 -[NSConcreteMutableData release] 的问题:消息发送到已释放的实例,我也附上了我的示例代码。 - (IBAction)uploadImage { NSString
我试图显示模型中的单个实例(数据库行),其中多个实例共享多行的相同字段(列)值。为了澄清这一说法,我有以下情况: ID/Title/Slug/Modified 1 Car A 1s ag
我正在尝试使用mockito来模拟服务。然而,我没有找到一种方法来告诉mockito,给定一个类的实例返回给我相同的实例: 类似于: given(service.add(any(Individua
我知道如何从父类(super class)原型(prototype)创建子类原型(prototype)。但是,如果我已经有了父类(super class)对象的实例来创建子类对象怎么办? 在 JS 中
鉴于 Kotlin 1.1。对于某个类的 instance,instance::class.java 和 instance.javaClass 似乎几乎是等价的: val i = 0 println(
这个问题在这里已经有了答案: 8年前关闭。 Possible Duplicate: Find out the instance id from within an ec2 machine 我正在寻找从
为什么我的 Instantiate 函数没有创建 That 的“空白”实例? 我有以下最小类: classdef That < handle properties This = '' end
Session session = HibernateUtil.getSessionFactory().openSession(); Transaction tx = session.beginTra
考虑以下几点: public class A { public String name = "i am a A instance"; } public class B extends A {
我正在使用 Scalr 来扩展网站服务器。 在 Apache 服务器上,我安装了 Sakai,并为 Linux 机器创建了一个启动脚本。 问题是,如何确保MySQL实例在Apache服务器启动之前启动
Android Realm DB 允许使用 Realm.getInstance() 获取多个实例。这些中的最佳实践是什么? :1.创建单个实例(应用程序范围)并在任何地方使用它2. 在需要时获取一个新
我很难理解为什么修改实例 a 中的属性会修改实例 b 中的相同属性。 var A = function (){ }; A.prototype.data = { value : 0 }; var
我将 Weka 用作更长管道的一部分,因此,我无法承受将所有数据写入文件或数据库只是为了创建一个 Instances。目的。我可以即时做的是创建 Instance 的列表对象。 来自 this pag
class C: def func(self, a): print(a) c = C() print(c.__dict__) # {} c.func = c.func # c.func i
Angular Routing 文档提到了组件实例创建、组件实例激活和路由激活。 文档没有解释这些概念的区别,以及每次创建/激活发生的时间。 问题 实例创建和实例激活有什么区别? 实例激活和路由激活有
我是一名优秀的程序员,十分优秀!