0

0

Java 新手经常踩的 20 个坑(真实案例)

冷漠man

冷漠man

发布时间:2025-12-21 11:56:40

|

892人浏览过

|

来源于php中文网

原创

字符串比较应使用equals而非==,因==比较引用地址;循环中修改ArrayList需用迭代器remove或延迟删除;Integer等包装类比较须用equals且注意缓存范围与空指针;异常不可裸吞,须记录日志;日期处理优先用Java 8+时间API;HashMap应预设初始容量;SimpleDateFormat不可共享;异步任务须处理异常并等待完成;浮点数比较应使用误差范围判断。

java 新手经常踩的 20 个坑(真实案例)

字符串比较不用 equals 却用 ==

新手常写 "abc" == str 判断内容是否相等,结果在某些情况下对、某些情况错。因为 == 比的是引用地址,而字符串字面量可能被 JVM 缓存(在字符串常量池里),但 new String("abc") 就一定不等于 "abc"。正确写法永远是 str != null && str.equals("abc"),更安全可用 "abc".equals(str) 避免空指针。

循环中修改 ArrayList 导致 ConcurrentModificationException

一边遍历 ArrayList,一边调用 list.remove(),哪怕只是删一个元素,也会直接抛异常。这不是线程问题,而是 fail-fast 机制在起作用。解决办法:用迭代器的 iterator.remove();或先收集要删的元素,遍历完再批量删;或改用 CopyOnWriteArrayList(仅适合读多写少场景)。

忽略自动装箱拆箱的坑

Integer a = 128, b = 128; System.out.println(a == b); 输出 false —— 因为 Integer 缓存范围是 -128 到 127,超出就新建对象。同样,Integer c = null; int d = c; 运行时直接 NPE。比较包装类用 equals,赋值前务必判空,基本类型运算前确保非 null。

try-catch 吞掉异常却不处理也不记录

常见写法:catch (Exception e) { } 或只写 e.printStackTrace();。前者让错误静默失败,后者日志没进系统日志体系,线上根本查不到。正确做法:记录带上下文的 warn/error 日志(用 SLF4J),必要时重新抛出或封装成业务异常,绝不裸吞。

立即学习Java免费学习笔记(深入)”;

日期处理用 Date 和 Calendar 硬刚

new Date(2023, 1, 1) 其实是 2024 年 2 月 1 日(月份从 0 开始);Calendar 的 set 和 add 行为容易混淆;时区、夏令时、跨年计算全是雷。Java 8 起请直接用 LocalDateTime / ZonedDateTime / DateTimeFormatter,不可变、线程安全、API 清晰。

HashMap 初始化不设初始容量

默认容量 16,加载因子 0.75,意味着放 13 个元素就扩容。频繁扩容(rehash)很耗时。如果预估要存 1000 个键值对,建议初始化写成 new HashMap(1280)(1000 ÷ 0.75 ≈ 1334,向上取 2 的幂得 2048,但 1280 更省空间且够用)。

把 SimpleDateFormat 当共享变量用

声明为 static 成员、多个线程共用同一个实例,必现解析错乱或报错。SimpleDateFormat 不是线程安全的。方案有三:方法内局部创建(轻量,推荐);用 ThreadLocal 包裹;换成线程安全的 DateTimeFormatter(Java 8+)。

异步任务忘了处理异常或等待完成

executor.submit(() -> { riskyCode(); }); 后不管不问,异常被吃掉,主线程也完全不知道任务失败。submit 返回 Future,必须调用 future.get()(会阻塞并抛出执行异常);或者用 CompletableFuture 链式处理异常(.exceptionally / .handle)。

用 == 比较浮点数

double a = 0.1 + 0.2; if (a == 0.3) {...} 几乎永远为 false。浮点数有精度误差。正确方式是用差值绝对值判断:Math.abs(a - 0.3) ;金额等关键场景一律用 BigDecimal

静态方法/变量滥用导致内存泄漏

把 Activity、Context、View 等持有强引用塞进 static Map 或 static List,Android 里极易引发 OOM;Java SE 中也可能让本该回收的对象无法 GC。原则:静态容器只存生命周期长、无依赖的对象;必须存短生命周期对象时,用 WeakReference 包裹。

JSON 序列化忽略 null 字段却没配好框架

以为 Jackson 默认不输出 null,结果发现还是打了 "field": null。其实是默认行为是输出 null;需显式配置 objectMapper.setSerializationInclusion(JsonInclude.Include.NON_NULL)。Gson 同理要 setExclusionStrategies。

OpenArt
OpenArt

在线AI绘画艺术图片生成器工具

下载

流式操作(Stream)中途修改原集合

list.stream().filter(...).forEach(list::remove); 会抛 ConcurrentModificationException。Stream 不是“另一个 for 循环”,它背后仍基于原集合迭代。要删元素,请用 removeIf 或收集新集合再替换。

接口实现类没加 @Override 却写错了方法签名

比如接口定义 void save(User user),实现类写了 public void save(User u),看着像重写,其实成了重载——接口方法根本没被实现,运行时报 AbstractMethodError。加上 @Override 注解,编译器立刻报错提醒。

文件流没关闭,也没用 try-with-resources

