背景
今天前端联调的一个小伙伴说一个接口报错了。
我初略看了下代码,发现没有什么问题。
先贴下报错的地方的代码
...
IntegralRecord integralRecord = new IntegralRecord();
integralRecord.setUid(uid);
integralRecord.setLinkType(IntegralRecordConstants.TYPE_SIGN);
integralRecord.setTitle(IntegralRecordConstants.INTEGRAL_TASK);
integralRecord.setIntegral(Integer.parseInt(amount));
integralRecord.setStatus(1);
integralRecord.setType(IntegralRecordConstants.TYPE_ADD);
integralRecord.setMark(StrUtil.format("完成任务赠送{}积分", amount));
userIntegralRecordService.save(integralRecord);
...
问题定位
没办法,只能debugger一行行去跑了。
在跑到
integralRecord.setIntegral(Integer.parseInt(amount));
这行的话,然后报了For input string的错误信息。
这行代码,我看了下,只是对类型做了一下转换操作。
难道是传过来的数据有问题?
再跑一次,发现amount=2.00
说明:因为amount涉及了多个地方共用,考虑到可能会存在小数的情况,在前端传递的时候就使用了String,后端根据实际情况使用对应的类型进行转换。
转换的时候,确实是这行出了问题。
具体的测试情况如下所示:
然后我们改成
amount="2"
结果如下
那么问题就可以定义了,是因为转换的时候,Integer无法识别带小数的情况。
问题跟进
Integer.parseInt源码初探
我们发现,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");
}
boolean negative = false;
int i = 0, len = s.length();
int limit = -Integer.MAX_VALUE;
if (len > 0) {
char firstChar = s.charAt(0);
if (firstChar < '0') { // Possible leading "+" or "-"
if (firstChar == '-') {
negative = true;
limit = Integer.MIN_VALUE;
} else if (firstChar != '+') {
throw NumberFormatException.forInputString(s);
}
if (len == 1) { // Cannot have lone "+" or "-"
throw NumberFormatException.forInputString(s);
}
i++;
}
int multmin = limit / radix;
int result = 0;
while (i < len) {
int 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);
}
result -= digit;
}
return negative ? result : -result;
} else {
throw NumberFormatException.forInputString(s);
}
}
源码定位
在代码中我们可以看到如果不再指定范围内就会报错
int digit = Character.digit(s.charAt(i++), radix);
if (digit < 0 || result < multmin) {
throw NumberFormatException.forInputString(s);
}
java.lang.Character.digit(char ch, int radix)方法 :
在指定的基数返回字符ch的数值,如果基数是不在范围内MIN_RADIX≤基数≤MAX_RADIX或如果该值的通道是不是一个有效的数字在指定的基数-1,则返回。
radix的值是10,这个是因为转换Int类型的是时候,单个字符0~9。
所以如果值不是0~9之间的数字就会报错。
抛出的异常就是forInputString。
解决方案
字符串截取
这个示例中,已经很明确的知道了只需要整数部分,那么可以根据指定字符"."进行截取。
然后转换之后就可以了。
借用第三方
这里推荐使用下hutool这个工具包里面的NumberUtil
可以直接将类似的都进行转换。
方法:NumberUtil.parseInt
String val ="2.00";
int i = NumberUtil.parseInt(val);
类型转换
因为可以判定,值一定是数值类型的。
所以可以直接转换为BigDecimal类型。
BigDecimal有个intValue方法,直接转换后调用即可
String val ="2.00";
int i = new BigDecimal(val).intValue();
总结
其实问题都比较简单,只要你多花点功夫去研究原理。你发现其实都差不多的。没有什么东西是很难的