最近我一直在对我公司的数据库产品的写入性能做一些基准测试,我发现只需切换到64位JVM就可以实现20-30%的一致性提升。
我不允许详细介绍我们的产品,但基本上它是一个面向列的数据库,针对存储日志进行了优化。该基准测试涉及为其提供几千兆字节的原始日志,并计算分析它们并将其作为结构化数据存储在数据库中所需的时间。 CPU和I / O的处理非常繁重,但很难说它的比例很高。
关于设置的一些注意事项:
Processor: Xeon E5640 2.66GHz (4 core) x 2
RAM: 24GB
Disk: 7200rpm, no RAID
OS: RHEL 6 64bit
Filesystem: Ext4
JVMs: 1.6.0_21 (32bit), 1.6.0_23 (64bit)
Max heap size (-Xmx): 512 MB (for both 32bit and 64bit JVMs)
两个JVM的常量:
为简单起见,我已经关闭了产品中的所有多线程选项,因此几乎所有处理都以单线程方式进行。 (当我打开多线程时,当然系统变得更快,但32位和64位性能之间的比率保持不变。)
所以,我的问题是......为什么我在使用64位JVM时会看到20-30%的速度提升?有没有人见过类似的结果?
直到现在我的直觉如下:
64位指针更大,因此L1和L2缓存更容易溢出,因此64位JVM的性能更差。
JVM使用一些花哨的指针压缩技巧来尽可能地缓解上述问题。有关Sun网站的详细信息here。
在64位模式下运行时,允许JVM使用更多寄存器,这会略微提高速度。
鉴于以上三点,我预计64位性能会稍微慢一点,或大约等于32位JVM。
有什么想法吗?提前谢谢。
编辑:澄清了有关基准环境的一些观点。
答案 0 :(得分:10)
来自:http://www.oracle.com/technetwork/java/hotspotfaq-138619.html#64bit_performance
“通常,与32位VM上运行相同的应用程序相比,能够处理大量内存的好处在于64位虚拟机的性能损失很小。这是因为每个本地指针都是如此在系统中占用8个字节而不是4个。加载这些额外数据会对内存使用产生影响,这会导致执行速度稍慢,具体取决于在执行Java程序期间加载了多少指针。好消息是, AMD64和EM64T平台以64位模式运行,Java VM可以使用一些额外的寄存器来生成更高效的本机指令序列。这些额外的寄存器可以提高性能,使得在比较时通常没有性能损失32到64位执行速度 当您迁移到64位VM时,将64位平台上运行的应用程序与SPARC上的32位平台相比,性能差异大约为10-20%。在AMD64和EM64T平台上,这种差异范围为0-15%,具体取决于访问应用程序的指针数量。“
答案 1 :(得分:8)
在不知道你的硬件的情况下,我只是采取一些狂野的刺伤
strictfp
,数学差异可能是真正的杀手。答案 2 :(得分:2)
我最好的猜测,基于快速谷歌的32位与64位性能排行榜, 是64位I / O更有效。我想你做了很多I / O ......
如果在移动数据时涉及memcpy,那么复制long可能比整数更有效。
答案 3 :(得分:2)
64位指令集还有8个寄存器,这样可以使代码整体更快。
但是,由于现在的处理器大多等待内存或磁盘,我认为内存子系统或磁盘i / o在64位模式下可能更有效。
答案 4 :(得分:1)
意识到64位JVM并不是制作Java应用程序的神奇小精灵 走得更快64位JVM允许堆>>因此,只有4 GB才有意义 对于可以利用系统上的大量内存的应用程序 拥有它。
通常会有轻微的改善(由于某些硬件 某些平台上的优化)或轻微的退化(由于增加 指针大小)。一般来说,需要更少的GC - 但是 当他们确实发生时,他们可能会更长。
在可以使用增加的内存的内存数据库或搜索引擎中 用于缓存对象,从而避免IPC或磁盘访问将看到最大的 应用程序级改进。另外还有一个64位的JVM 允许你运行比32位更多的线程,因为 对于诸如线程堆栈之类的东西,有更多的地址空间 一般情况下,32位JVM的最大线程数是~1000但是~100000个线程和64位JVM。
但有些缺点:
64位JVM的其他问题是某些客户端
面向Java Plug-in和Java Web Start的功能
不受支持。此外,任何本机代码也需要
兼容(例如,类型II JDBC驱动程序的JNI)。
对于纯Java开发人员来说,这是纯粹的应用程序应该得到的奖励
刚刚开箱即用。
有关Java.net
的帖子的更多信息