Java集合不支持基本类型因泛型擦除后操作Object,而基本类型非Object子类;自动装箱/拆箱通过valueOf()和xxxValue()实现,但存在空指针与性能陷阱。
Java集合不能直接存储基本数据类型,必须使用对应的包装类型(如 int → Integer)。
因为Java集合框架(如 ArrayList、
HashMap)的底层设计基于泛型,而泛型在编译后会进行类型擦除,最终操作的是 Object 类型。基本类型不是 Object 的子类,无法参与引用类型体系。
ArrayList 编译报错:「Type argument cannot be of primitive type」int、boolean 等不满足该约束Java 5 引入自动装箱(autoboxing)和拆箱(unboxing),让基本类型和包装类型之间“看起来”能无缝转换,但背后有隐式对象创建和性能开销。
Integer.valueOf(int)、Boolean.valueOf(boolean) 等静态工厂方法intValue()、booleanValue() 等实例方法Integer.valueOf(127) 返回常量池对象,Integer.valueOf(128) 每次新建对象ArrayListlist = new ArrayList<>(); list.add(42); // 自动装箱:等价于 list.add(Integer.valueOf(42)) int x = list.get(0); // 自动拆箱:等价于 list.get(0).intValue()
自动转换掩盖了对象创建、空指针、缓存边界等真实问题,实际编码中高频出错。
NullPointerException:当集合中存了 null,又执行拆箱操作,例如 Integer i = null; int j = i;
for (int i = 0; i )会生成大量临时 Integer 对象
Integer 对象用 == 比较结果为 false(未命中缓存,地址不同)list.remove(1) 本意是删索引 1 的元素,但因传入 int,实际调用的是 remove(int index);要删值为 1 的元素,得写 list.remove(Integer.valueOf(1))
对性能或内存敏感的场景(如高频数值计算、大数据量缓存),应避免依赖集合 + 包装类型。
fastutil 的 IntArrayList、Int2IntOpenHashMap
Vector API 做向量化计算,但不解决集合存储问题int[] data + int size),适用于固定模式的高性能需求包装类型不是语法糖,而是运行时实实在在的对象。理解 Integer 不等于 int,才能避开空指针、缓存混淆和装箱爆炸这些静默故障。