javaString转Integer分析
我们经常为⽤到Integer.valueOf(String str)这个⽅法,如果字符串格式不对,这个⽅法会抛出⼀个系统异常NumberFormatException 这⾥我们就要分析⼀下这个⽅法,其中Byte,Short也是调⽤了Ingeter中的⽅法.
在Integer类中的定义如下:
public static Integer valueOf(String s) throws NumberFormatException
{
return new Integer(parseInt(s, 10));
}
这⾥因为parseInt⽅法返回的int型的,这⾥调⽤了⼀个构造函数产⽣了⼀个新的Integer实例.
这⾥关⼼的是parseInt⽅法,该⽅法代码如下:
public static int parseInt(String s, int radix)
throws NumberFormatException
{
if (s == null) {
throw new NumberFormatException("null");
}
if (radix < Character.MIN_RADIX) {
throw new NumberFormatException("radix " + radix +
" less than Character.MIN_RADIX");
}
if (radix > Character.MAX_RADIX) {
throw new NumberFormatException("radix " + radix +
" greater than Character.MAX_RADIX");
}
int result = 0;
boolean negative = false;
int i = 0, max = s.length();
int limit;
int multmin;
int digit;
if (max > 0) {
if (s.charAt(0) == '-') {
negative = true;
limit = Integer.MIN_VALUE;
i++;
} else {
limit = -Integer.MAX_VALUE;
}
if (i < max) {
digit = Character.digit(s.charAt(i++),radix);
if (digit < 0) {
throw NumberFormatException.forInputString(s);
} else {
result = -digit;
}
}
while (i < max) {
// Accumulating negatively avoids surprises near MAX_VALUE
digit = Character.digit(s.charAt(i++),radix);
if (digit < 0) {
throw NumberFormatException.forInputString(s);
}
if (result < multmin) {
throw NumberFormatException.forInputString(s); 异常1
}
result *= radix;
if (result < limit + digit) {
throw NumberFormatException.forInputString(s); 异常2
}
result -= digit;
}
} else {
java valueofthrow NumberFormatException.forInputString(s);
}
if (negative) {
if (i > 1) {
return result;
} else { /* Only got "-" */
throw NumberFormatException.forInputString(s);
}
} else {
return -result;
}
}
很显然,该⽅法的第⼆个参数表⽰是基数(最常⽤的是⼗进制,还有⼗六机制,⼋进制等等).
如果字符串是空指针,直接抛出异常.
如果基础⼩于2或者⼤于36的话,抛出异常(这种情况⼀般不会出现,因为我们⽤的最多就是⼗进制的了).
如果是空字符串,也抛出异常,也就是max=0的情况了.
我们来关注下⾯的转换过程:
这⾥使⽤了Character中的静态⽅法digit,这个⽅法⽐较复杂,这⾥先说明它的功能:对于给定的基数,如果是合法的字符(可以转化为数字),返回该数字值,否则返回-1.⽐如digit('3',10)返回3,digit('a',10)返回-1.
这段程序看起来很简单,其实还真不容易看懂,这⾥先说明⼏个局部变量的含义吧:
result:记录返回值
negative:符号标志
i:字符串位置
s:字符串长度
limit:界限
multmin:也是⼀个界限
digit:当前字符表⽰的数字
先看第⼀个字符是否是'-'号,设定符号标志negative和极限值limit.
注意到limit⼀定是⼀个负值.
处理最⾼位,这⾥result保存的是负值,这样就可以对正负数统⼀处理.
关键就是这个while循环了,第⼀个if不⽤解释了,肯定是因为⾮法字符.
第⼆个if语句的含义:如果result⼩于multmin,会产⽣什么结果呢?
是不是⼀定会溢出呢?假设不会溢出,就是说结果必须>=limit.
result⼩于multmin,result⾄少应该位multmin-1,后⾯有result=result*radix=(multmin-1)*radix=multmin*radix-radix
该值肯定⼩于limit,其中multmin=limit/radix,注意这⾥都是负数.
所以假设不成⾥,如果result⼩于multmin的话,后⾯⼀定会溢出.
如果这⾥没有判断的话,溢出就⿇烦了,正数也会变负数了.
第三个if语句的含义:在这条语句以前肯定没有溢出,但是有可能加上最后⼀位digit就溢出了,所以这个判断也是必要的.
后⾯的就⽐较好理解了,else是表⽰空字符串"".
如果是负数的还要看是否长度是1,就只是⼀个'-'号的情况.
如果是正数的话返回相反数就可以了.
这⾥有好多地⽅都有可能抛出异常,只要看明⽩了程序就知道这个异常是那条语句抛出的了,这⾥考虑溢出异常:异常1和异常2.
Ingeter.Max_VALUE=2147483647
下⾯的两条语句在不同的地⽅抛出异常.
Ingeter.valueOf("2147483648");这个在异常2抛出的.
Ingeter.valueOf("21474836471");这个在异常1抛出的.
这⾥简单的分析了String转化为Ingeter的过程,其实整个Ingeter类也就主要是这个⽅法了,Byte和Short
都是调⽤这个⽅法的.
看看Byte的代码:
public static byte parseByte(String s, int radix)
throws NumberFormatException {
int i = Integer.parseInt(s, radix);
if (i < MIN_VALUE || i > MAX_VALUE)
throw new NumberFormatException(
"Value out of range. Value:\"" + s + "\" Radix:" + radix);
return (byte)i;
}
了解这个⽅法后就再也不会为Integer.valueOf()产⽣的异常感到意外了,特别是在JSP中,因为参数都是String型的,转换的时候动不动就出现异常,你该知道怎么回事了吧.

版权声明:本站内容均来自互联网,仅供演示用,请勿用于商业和其他非法用途。如果侵犯了您的权益请与我们联系QQ:729038198,我们将在24小时内删除。