JVM 区分 32 位和 64 位重要吗?

 2022-09-26

简介

在本文中找到有关于32位JVM和64位JVM之间的区别的一些常见问题的答案。你将学到在这两者之间进行迁移时要考虑的所有事情,将有助于你更好地实施。

32位JVM和64位JVM之间存在一些明显的区别和细微差别。我将尝试通过此问答文章来澄清它们。

我们是否需要了解32位JVM和64位JVM之间的区别?

如果你不是在构建性能型Java服务,则不必了解它们之间的区别。32位JVM和64位JVM之间的细微差别不会对你的Java服务产生太大影响。

64位JVM的性能是否优于32位JVM?

我们大多数人认为64位大于32位,因此64位JVM性能将优于32位JVM性能。其实不然,事实并非如此。与32位JVM相比,64位JVM的性能可能会有所下降。以下是Oracle JDK文档中有关64位JVM性能的摘录:

通常,与在32位VM上运行同一应用程序相比,能够处理大量内存的好处是64位VM的性能损失较小。当你迁移到64位VM时,将在64位平台上运行的应用程序与在SPARC上的32位平台运行的应用程序相比,性能差异降低了10-20%。在AMD64和EM64T平台上,此差异范围为0-15%,具体取决于访问应用程序所执行的指针的数量。”

如果性能受到影响,为什么有人会使用64位JVM?

在32位JVM中,最大可寻址内存空间仅为2 ^ 32(即〜4gb)。这意味着Java进程的最大内存大小不能超过4GB。实际上,由于各种其他限制(例如可用交换,内核地址空间使用,内存碎片和VM开销),该限制要低得多。下表总结了可以在32位JVM上设置的最大堆大小(即-Xmx):

202209262238105811.png

而如果你在64位JVM上运行应用程序,则最大可寻址内存空间为2 ^ 64(即... 没算出来--_--)。这意味着你的应用程序的最大可寻址内存大小接近无限。
`

为什么64位JVM性能可能比32位JVM慢

这是由于以下事实:系统中的每个本机指针占用的是8个字节而不是4个字节。此额外数据的加载对内存使用有影响,这取决于执行过程中加载了多少个指针,从而导致执行速度稍慢。

好消息是,由于以64位模式运行的AMD64和EM64T平台,Java VM获得了一些额外的寄存器,可用于生成更有效的本机指令序列。比较32位和64位执行速度时,这些额外的寄存器将性能提高到通常根本没有性能损失的地步。

从32位JVM迁移到64位JVM时需要考虑哪些事项?

GC Stop World时间

从32位JVM迁移到64位JVM的主要原因是要获得较大的堆大小(即-Xmx)。当增加堆大小时,GC暂停时间会自动开始变长,因为现在内存中有更多垃圾需要清除。在进行迁移之前,你需要进行适当的GC调整,否则,你的应用程序可能会经历几秒钟到几分钟的暂停时间。于此同时,有一个专门的GC分析网站可以供你去配置使用GCeasy

Native library

如果你的应用程序使用Java本机接口(JNI)访问本机库,那么你还需要升级Native library,因为32位JVM只能使用32位Native library。同样,64位JVM只能使用64位Native library。

什么是CompressedOops?它与32位,64位JVM有关吗?

202209262238122502.png

是的,CompressedOOps与32位和64位JVM有关。

我们用数据字段定义对象。当在内存中创建该对象以及数据字段时,还将创建对象标头。JVM需要对象标头来执行内部处理,虚拟方法调用,垃圾回收,锁定等。

在32位JVM中,此对象标头占用8个字节;在64 位JVM中,该对象标头占用16个字节。增加8个字节听起来可能并不多,但是鉴于你的应用程序在其运行时创建了数百万个对象,将8个字节乘以数百万个对象可能会增加可观的开销。

你可以通过传递XX:+ UseCompressedOops JVM参数来缓解此问题。当你传递此参数时,JVM会做出一个巧妙的技巧,并将对象标头的大小优化为即使在64位中也仅使用12个字节。只要你的JVM堆大小(即-Xmx)小于32GB,此技巧就会起作用。如果超过32 GB,则对象标头大小将再次变为16个字节。

注意: 自Java SE 6u23及更高版本以来,已将-XX:+ UseCompressedOops设置为默认值。仅当你在JDK 6u23或更早版本上运行时,才传递-XX:+ UseCompressedOopsargument。

什么时候应该使用32位和64位JVM?

<2GB内存 :如果你的应用程序的堆大小(即-Xmx)小于2GB,则无需费力地决定。使用32位JVM。

> 2GB内存 :如果你的应用程序需要2GB以上的内存,那也不是明智的决定。使用64位JVM。但是,请进行适当的性能测试以衡量和减轻影响。

如何查找我的应用程序是在32位还是64位JVM上运行?

方法一:在命令提示符下,发出命令:

    java -version

如果是64位JVM,你将看到输出包含单词:“64-Bit”。例:

    java version "1.8.0_181"
    Java(TM) SE Runtime Environment (build 1.8.0_181-b13)
    Java HotSpot(TM) 64-Bit Server VM (build 25.181-b13, mixed mode)

如果它是32位JVM,则不会显示单词:“64-Bit”。例:

    java version "1.8.0_211"
    Java(TM) SE Runtime Environment (build 1.8.0_211-b12)
    Java HotSpot(TM) Client VM (build 25.211-b12, mixed mode)

方法二:你从Java程序发出以下语句:

    System.out.println(System.getProperty("sun.arch.data.model") + "-bit JVM");

根据JVM类型,适当的版本将被打印在控制台上。

我可以在64位操作系统上运行32位JVM吗?

有32位OS和64位OS。如果你在32位操作系统上运行(这在今天很难找到),则只能运行32位JVM。另一方面,如果你在64位操作系统上运行,则可以在32位JVM或64位JVM上运行应用程序。

如何下载32位JVM或64位JVM?

转到Oracle JDK下载页面时,将看到用于下载特定于你的操作系统的JDK的选项:

202209262238131803.png

如果你选择x86,则将在此处下载32位JVM。如果选择x64,则将下载64位JVM。

在32位JVM上编译的代码可以在64位JVM上运行吗?

我们使用javac即java编译器将Java代码编译为字节代码(即* .class文件)。生成的字节码与32位和64位JVM无关。它可以在两个JVM上运行。记住Java古老的诺言:“Write once, run anywhere”。