FileInputStream fis = new FileInputStream("x.txt"); ... fis.close();,但中间抛异常,close 就跳过了。资源泄露长期运行必崩。Java 7+ 必须用 try (FileInputStream fis = ...) { ... },异常时自动 close,连 finally 都不用写。

用 Arrays.asList() 返回的 List 做增删操作

List list = Arrays.asList("a", "b"); list.add("c"); 直接抛 UnsupportedOperationException。因为 Arrays.asList 返回的是 Arrays 内部的固定大小 List 实现,不支持结构性修改。需要可变列表,就包一层:new ArrayList(Arrays.asList(...))

并发修改 HashSet / HashMap 不加同步

多线程往同一个 HashMap put,即使没显式同步,也可能导致死循环(JDK 7 链表成环)、数据丢失或扩容异常。别迷信“我只读不写”——读操作也可能触发 resize。要用 ConcurrentHashMap,或 Collections.synchronizedMap()(注意迭代仍需手动同步)。

异常捕获太宽泛,掩盖真正问题

catch (Exception e) 处理所有异常,结果 IOException、NullPointerException、IllegalArgumentException 全被同一段逻辑吞掉,排查时完全分不清是网络超时还是参数错了。应按业务分层捕获:底层抛具体异常,上层按需 catch 并分类处理。

单元测试只测 happy path,不覆盖边界和异常分支

比如测试一个除法方法,只传 10/2,不试 10/0、null 参数、负数输入、超大数溢出。结果上线后遇到 0 除直接崩。JUnit 5 推荐用 @ParameterizedTest 覆盖多组输入,用 @Test(expected = ArithmeticException.class) 或 assertThrows 验证异常路径。

随意使用 System.out.println 调试,上线忘记删

控制台打印混进生产代码,不仅污染日志,还可能泄露敏感信息(用户 ID、token、SQL 参数)。所有调试输出必须用日志框架(SLF4J + Logback),且日志级别设为 DEBUG,并确保生产环境日志级别为 INFO 或更高。

final 字段初始化顺序搞错导致值为 null

父类构造器里调用了被子类重写的 final 方法,而该方法访问了子类的 final 字段——此时子类字段还没初始化,值为 null。根源是 Java 构造顺序:父类构造器先执行,子类字段后赋值。避免在构造器中调用可被重写的方法,尤其涉及未初始化字段时。

基本上就这些。不复杂,但容易忽略。踩过一遍,下次看到类似代码本能警觉,就是新手走向靠谱开发者的开始。

相关专题

更多
java
java

Java是一个通用术语,用于表示Java软件及其组件,包括“Java运行时环境 (JRE)”、“Java虚拟机 (JVM)”以及“插件”。php中文网还为大家带了Java相关下载资源、相关课程以及相关文章等内容,供大家免费下载使用。

835

2023.06.15

java正则表达式语法
java正则表达式语法

java正则表达式语法是一种模式匹配工具,它非常有用,可以在处理文本和字符串时快速地查找、替换、验证和提取特定的模式和数据。本专题提供java正则表达式语法的相关文章、下载和专题,供大家免费下载体验。

740

2023.07.05

java自学难吗
java自学难吗

Java自学并不难。Java语言相对于其他一些编程语言而言,有着较为简洁和易读的语法,本专题为大家提供java自学难吗相关的文章,大家可以免费体验。

736

2023.07.31

java配置jdk环境变量
java配置jdk环境变量

Java是一种广泛使用的高级编程语言,用于开发各种类型的应用程序。为了能够在计算机上正确运行和编译Java代码,需要正确配置Java Development Kit(JDK)环境变量。php中文网给大家带来了相关的教程以及文章,欢迎大家前来阅读学习。

397

2023.08.01

java保留两位小数
java保留两位小数

Java是一种广泛应用于编程领域的高级编程语言。在Java中,保留两位小数是指在进行数值计算或输出时,限制小数部分只有两位有效数字,并将多余的位数进行四舍五入或截取。php中文网给大家带来了相关的教程以及文章,欢迎大家前来阅读学习。

399

2023.08.02

java基本数据类型
java基本数据类型

java基本数据类型有:1、byte;2、short;3、int;4、long;5、float;6、double;7、char;8、boolean。本专题为大家提供java基本数据类型的相关的文章、下载、课程内容,供大家免费下载体验。

446

2023.08.02

java有什么用
java有什么用

java可以开发应用程序、移动应用、Web应用、企业级应用、嵌入式系统等方面。本专题为大家提供java有什么用的相关的文章、下载、课程内容,供大家免费下载体验。

430

2023.08.02

java在线网站
java在线网站

Java在线网站是指提供Java编程学习、实践和交流平台的网络服务。近年来,随着Java语言在软件开发领域的广泛应用,越来越多的人对Java编程感兴趣,并希望能够通过在线网站来学习和提高自己的Java编程技能。php中文网给大家带来了相关的视频、教程以及文章,欢迎大家前来学习阅读和下载。

16926

2023.08.03

高德地图升级方法汇总
高德地图升级方法汇总

本专题整合了高德地图升级相关教程,阅读专题下面的文章了解更多详细内容。

43

2026.01.16

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Kotlin 教程
Kotlin 教程

共23课时 | 2.6万人学习

C# 教程
C# 教程

共94课时 | 6.9万人学习

Java 教程
Java 教程

共578课时 | 47.2万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号