背景
今天前端联调的一个小伙伴说一个接口报错了。
我初略看了下代码,发现没有什么问题。
先贴下报错的地方的代码
...
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();
总结
其实问题都比较简单,只要你多花点功夫去研究原理。你发现其实都差不多的。没有什么东西是很难的

