Java類型轉(zhuǎn)換valueOf與parseInt區(qū)別探討解析
前言
在編程中,遇到類型轉(zhuǎn)換,好像會(huì)經(jīng)常用到 parseInt
和 valueOf
,當(dāng)然這里只拿 Integer
類型進(jìn)行陳述,其他類型也是雷同的;
想必有讀者也跟我一樣,經(jīng)常交叉使用這兩個(gè)方法,但卻不知道這兩者到底有什么區(qū)別,接下來(lái)就來(lái)探究一番;
區(qū)別
Integer.parseInt(s)
的作用就是把字符串 s 解析成有符號(hào)基本類型的 int;Integer.valueOf(s)
把字符串 s 解析成 Integer 對(duì)象類型,返回的對(duì)象可以調(diào)用 Integer 中的方法;
接下來(lái),通過(guò)源碼進(jìn)行逐一解析;
parseInt
我們首先點(diǎn)進(jìn) parseInt()
方法中,
public static int parseInt(String s) throws NumberFormatException { return parseInt(s, 10); }
可以看到,這個(gè)被我們調(diào)用的 parseInt()
方法返回了一個(gè)重載方法:
public static int parseInt(String s, int radix) throws NumberFormatException { if (s == null) { throw new NumberFormatException("null"); } else if (radix < 2) { throw new NumberFormatException("radix " + radix + " less than Character.MIN_RADIX"); } else if (radix > 36) { throw new NumberFormatException("radix " + radix + " greater than Character.MAX_RADIX"); } else { boolean negative = false; int i = 0; int len = s.length(); int limit = -2147483647; if (len <= 0) { throw NumberFormatException.forInputString(s); } else { char firstChar = s.charAt(0); if (firstChar < '0') { if (firstChar == '-') { negative = true; limit = -2147483648; } else if (firstChar != '+') { throw NumberFormatException.forInputString(s); } if (len == 1) { throw NumberFormatException.forInputString(s); } ++i; } int multmin = limit / radix; int result; int digit; for(result = 0; i < len; result -= digit) { digit = Character.digit(s.charAt(i++), radix); if (digit < 0 || result < multmin) { throw NumberFormatException.forInputString(s); } result *= radix; if (result < limit + digit) { throw NumberFormatException.forInputString(s); } } return negative ? result : -result; } } }
1、首先看到的是,該方法傳入了兩個(gè)參數(shù),parseInt(String s, int radix)
,這個(gè)可以根據(jù)被調(diào)用時(shí)傳入的參數(shù),return parseInt(s, 10);
,盲猜一下,s
就是表示要轉(zhuǎn)換成數(shù)字型的字符串,而 radix
英文是基數(shù)的意思,這里應(yīng)該表示進(jìn)制,即這個(gè)傳入的字符串是多少進(jìn)制的,那到底是不是呢,我們接著往下看;
2、這里先是對(duì)字符串 s
是否為空,以及 radix
的大小進(jìn)行一個(gè)判斷,不符合條件則拋出 NumberFormatException
異常,也就是數(shù)字格式化異常;
if (s == null){ throw new NumberFormatException("null"); } else if (radix < 2) { throw new NumberFormatException("radix " + radix + " less than Character.MIN_RADIX"); } else if (radix > 36) { throw new NumberFormatException("radix " + radix + " greater than Character.MAX_RADIX"); } else {
3、接著往下,再一次對(duì)長(zhǎng)度進(jìn)行一個(gè)校驗(yàn),
int len = s.length(); if (len <= 0) { throw NumberFormatException.forInputString(s); } else { ... }
我在這里只想到了一個(gè)能讓它拋出異常的條件,
Integer.parseInt("");
運(yùn)行結(jié)果:
Exception in thread "main" java.lang.NumberFormatException: For input string: ""
at java.base/java.lang.NumberFormatException.forInputString(NumberFormatException.java:65)
at java.base/java.lang.Integer.parseInt(Integer.java:662)
at java.base/java.lang.Integer.parseInt(Integer.java:770)
4、接下來(lái)會(huì)檢測(cè)第一個(gè)字符是啥,如果是 -
,則將 negative
設(shè)置成 true
,表示這是個(gè)負(fù)數(shù),并且將邊界 limit
設(shè)置成最小邊界;
如果不是 +
,則表示該字符既不是數(shù)字,不也是性質(zhì)符號(hào),因此拋出 NumberFormatException
異常;
如果字符串 s
的長(zhǎng)度只有1,則表明這是非數(shù)字,不符合要求,也拋出 NumberFormatException
異常;
++i
是因?yàn)槿绻谝晃皇欠?hào)的話,那么在后續(xù)的循環(huán)中追加數(shù)字則直接跳過(guò)首位;
char firstChar = s.charAt(0); if (firstChar < '0') { if (firstChar == '-') { negative = true; limit = -2147483648; } else if (firstChar != '+') { throw NumberFormatException.forInputString(s); } if (len == 1) { throw NumberFormatException.forInputString(s); } ++i; }
5、根據(jù)進(jìn)制來(lái)調(diào)整邊界,以防越界;
int multmin = limit / radix;
6、Character.digit()
用于將字符轉(zhuǎn)為為對(duì)應(yīng)進(jìn)制的整數(shù),如果該字符不是進(jìn)制內(nèi)的就返回-1,例如輸入的字符是9,但是進(jìn)制是2,那么就不符合,則會(huì)返回-1;
然后就是進(jìn)行計(jì)算;
int result; int digit; for(result = 0; i < len; result -= digit) { digit = Character.digit(s.charAt(i++), radix); if (digit < 0 || result < multmin) { throw NumberFormatException.forInputString(s); } result *= radix; if (result < limit + digit) { throw NumberFormatException.forInputString(s); } }
7、最后判斷是否為負(fù)數(shù)完成轉(zhuǎn)換;
return negative ? result : -result;
valueOf
照例查看源碼:
public static Integer valueOf(String s, int radix) throws NumberFormatException { return parseInt(s, radix); } public static Integer valueOf(String s) throws NumberFormatException { return parseInt(s, 10); } @HotSpotIntrinsicCandidate public static Integer valueOf(int i) { return i >= -128 && i <= Integer.IntegerCache.high ? Integer.IntegerCache.cache[i + 128] : new Integer(i); }
可以看出 valueOf(String s, int radix)
和 valueOf(String s)
都是直接調(diào)用返回了 parseInt
方法,而 valueOf(int i)
則是一個(gè) int
轉(zhuǎn)成 Integer
的自動(dòng)裝箱;
接下來(lái)探究一下 IntegerCache
,可以看出這是 Integer
的成員內(nèi)部類,來(lái)看源碼:
private static class IntegerCache { static final int low = -128; static final int high; static final Integer[] cache; static Integer[] archivedCache; private IntegerCache() { } static { int h = 127; String integerCacheHighPropValue = VM.getSavedProperty("java.lang.Integer.IntegerCache.high"); int size; if (integerCacheHighPropValue != null) { try { size = Integer.parseInt(integerCacheHighPropValue); size = Math.max(size, 127); h = Math.min(size, 2147483518); } catch (NumberFormatException var6) { } } high = h; VM.initializeFromArchive(Integer.IntegerCache.class); size = high - -128 + 1; if (archivedCache == null || size > archivedCache.length) { Integer[] c = new Integer[size]; int j = -128; for(int k = 0; k < c.length; ++k) { c[k] = new Integer(j++); } archivedCache = c; } cache = archivedCache; assert high >= 127; } }
整體就是初始化一個(gè) IntegerCache.cache
數(shù)組,數(shù)組里面存儲(chǔ)-128到127之間的數(shù)字當(dāng)做是緩存,源碼一開(kāi)始是分析數(shù)組長(zhǎng)度,然后給數(shù)組賦值;
總的來(lái)說(shuō),三個(gè)重構(gòu)的 valueOf()
方法還是大同小異的:
Integer valueOf(int i)
:返回一個(gè)表示指定的 int 值的 Integer 實(shí)例;Integer valueOf(String s)
:返回保存指定的 String 的值的 Integer 對(duì)象;Integer valueOf(String s, int radix)
:返回一個(gè) Integer 對(duì)象,該對(duì)象中保存了用第二個(gè)參數(shù)提供的基數(shù)進(jìn)行解析時(shí)從指定的 String 中提取的值;
以上就是Java 中 valueOf 和 parseInt 的區(qū)別探討解析的詳細(xì)內(nèi)容,更多關(guān)于Java valueOf與parseInt區(qū)別的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
SpringBoot Controller返回圖片的三種方式
在互聯(lián)網(wǎng)的世界里,圖片無(wú)處不在,它們是信息傳遞的重要媒介,也是視覺(jué)盛宴的一部分,而在Spring Boot項(xiàng)目中,如何優(yōu)雅地處理和返回圖片數(shù)據(jù),則成為了開(kāi)發(fā)者們不得不面對(duì)的問(wèn)題,今天,就讓我們一起來(lái)探索Spring Boot Controller的神奇轉(zhuǎn)換,需要的朋友可以參考下2024-07-07Java接口自動(dòng)化測(cè)試框架設(shè)計(jì)之Get請(qǐng)求方法和測(cè)試詳解
這篇文章主要介紹了Java接口自動(dòng)化測(cè)試框架設(shè)計(jì) Get請(qǐng)求方法和測(cè)試,框架設(shè)計(jì)我們只是介紹基本的組件,而且框架設(shè)計(jì)沒(méi)有想象那么難,一步一步跟著做就會(huì)了。這篇我們來(lái)演示,如果通過(guò)Java代碼來(lái)實(shí)現(xiàn)一個(gè)用純代碼實(shí)現(xiàn)Http中的Get請(qǐng)求過(guò)程,需要的朋友可以參考下2019-07-07Resty極簡(jiǎn)restful框架快速接入Spring
這篇文章主要為大家介紹了Resty極簡(jiǎn)的restful框架快速接入Spring詳細(xì)說(shuō)明,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-03-03java實(shí)現(xiàn)二分法查找出數(shù)組重復(fù)數(shù)字
這篇文章主要為大家詳細(xì)介紹了java實(shí)現(xiàn)二分法查找出數(shù)組重復(fù)數(shù)字,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2018-11-11Java中try-catch的使用及注意細(xì)節(jié)
現(xiàn)在有很多的語(yǔ)言都支持try-catch,比如常見(jiàn)的就是c++,java等,這篇文章主要給大家介紹了關(guān)于Java中try-catch的使用及注意細(xì)節(jié)的相關(guān)資料,文中通過(guò)圖文以及實(shí)例代碼介紹的非常詳細(xì),需要的朋友可以參考下2023-06-06IntelliJ IDEA基于Maven構(gòu)建Java項(xiàng)目
在 Java 開(kāi)發(fā)中,使用 Maven 是一種廣泛采用的構(gòu)建工具,本文主要介紹了IntelliJ IDEA基于Maven構(gòu)建Java項(xiàng)目,具有一定的參考價(jià)值,感興趣的可以了解一下2024-03-03