- mongodb - 在 MongoDB mapreduce 中,如何展平值对象?
- javascript - 对象传播与 Object.assign
- html - 输入类型 ="submit"Vs 按钮标签它们可以互换吗?
- sql - 使用 MongoDB 而不是 MS SQL Server 的优缺点
Possible Duplicate:
Does use of final keyword in Java improve the performance?
final 修饰符有 different consequences在java中取决于你应用它。我想知道的是,如果 另外 它可能会帮助编译器创建更高效的字节码。我想这个问题深入到 JVM 是如何工作的,并且可能是 JVM 特定的。
那么,根据您的专业知识,以下任何一项对编译器有帮助吗,还是仅出于正常的 java 原因使用它们?
谢谢!
编辑:感谢您的所有回答!请注意,正如@Zohaib 建议的那样,我的问题与this 重复。 .发帖前我搜索的不够好。我不会删除它,因为你们做出了很好的贡献,但答案可以合并。除非另有说明,否则我将让“投票赞成”系统决定。
最佳答案
如果您使用 final
,字节码的效率并不会显着提高或降低,因为 Java 字节码编译器通常很少做优化。效率奖励(如果有)将在 JIT 编译器生成的 native 代码中1。
理论上,使用 final
可为 JIT 编译器提供一个提示,帮助其优化。在实践中,最近的 HotSpot JIT 编译器可以通过忽略您的提示来做得更好。例如,现代 JIT 编译器通常会执行全局分析,以确定给定的方法调用是否是对应用程序当前加载的类的上下文中的叶方法的调用。此分析比您的 final
提示更准确,运行时甚至可以检测何时加载了使分析无效的新类......并为受影响的代码重做分析和 native 代码生成.
使用final
还有其他语义后果:
final
可防止您意外更改它。 (并向读者表达您的意图。)final
可防止在子类中被覆盖。final
完全防止子类化。final
会阻止子类对其进行更改。final
对线程安全有重要影响;见 JLS 17.5 .在适当的情况下,这些都是好的。但是,很明显,它们通过创建子类限制了您对重用的选择。在决定是否使用final
时需要考虑这一点。
所以好的做法是使用 final
来(广义上)表达您的设计意图,并实现您需要的其他语义效果。如果您将 final
仅仅用作优化提示,您将不会有太多收获。
有几个异常(exception)情况,final
可能会在某些平台上导致小的性能提升。
在某些情况下,将字段声明为 final
会改变字节码编译器处理它的方式。我在上面给出了一个例子。另一种是“常量变量”情况 (JLS 4.12.4),其中 static final
字段的值 将由字节码编译器在当前类和其他类中内联,这可能会影响观察到的代码行为。 (例如,引用常量变量不会触发类初始化。因此,添加 final
可能会改变类初始化的顺序。)
可以想象,将字段或局部参数声明为 final
可能会允许较小的 JIT 编译器优化,否则不会进行。但是,任何可以声明为 final 的字段也可以被 JIT 编译器推断为有效的 final。 (只是不清楚 JIT 编译器是否真的这样做,以及这是否会影响生成的 native 代码。)
但底线保持不变。您应该使用 final
来表达您的设计意图,而不是作为优化提示。
1 - 这个答案假设我们正在谈论一个具有良好 JIT 或 AOT 编译器的最新 JVM。 1) 最早的 Sun Java 实现根本没有 JIT 编译器。 2) 早期的 Android Java 实现的编译器在优化方面做得很差。事实上,早期的 Android 开发人员文档建议使用各种源级微优化来进行补偿。此建议已被删除。
关于java - java final 是否有助于编译器创建更高效的字节码?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/8354412/
美好的一天!我试图添加两个字节变量并注意到奇怪的结果。 byte valueA = 255; byte valueB = 1; byte valueC = (byte)(valueA + valueB
嗨,我是 swift 的新手,我正在尝试解码以 [Byte] 形式发回给我的字节数组?当我尝试使用 if let string = String(bytes: d, encoding: .utf8)
我正在使用 ipv4 和 ipv6 存储在 postgres 数据库中。 因为 ipv4 需要 32 位(4 字节)而 ipv6 需要 128(16 字节)位。那么为什么在 postgres 中 CI
我很好奇为什么 Go 不提供 []byte(*string) 方法。从性能的角度来看,[]byte(string) 不会复制输入参数并增加更多成本(尽管这看起来很奇怪,因为字符串是不可变的,为什么要复
我正在尝试为UDP实现Stop-and-Wait ARQ。根据停止等待约定,我在 0 和 1 之间切换 ACK。 正确的 ACK 定义为正确的序列号(0 或 1)AND消息长度。 以下片段是我的代码的
我在下面写了一些代码,目前我正在测试,所以代码中没有数据库查询。 下面的代码显示 if(filesize($filename) != 0) 总是转到 else,即使文件不是 0 字节而是 16 字节那
我使用 Apache poi 3.8 来读取 xls 文件,但出现异常: java.io.IOException: Unable to read entire header; 0 by
字典大小为 72 字节(根据 getsizeof(dict) 在字典上调用 .clear() 之后发生了什么,当新实例化的字典返回 240 字节时? 我知道一个简单的 dict 的起始大小为“8”,并
我目前正在努力创建一个函数,它接受两个 4 字节无符号整数,并返回一个 8 字节无符号长整数。我试图将我的工作基于 this research 描述的方法,但我的所有尝试都没有成功。我正在处理的具体输
看看这个简单的程序: #include using namespace std; int main() { unsigned int i=0x3f800000; float* p=(float*)(
我创建了自己的函数,将一个字符串转换为其等效的 BCD 格式的 bytes[]。然后我将此字节发送到 DataOutputStram (使用需要 byte[] 数组的写入方法)。问题出在数字字符串“8
此分配器将在具有静态内存的嵌入式系统中使用(即,没有可用的系统堆,因此“堆”将只是“char heap[4096]”) 周围似乎有很多“小型内存分配器”,但我正在寻找能够处理非常小的分配的一个。我说的
我将数据库脚本从 64 位系统传输到 32 位系统。当我执行脚本时,出现以下错误, Warning! The maximum key length is 900 bytes. The index 'U
想知道 128 字节 ext2 和 256 字节 ext3 文件系统之间的 inode 数据结构差异。 我一直在为 ext2、128 字节 inode 使用此引用:http://www.nongnu.
我试图理解使用 MD5 哈希作为 Cassandra key 在“内存/存储消耗”方面的含义: 我的内容(在 Java 中)的 MD5 哈希 = byte[] 长 16 个字节。 (16 字节来自维基
检查其他人是否也遇到类似问题。 shell脚本中的代码: ## Convert file into Unix format first. ## THIS is IMPORTANT. ###
我们有一个测量数据处理应用程序,目前所有数据都保存为 C++ float,这意味着在我们的 x86/Windows 平台上为 32 位/4 字节。 (32 位 Windows 应用程序)。 由于精度成
我读到在 Java 中 long 类型可以提升为 float 和 double ( http://www.javatpoint.com/method-overloading-in-java )。我想问
我有一个包含 n 个十进制元素的列表,其中每个元素都是两个字节长。 可以说: x = [9000 , 5000 , 2000 , 400] 这个想法是将每个元素拆分为 MSB 和 LSB 并将其存储在
我使用以下代码进行 AES-128 加密来编码一个 16 字节的 block ,但编码值的长度给出了 2 个 32 字节的 block 。我错过了什么吗? plainEnc = AES.enc
我是一名优秀的程序员,十分优秀!