前言
數(shù)字在某些領(lǐng)域經(jīng)常用字符串來進(jìn)行表示和傳遞。那么我們?nèi)绾闻袛嘁粋€(gè)字符串是否是數(shù)字呢?今天我們來探討一下這個(gè)話題。
空字符和null
首先我們可以很清晰的知道空字符""和null 肯定不是數(shù)字。其實(shí)我們寫其他邏輯一樣。把一些最極端最容易辨別處理的邏輯判斷直接要優(yōu)先進(jìn)行處理。這是一個(gè)小的竅門。
toCharArray
排除了先前情況的字符串可以通過toCharArray() 方法轉(zhuǎn)換為char數(shù)組。而Character.isDigit(int) 很容易判斷該char元素是否是數(shù)字(不要問char為什么是int?。?。 那么該方法會(huì)不會(huì)奏效呢?我們來操作一波各種情況:
public class Main {
public static void main(String[] args) {
// false
System.out.println("\"\" 是不是數(shù)字: "+isNumeric(""));
// false
System.out.println("\" \" 是不是數(shù)字: "+isNumeric(" "));
// false
System.out.println("null 是不是數(shù)字: "+isNumeric(null));
// false
System.out.println("1,200 是不是數(shù)字: "+isNumeric("1,200"));
// true
System.out.println("1 是不是數(shù)字: "+isNumeric("1"));
// 預(yù)期是負(fù)數(shù) 卻為 false
System.out.println("-1 是不是數(shù)字: "+isNumeric("-1"));
// true
System.out.println("200 是不是數(shù)字: "+isNumeric("200"));
// 預(yù)期是保留兩位的浮點(diǎn)數(shù) 卻為false
System.out.println("3000.00 是不是數(shù)字: "+isNumeric("3000.00"));
// 二進(jìn)制
System.out.println("0b11001 是不是數(shù)字: "+isNumeric("0b11001"));
// 八進(jìn)制 true
System.out.println("012 是不是數(shù)字: "+isNumeric("012"));
// 十六進(jìn)制 false
System.out.println("0x12 是不是數(shù)字: "+isNumeric("0x12"));
// A-F 代表十六進(jìn)制中的 10-15 false
System.out.println("0xAF 是不是數(shù)字: "+isNumeric("0xAF"));
// double false
System.out.println("12.12d 是不是數(shù)字: "+isNumeric("12.12d"));
// double 科學(xué)計(jì)數(shù)法 false
System.out.println("12E4 是不是數(shù)字: "+isNumeric("12E4"));
// float false
System.out.println("12.123f 是不是數(shù)字: "+isNumeric("12.123f"));
// 分隔符 jdk1.7 false
System.out.println("1_000_000 是不是數(shù)字: "+isNumeric("1_000_000"));
}
public static boolean isNumeric(final String str) {
// null or empty
if (str == null || str.length() == 0) {
return false;
}
return str.chars().allMatch(Character::isDigit);
}
}
從上面可以看出,中規(guī)中矩的十進(jìn)制正整數(shù)沒有太多的問題。一旦是浮點(diǎn)數(shù),小數(shù),負(fù)數(shù),二進(jìn)制,十六進(jìn)制,科學(xué)計(jì)數(shù)法,分隔符其中一種的情況下,這個(gè)方法就不太好用了。突然想到包裝類還有一些方法可用。
parse
數(shù)字的包裝類都有對(duì)應(yīng)的parse方法 。 如果字符串不符合對(duì)應(yīng)數(shù)字類型的規(guī)則將拋出NumberFormatException 異常。所以我們這里改動(dòng)一下我們的判斷方法:
public static boolean isNumeric(final String str) {
// null or empty
if (str == null || str.length() == 0) {
return false;
}
try {
Integer.parseInt(str);
return true;
} catch (NumberFormatException e) {
try {
Double.parseDouble(str);
return true;
} catch (NumberFormatException ex) {
try {
Float.parseFloat(str);
return true;
} catch (NumberFormatException exx) {
return false;
}
}
}
}
再執(zhí)行一下,得出如下結(jié)果:
"" 是不是數(shù)字: false
" " 是不是數(shù)字: false
null 是不是數(shù)字: false
1,200 是不是數(shù)字: false
1 是不是數(shù)字: true
-1 是不是數(shù)字: true
200 是不是數(shù)字: true
3000.00 是不是數(shù)字: true
0b11001 是不是數(shù)字: false
012 是不是數(shù)字: true
0x12 是不是數(shù)字: false
0xAF 是不是數(shù)字: false
12.12d 是不是數(shù)字: true
12E4 是不是數(shù)字: true
12.123f 是不是數(shù)字: true
1_000_000 是不是數(shù)字: false
從以上的第五行開始,數(shù)字的表示都是java支持的。從執(zhí)行結(jié)果來看除了二進(jìn)制,十六進(jìn)制,分隔符其它都是符合預(yù)期的。 雖然該方式不夠完美,但是我們也能從中學(xué)到對(duì)應(yīng)parse方法的一些規(guī)則。這才是重點(diǎn)。
第三方類庫(kù)
也就是說jdk提供的api并沒有銀彈。那么是否有第三方庫(kù)來檢測(cè)呢?我們使用了commons-lang3庫(kù)(3.9版本)中提供的NumberUtils工具類來進(jìn)行處理,我對(duì)其isParsable、isDigits、isCreatable 方法進(jìn)行分別測(cè)試,發(fā)現(xiàn)isCreatable 方法效果最好,只有分隔符沒有達(dá)到我們的預(yù)期。如果你不考慮這種情況應(yīng)該是isCreatable 基本能滿足需要的。其它類庫(kù)不知道你知道哪個(gè)好用可以通過我的公眾號(hào): Felordcn 告訴我。也歡迎訪問我個(gè)人博客站:https://felord.cn
正則表達(dá)式
其實(shí)使用正則表達(dá)式是最能完美解決問題的。
總結(jié)
今天通過對(duì)字符串是否是java的數(shù)字類型進(jìn)行的一些驗(yàn)證,讓我們對(duì)java中的數(shù)字如何表示和轉(zhuǎn)換又進(jìn)行了回顧。從中發(fā)現(xiàn)了一些不少的規(guī)則。相信本文會(huì)讓你對(duì)java中的數(shù)字有更加深刻的印象,也希望對(duì)你的工作有所幫助。
原創(chuàng)作者:碼農(nóng)小胖哥
轉(zhuǎn)載地址:https://gper.club/articles/7e7e7f7ff7g59gcag64