Java求余%操作引發(fā)的一連串故事
操作符%通常用在正整數(shù)上,但同樣可以用在負(fù)整數(shù)和浮點(diǎn)數(shù)上。
注意:只有當(dāng)被除數(shù)是負(fù)數(shù)時(shí), 余數(shù)才是負(fù)的。
C1 RCE對(duì)%的處理
HotSpot VM的C1有個(gè)RCE(Range Check Elimination,范圍檢查消除)優(yōu)化,所謂范圍檢查消除,就是為了正確的拋出數(shù)組越界異常,虛擬機(jī)需要在數(shù)組訪問(wèn)的一些地方插入隱式的檢查,但是這些檢查會(huì)降低性能,比如在循環(huán)中每次循環(huán)都得檢查一次,所以HotSpot VM會(huì)想辦法在可能的地方消除這些檢查。我在看C1 RCE的時(shí)候發(fā)現(xiàn)目前它對(duì)求余符號(hào)的支持較為薄弱,它只能處理形如下面的代碼:
arr[x%arr.length] // 只有除數(shù)是x.length的時(shí)候,才能應(yīng)用RCE優(yōu)化
如果余數(shù)是整數(shù)常量,它就不能工作了:
arr[x%3]
for(int i=0;i<10;i++){
arr[x%10]
}
實(shí)際上,根據(jù)JLS的定義,我們知道如果除數(shù)為整數(shù)常量(且等于零,因?yàn)?作為除數(shù)會(huì)拋出運(yùn)行時(shí)異常),是可以推導(dǎo)出結(jié)果的上下界的(也取決于被除數(shù)的正負(fù)),規(guī)則如下:
- x % -y ==> [0, y - 1]
- x % y ==> [0, y - 1]
- -x % y ==> [-y + 1, 0]
- -x % -y ==> [-y + 1, 0]
于是,我給JDK發(fā)了個(gè)patch,這個(gè)問(wèn)題算是解決了。但是Nils提到,C2是否有相同的優(yōu)化呢?后面Tobias幫忙確認(rèn)了一下C2沒(méi)有,我再后來(lái)也進(jìn)一步確認(rèn)了,所以下一步是調(diào)研C2是否能應(yīng)用同樣的優(yōu)化。
調(diào)研為C2應(yīng)用同樣的優(yōu)化
本來(lái)以為是比較trivial的事情,為求余節(jié)點(diǎn)的類型系統(tǒng)加點(diǎn)代碼,推導(dǎo)一下上下界即可,實(shí)際上我也這么做的,但是最后發(fā)現(xiàn)這樣沒(méi)有消除上下界。默認(rèn)開啟-XX:+GenerateRangeChecks后,在數(shù)組訪問(wèn)過(guò)程中(Parse::array_addressing),C2仍然生成了范圍檢查。
調(diào)試后發(fā)現(xiàn)推導(dǎo)上下界根本沒(méi)有執(zhí)行,因?yàn)镃2創(chuàng)建完求余節(jié)點(diǎn)后,會(huì)執(zhí)行一個(gè)IGVN的過(guò)程,即迭代的應(yīng)用多種優(yōu)化,其中就包括理想化,C2理想化是指應(yīng)用很多局部小優(yōu)化的過(guò)程,在這個(gè)例子中就是特殊處理形如x%2^n,x%2^n-1和x%1的情況,如果除數(shù)是整數(shù)常量,它還會(huì)使用一個(gè)來(lái)自https://book.douban.com/subject/1784887/書里面的算法,即Division by Invariant Integers using Multiplication(by Granlund and Montgomery),搜了一下知乎有類似的文章,想要了解細(xì)節(jié)可以讀讀https://zhuanlan.zhihu.com/p/151038723。知道了原因,于是我改了下代碼,禁止了求余節(jié)點(diǎn)的理想化,心想這總可以了吧。
還是不行
是的,還是不行。盡管我已經(jīng)禁止了對(duì)求余符號(hào)的理想化優(yōu)化,但是范圍檢查還是生成了。。。我又繼續(xù)看代碼,發(fā)現(xiàn)除了理想化的這個(gè)優(yōu)化之外,C2在IR(中間表示)構(gòu)造的過(guò)程中又 又 又 又 又對(duì)求余運(yùn)算做了個(gè)優(yōu)化!如果除數(shù)是正整數(shù)常量,且是2^n,那么C2會(huì)對(duì)它進(jìn)行變形,IR如圖所示:

