Java源碼解析Integer方法解讀
toUnsignedString方法解讀
看到Integer中有這樣的一個(gè)方法把int轉(zhuǎn)為Unsigned類型的字符串,但是有幾個(gè)點(diǎn)不是很清楚,經(jīng)過查詢資料弄懂了,解讀如下:
/** * Convert the integer to an unsigned number. */ private static String toUnsignedString(int i, int shift) { char[] buf = new char[32]; int charPos = 32; int radix = 1 << shift; int mask = radix - 1; do { buf[--charPos] = digits[i & mask]; i >>>= shift; } while (i != 0); return new String(buf, charPos, (32 - charPos)); }
這里的參數(shù)shift是代表的進(jìn)制,如果是二進(jìn)制的話shift是2,八進(jìn)制那么就是8,相應(yīng)的其mask就計(jì)算成1和7了。通過mask與i相與不斷取出digits數(shù)組中對應(yīng)的字符。
在就是i每次進(jìn)行邏輯右移的運(yùn)算,最高位補(bǔ)充零,這樣最終經(jīng)過不斷的邏輯右移后i會變?yōu)?
此外,采用do-while是防止i本身是0的情況下,buf數(shù)組無法獲得其值。
toString方法解讀
// 這個(gè)數(shù)組表示的是數(shù)字的十位部分,下面會用到這個(gè)數(shù)組。 final static char [] DigitTens = { '0', '0', '0', '0', '0', '0', '0', '0', '0', '0', '1', '1', '1', '1', '1', '1', '1', '1', '1', '1', '2', '2', '2', '2', '2', '2', '2', '2', '2', '2', '3', '3', '3', '3', '3', '3', '3', '3', '3', '3', '4', '4', '4', '4', '4', '4', '4', '4', '4', '4', '5', '5', '5', '5', '5', '5', '5', '5', '5', '5', '6', '6', '6', '6', '6', '6', '6', '6', '6', '6', '7', '7', '7', '7', '7', '7', '7', '7', '7', '7', '8', '8', '8', '8', '8', '8', '8', '8', '8', '8', '9', '9', '9', '9', '9', '9', '9', '9', '9', '9', } ; // 這個(gè)數(shù)組表示的是數(shù)字的個(gè)位部分,下面會用到這個(gè)數(shù)組。把數(shù)組的每個(gè)部分進(jìn)行組合的話可以得到100以內(nèi)的所有的情況的二位整數(shù)。 final static char [] DigitOnes = { '0', '1', '2', '3', '4', '5', '6', '7', '8', '9', '0', '1', '2', '3', '4', '5', '6', '7', '8', '9', '0', '1', '2', '3', '4', '5', '6', '7', '8', '9', '0', '1', '2', '3', '4', '5', '6', '7', '8', '9', '0', '1', '2', '3', '4', '5', '6', '7', '8', '9', '0', '1', '2', '3', '4', '5', '6', '7', '8', '9', '0', '1', '2', '3', '4', '5', '6', '7', '8', '9', '0', '1', '2', '3', '4', '5', '6', '7', '8', '9', '0', '1', '2', '3', '4', '5', '6', '7', '8', '9', '0', '1', '2', '3', '4', '5', '6', '7', '8', '9', } ; public static String toString(int i) { if (i == Integer.MIN_VALUE) // 這里的加1,開始不太清楚什么意思,后來發(fā)現(xiàn)負(fù)數(shù)的話 // 需要在前面加負(fù)號的所以串的大小要加1才行 // 這里傳入stringSize的部分是正的,在下面的數(shù)組中 // 進(jìn)行映射 int size = (i < 0) ? stringSize(-i) + 1 : stringSize(i); char[] buf = new char[size]; getChars(i, size, buf); return new String(0, size, buf); } static void getChars(int i, int index, char[] buf) { int q, r; int charPos = index; char sign = 0; if (i < 0) { sign = '-'; i = -i; } // 超過65536的整數(shù),先進(jìn)行下面這樣的一個(gè)處理, // 這個(gè)處理中以100為單位,也就是,余數(shù)控制在兩位 // 這樣正好映射到上面的十位和個(gè)位數(shù)組,一次性寫入 // buf數(shù)組中兩位,這樣毫無疑問比求出每一位是要快很多的 while (i >= 65536) { q = i / 100; // really: r = i - (q * 100); r = i - ((q << 6) + (q << 5) + (q << 2)); i = q; buf [--charPos] = DigitOnes[r]; buf [--charPos] = DigitTens[r]; } // Fall thru to fast mode for smaller numbers // assert(i <= 65536, i); // 對于小于等于65536的整數(shù)而言,可以直接進(jìn)行下面的部分 // 而且這個(gè)地方是以除以10進(jìn)行的,但是實(shí)現(xiàn)并不是直接除 // 而是先求一個(gè)52429/2^19約等于0.1000... // 相當(dāng)于i除以了10,但是我不清楚的是為什么這里不直接 // 除以10,或許是因?yàn)榫炔粔虬?,除法產(chǎn)生浮點(diǎn)數(shù), // 或許會不精確,然后得到的除數(shù)再乘以10,得到10位以上 // 部分的數(shù),通過i-該部分十位以上的數(shù),得到個(gè)位的數(shù)字 for (;;) { q = (i * 52429) >>> (16+3); r = i - ((q << 3) + (q << 1)); // r = i-(q*10) ... buf [--charPos] = digits [r]; i = q; if (i == 0) break; } if (sign != 0) { buf [--charPos] = sign; } } final static int [] sizeTable = { 9, 99, 999, 9999, 99999, 999999, 9999999, 99999999, 999999999, Integer.MAX_VALUE }; // 這里應(yīng)該是進(jìn)行了優(yōu)化,通過sizeTable存儲了整型數(shù)據(jù)的位 // 的情況,從一位一直到10位:2147483647的情況, // 這個(gè)處理方式很巧妙 static int stringSize(int x) { for (int i=0; ; i++) if (x <= sizeTable[i]) return i+1; }
highestOneBit方法解讀
public static int highestOneBit(int i) { // HD, Figure 3-1 i |= (i >> 1); i |= (i >> 2); i |= (i >> 4); i |= (i >> 8); i |= (i >> 16); return i - (i >>> 1); }
這個(gè)方法很有意思,我自己算了算,然后才明白了他的精髓,這個(gè)方法的作用是求構(gòu)成一個(gè)整數(shù)的最大的位所代表的整數(shù)的值。這里通過位移的方式實(shí)現(xiàn)了這個(gè)功能。接下來舉個(gè)簡單的例子,128來講二進(jìn)制是1000 0000。下面以他為例子算下:
移1位
1000 0000
0100 0000
|-------------
移2位
1100 0000
0011 0000
|------------
移4位
1111 0000
0000 1111
|------------
移8位
1111 1111
0000 0000
|------------
移動16位
1111 1111
0000 0000
|------------
1111 1111
最終的結(jié)果如你所看到的,后面的位全部填充為1,把后面的位全部減掉就得到了最高的位代表的整數(shù)。
bitCount方法解析
public static int bitCount(int i) { // HD, Figure 5-2 i = i - ((i >>> 1) & 0x55555555); i = (i & 0x33333333) + ((i >>> 2) & 0x33333333); i = (i + (i >>> 4)) & 0x0f0f0f0f; i = i + (i >>> 8); i = i + (i >>> 16); return i & 0x3f; }
這個(gè)方法著實(shí)廢了半天功夫研究,后來算是搞懂了個(gè)大概:
第一行,實(shí)現(xiàn)的是把整型的二進(jìn)制位進(jìn)行兩個(gè)兩個(gè)的分組,然后統(tǒng)計(jì)這兩個(gè)位中的1的個(gè)數(shù),我不知道這個(gè)公式是怎么來的,但是算出來確實(shí)是這樣的。
第二行,實(shí)現(xiàn)的是把整型的二進(jìn)制位進(jìn)行四個(gè)四個(gè)的分組,然后計(jì)算段內(nèi)移位相加,就是1001-> 10 + 01 = 11 相當(dāng)于三個(gè)1了
第三行,就是把整型的二進(jìn)制位八個(gè)一組,然后類似上面的方式,進(jìn)行位移相加,當(dāng)然這里通過一些特定的移位以及與運(yùn)算實(shí)現(xiàn)的。
接下來就是十六個(gè)一組,三十二個(gè)一組最終將統(tǒng)計(jì)數(shù)字歸并到最后的幾位表示的統(tǒng)計(jì)數(shù)值中。
以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
Java下利用Jackson進(jìn)行JSON解析和序列化示例
本篇文章主要介紹了Java下利用Jackson進(jìn)行JSON解析和序列化示例,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下。2017-02-02Spring?Security過濾器鏈加載執(zhí)行流程源碼解析
Spring?Boot?對于?Spring?Security?提供了自動化配置方案,可以使用更少的配置來使用?Spring?Security。那么這個(gè)過濾器鏈?zhǔn)窃趺醇虞d和實(shí)現(xiàn)攔截的呢,對Spring?Security過濾器鏈加載執(zhí)行流程感興趣的朋友一起看看吧2021-12-12解決@Transaction注解導(dǎo)致動態(tài)切換更改數(shù)據(jù)庫失效問題
這篇文章主要介紹了解決@Transaction注解導(dǎo)致動態(tài)切換更改數(shù)據(jù)庫失效問題,具有很好的參考價(jià)值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2021-09-09詳解基于Spring Data的領(lǐng)域事件發(fā)布
這篇文章主要介紹了詳解基于Spring Data的領(lǐng)域事件發(fā)布,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2020-04-04