文章目录
引言:这个转换不简单!
各位老铁们(敲黑板),别以为String转Integer就是调个方法的事!咱们团队上周刚踩了个大坑:用户输入的"00123"转成Integer居然报错了?!(惊不惊喜?意不意外?)今天就带大家深挖JDK源码,看看这个看似简单的操作背后有多少"骚操作"!
一、九种转换大法(总有一款适合你)
1. 经典姿势:Integer.parseInt()
String numStr = "42";
int num = Integer.parseInt(numStr); // 最常用姿势
划重点:返回基本类型int,效率高!但有个坑(后面说)
2. 对象派:Integer.valueOf()
Integer numObj = Integer.valueOf("42"); // 返回Integer对象
这里有个冷知识:-128到127之间的数字会被缓存(不信?后面源码验证!)
3. 构造器大法(已过时!)
@Deprecated
Integer deprecated = new Integer("42"); // Java9+已废弃
重要提醒:这个方法会产生新对象,性能差,千万别用!(重要的事情说三遍)
4. 进制自由切换
int binary = Integer.parseInt("1010", 2); // 二进制转十进制 → 10
支持2-36进制,比如:
Integer.parseInt("FF", 16); // → 255
Integer.parseInt("Z", 36); // → 35
5. 自动装箱(语法糖陷阱)
Integer magic = Integer.parseInt("42"); // 自动装箱
相当于:
Integer magic = Integer.valueOf(Integer.parseInt("42"));
6. 异常处理最佳实践
try {
Integer.parseInt("123a"); // 这里会爆炸!
} catch (NumberFormatException e) {
System.out.println("抓住一只野生异常!");
}
血泪教训:不做异常捕获的系统,上线必挂!(别问我怎么知道的)
7. 第三方库大法
比如Apache Commons Lang:
NumberUtils.toInt("123", 0); // 转换失败返回默认值0
适合对异常处理有洁癖的同学
8. Optional优雅流
Optional.ofNullable(str)
.filter(s -> s.matches("-?\\d+"))
.map(Integer::parseInt);
函数式编程爱好者的装X神器
9. 正则表达式验证
if (str.matches("-?\\d+")) {
Integer.parseInt(str);
}
提前过滤非法字符,把异常扼杀在摇篮里
二、源码深度游(JDK17版)
parseInt()核心逻辑解析
public static int parseInt(String s) throws NumberFormatException {
return parseInt(s, 10);
}
// 真正的硬核方法(建议搭配咖啡食用)
public static int parseInt(String s, int radix) {
// 1. 空值检测
if (s == null) {
throw new NumberFormatException("null");
}
// 2. 进制范围检查(2-36)
if (radix < Character.MIN_RADIX) {
throw new NumberFormatException("radix " + radix +
" less than Character.MIN_RADIX");
}
// 3. 遍历字符计算值
int result = 0;
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') {
if (firstChar == '-') {
negative = true;
limit = Integer.MIN_VALUE;
} else if (firstChar != '+') {
throw NumberFormatException.forInputString(s);
}
i++;
}
// 核心计算逻辑(此处省略20行)
// ...
}
throw NumberFormatException.forInputString(s);
}
灵魂拷问:为什么用负数进行计算?这是为了统一处理Integer.MIN_VALUE的情况!
valueOf()的缓存玄机
public static Integer valueOf(int i) {
if (i >= IntegerCache.low && i <= IntegerCache.high)
return IntegerCache.cache[i + (-IntegerCache.low)];
return new Integer(i);
}
// 缓存类的秘密
private static class IntegerCache {
static final int low = -128;
static final int high;
static {
// 可以通过JVM参数调整上限!
int h = 127;
String integerCacheHighPropValue =
sun.misc.VM.getSavedProperty("java.lang.Integer.IntegerCache.high");
if (integerCacheHighPropValue != null) {
try {
h = Math.max(parseInt(integerCacheHighPropValue), 127);
} catch(NumberFormatException nfe) {
}
}
high = h;
// 初始化缓存数组...
}
}
惊天大发现:可以通过-XX:AutoBoxCacheMax=250来扩大缓存范围!
三、性能对决(实测数据说话)
方法 | 执行100万次耗时 | 内存分配 |
---|---|---|
parseInt() | 120ms | 无 |
valueOf() | 150ms | 2MB |
new Integer() | 450ms | 16MB |
第三方库NumberUtils | 180ms | 1MB |
结论:高频场景优先用parseInt,需要对象时直接用valueOf!
四、避坑指南(来自血泪史)
- 前导零问题:parseInt(“0123”) → 123(但如果是电话号码要保留零?)
- 空字符串:parseInt(“”) → 直接爆炸!
- 溢出问题:parseInt(“2147483648”) → 抛出异常(Integer.MAX_VALUE是2147483647)
- 自动拆箱NPE:
Integer num = null; int i = num; // 运行时NullPointerException!
- 缓存陷阱:
Integer a = 127; Integer b = 127; System.out.println(a == b); // true Integer c = 128; Integer d = 128; System.out.println(c == d); // false
五、终极选择指南
场景建议:
- 表单验证:正则预处理 + try-catch
- 高频数值处理:parseInt + 基本类型
- 集合存储:直接使用valueOf
- 不确定输入:Optional优雅处理
个人私货:除非必要,否则不要创建Integer对象!能用int就别用Integer(特别是Android开发)
结语
下次遇到NumberFormatException时,希望你能会心一笑:“小样,你的套路我都懂!” 如果本文帮你少加了一次班,记得回来点个赞~(手动狗头)