编译错误发生在javac阶段,运行错误发生在java启动后;前者是源码未通过语法/语义检查导致.class未生成,后者是字节码执行中JVM抛出异常或崩溃。

编译错误发生在 javac 阶段,运行错误发生在 java 启动后
Java 的错误分两类,本质区别在于触发时机:编译错误是源码没通过语法/语义检查,javac 直接拒绝生成 .class 文件;运行错误是字节码已加载并执行到某行时出问题,JVM 抛出异常或直接崩溃。
典型表现:javac HelloWorld.java 输出一堆 error: 行,且无 HelloWorld.class 生成 → 编译错误;java HelloWorld 正常启动但中途打印 Exception in thread "main" ... → 运行错误。
javac 报错常见类型和快速定位方法
编译错误全是静态检查结果,不依赖输入数据或运行环境。最常遇到的有:
-
cannot find symbol:变量、方法、类名拼错,或没 import,或作用域不对(比如在if块里声明的变量用在块外) -
missing return statement:非void方法末尾没写return,或所有分支路径都没覆盖(如if-else缺else) -
class, interface, or enum expected:大括号不匹配、类定义外写了可执行语句、文件名和public class名不一致 -
incompatible types:赋值时类型不兼容(如把String赋给int),或泛型擦除后实际类型冲突
注意:javac 不报空指针、数组越界这类问题——它们合法通过编译,只在运行时暴露。
立即学习“Java免费学习笔记(深入)”;
运行错误分 Error 和 Exception,处理策略完全不同
运行中抛出的 Throwable 子类,分两大类:
-
Error(如OutOfMemoryError、StackOverflowError):JVM 底层问题,程序无法合理恢复,一般不捕获,应排查配置或逻辑缺陷 -
Exception分为「受检」(IOException、SQLException)和「非受检」(NullPointerException、ArrayIndexOutOfBoundsException):前者强制要求try-catch或throws,后者编译期不管,全靠测试和代码审查提前发现
一个关键事实:NullPointerException 永远不会在编译时报错,哪怕你写了 String s = null; s.length(); ——这行能顺利编译,只等运行时炸。
如何用 IDE 和命令行快速区分错误类型
IDE(如 IntelliJ、Eclipse)会在编辑时标红编译错误,但对运行错误只在调试或运行后才显示;命令行下则更清晰:
javac Test.java # 如果失败,错误信息一定含 "error:",且退出码非 0
java Test # 如果失败,错误信息一定以 "Exception in thread" 或 "Error" 开头,且堆栈指向具体行号
容易忽略的一点:某些“看似运行错误”的现象其实是编译器优化导致的,比如 final static String s = "a"; System.out.println(s.toLowerCase()); 中的 toLowerCase() 可能在编译期被内联,而 s 若换成非 final 变量,同样调用就变成纯运行时行为。这种边界情况会让新手误判错误阶段。