左邊的IR是 IR構(gòu)造的時(shí)候C2做的優(yōu)化后的效果,右邊是理想化優(yōu)化后的效果。實(shí)際上它們做的事情本身是比較重復(fù)的,而且經(jīng)過(guò)測(cè)試發(fā)現(xiàn),理想化優(yōu)化的算法要好于IR構(gòu)造過(guò)程中的優(yōu)化,所以我又提了個(gè)patch解決這個(gè)問(wèn)題(不過(guò)還在review中)。
結(jié)語(yǔ)
我認(rèn)為為求余節(jié)點(diǎn)推導(dǎo)上下界也是有意義的,如果以后有其他優(yōu)化會(huì)變形為求余運(yùn)算,那么它們可以應(yīng)用這個(gè)推導(dǎo),同時(shí),為求余做統(tǒng)一完善的類型推導(dǎo)這件事本身也是正確的,所以我又提了個(gè)patch??梢钥吹?,最終我只消除了C1 arr[x%4]的范圍檢查,還是沒(méi)能消除C2 arr[x%4]的范圍檢查,是不是以后可以說(shuō)C1有的地方做的比C2好了(狗頭hh。
以上就是Java求余%操作引發(fā)的一連串故事的詳細(xì)內(nèi)容,更多關(guān)于Java求余%的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
SpringBoot+layui實(shí)現(xiàn)文件上傳功能
Spring Boot是由Pivotal團(tuán)隊(duì)提供的全新框架,其設(shè)計(jì)目的是用來(lái)簡(jiǎn)化新Spring應(yīng)用的初始搭建以及開發(fā)過(guò)程。這篇文章主要介紹了SpringBoot+layui實(shí)現(xiàn)文件上傳,需要的朋友可以參考下2018-09-09
解析springboot整合谷歌開源緩存框架Guava Cache原理
本文主要為大家解析了springboot整合谷歌開源緩存框架Guava Cache的原理以及在實(shí)際開發(fā)過(guò)程中的使用,附含源碼,有需要的朋友可以參考下2021-08-08
Dependency ‘XXX:‘ not found問(wèn)題的三步解決
這篇文章主要介紹了Dependency ‘XXX:‘ not found問(wèn)題的三步解決方案,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2024-01-01
SpringMVC獲取請(qǐng)求參數(shù)筆記整理
本文記錄和分享在學(xué)習(xí)Spring MVC過(guò)程中的筆記,通過(guò)案例示例代碼分析給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2022-04-04
spring boot validation參數(shù)校驗(yàn)實(shí)例分析
這篇文章主要介紹了spring boot validation參數(shù)校驗(yàn),結(jié)合實(shí)例形式分析了spring boot validation進(jìn)行數(shù)據(jù)有效性驗(yàn)證的相關(guān)操作技巧,需要的朋友可以參考下2019-11-11
Invalid bound statement(not found):錯(cuò)誤的解決方案
本文介紹了在開發(fā)Java SpringBoot應(yīng)用程序時(shí)出現(xiàn)的"Invalidboundstatement(notfound)"錯(cuò)誤的原因及解決方法,該錯(cuò)誤通常與MyBatis或其他持久化框架相關(guān),可能是由于配置錯(cuò)誤、拼寫錯(cuò)誤或其他問(wèn)題引起的,解決方法包括檢查SQL映射文件2025-01-01
Springboot創(chuàng)建子父工程過(guò)程圖解
這篇文章主要介紹了Springboot創(chuàng)建子父工程過(guò)程圖解,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2020-02-02
Java中json使用方法_動(dòng)力節(jié)點(diǎn)Java學(xué)院整理
JSON(JavaScript Object Notation) 是一種輕量級(jí)的數(shù)據(jù)交換格式, json是個(gè)非常重要的數(shù)據(jù)結(jié)構(gòu),在web開發(fā)中應(yīng)用十分廣泛。下面通過(guò)本文給大家講解Java中json使用方法,感興趣的朋友一起看看吧2017-07-07
Java?Web開發(fā)中的分頁(yè)與參數(shù)校驗(yàn)舉例詳解
這篇文章主要介紹了JavaWeb開發(fā)中的分頁(yè)設(shè)計(jì)和參數(shù)校驗(yàn),分頁(yè)設(shè)計(jì)通過(guò)分頁(yè)查詢參數(shù)優(yōu)化查詢性能,文中通過(guò)代碼介紹的非常詳細(xì),需要的朋友可以參考下2025-02-02

