当前位置:首页 > 文章列表 > 文章 > java教程 > Java新手必看的20个常见错误

Java新手必看的20个常见错误

2026-04-27 21:57:58 0浏览 收藏
这篇文章直击Java新手在日常编码中高频踩坑的20个典型错误,从字符串比较误用==、循环中修改集合引发并发修改异常,到包装类缓存陷阱、异常裸吞、日期API误用、HashMap扩容性能问题,再到线程安全疏忽、浮点数精度误判、资源未关闭、JSON序列化配置遗漏等,覆盖语法细节、JVM机制、并发编程、时间处理、性能优化和工程实践多个维度——每个错误都配以真实反例、原理剖析和简洁可靠的修复方案,帮你避开“看似能跑、实则埋雷”的陷阱,快速建立健壮、可维护、生产就绪的Java编码直觉。

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),必要时重新抛出或封装成业务异常,绝不裸吞。

日期处理用 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。

流式操作(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 构造顺序:父类构造器先执行,子类字段后赋值。避免在构造器中调用可被重写的方法,尤其涉及未初始化字段时。

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

今天带大家了解了的相关知识,希望对你有所帮助;关于文章的技术知识我们会一点点深入介绍,欢迎大家关注golang学习网公众号,一起学习编程~

美图秀秀长图制作技巧分享美图秀秀长图制作技巧分享
上一篇
美图秀秀长图制作技巧分享
Chromebook能开发HTML吗?硬件适配全解析
下一篇
Chromebook能开发HTML吗?硬件适配全解析
查看更多
最新文章
查看更多
课程推荐
  • 前端进阶之JavaScript设计模式
    前端进阶之JavaScript设计模式
    设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
    543次学习
  • GO语言核心编程课程
    GO语言核心编程课程
    本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
    516次学习
  • 简单聊聊mysql8与网络通信
    简单聊聊mysql8与网络通信
    如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
    500次学习
  • JavaScript正则表达式基础与实战
    JavaScript正则表达式基础与实战
    在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
    487次学习
  • 从零制作响应式网站—Grid布局
    从零制作响应式网站—Grid布局
    本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
    485次学习
查看更多
AI推荐
  • 剧云 - 免费 AI 智能中文剧本创作平台
    剧云
    剧云是专业中文剧本创作平台,安全稳定运行十余年,集成AI编剧、剧本医生审核、人物小传、剧情关系图、大纲编写、多人协作、Word导入导出、版权管控功能,数据安全防护,轻松高效创作剧本。
    96次使用
  • 万象有声 - AI 一站式有声内容创作平台
    万象有声
    万象有声,一个专为有声创作者打造的新一代智能有声内容创作平台。平台提供专业的智能拆章、智能画本编辑、AI配音、AI生成音效、后期制作、智能对轨、智能审听等有声创作全流程工具,可以帮助创作者高效、低成本创作出引人入胜的有声作品。立即体验,让有声书制作更简单!
    100次使用
  • Red Skill - 小红书推出的 AI Skill 分发平台
    Red Skill
    小红书创作服务平台为小红书创作者和机构提供视频上传、数据分析、粉丝管理、创作指导等多项运营服务,助力用户解锁更多创作者专属功能,体验高效创作!
    101次使用
  • MiMo Code - 小米大模型团队开源的新一代 AI 编程助手
    MiMo Code
    MiMo Code 是小米大模型团队开源的新一代 AI 编程助手,面向开发者提供代码理解、生成与辅助开发能力,适合作为 AI 编程工具收藏和体验。
    203次使用
  • TRAE Work - 字节跳动推出的 AI 原生工作台
    TRAE Work
    TRAE AI IDE | 国内首款 AI 原生集成开发环境,深度集成 Doubao-1.5-pro 与 DeepSeek 模型,支持中文自然语言一键生成完整代码框架,实时预览前端效果并智能修复 BUG。首创 Builder 模式实现需求到代码的自动化开发,兼容 Windows/macOS 系统,官网下载即用。
    234次使用
微信登录更方便
  • 密码登录
  • 注册账号
登录即同意 用户协议隐私政策
返回登录
  • 重置密码