Java程序需先经javac编译为平台无关的.class字节码,再由JVM严格经历加载、链接(验证/准备/解析)、初始化三阶段后,才反射调用符合签名的public static void main(String[] args)启动执行。

Java程序不是直接由CPU执行的,它必须经过编译成字节码、再由JVM加载并解释/编译执行两个关键阶段。跳过其中任一环节,比如试图用java命令直接运行.java源文件(不先编译),就会报Error: Could not find or load main class这类错误。
javac 编译出的是 .class 文件,不是机器码
javac把.java源文件翻译成JVM能识别的字节码,保存为.class文件。这个过程不涉及操作系统或CPU架构,所以字节码是平台无关的。但注意:javac默认生成的字节码版本和当前JDK绑定——比如用JDK 17编译,默认生成class文件版本是61(对应Java 17),如果在JRE 11上运行就会抛java.lang.UnsupportedClassVersionError。
- 检查class版本:用
javap -verbose MyClass | grep major - 降级编译:加
-source 11 -target 11参数(需JDK支持) - 别误以为
.class能直接被Windows或Linux执行——它必须靠JVM加载
JVM 启动时真正做了三件事:加载、链接、初始化
当你敲下java MyClass,JVM不是简单“打开文件就跑”,而是严格走完类加载子系统流程:
-
加载:用
ClassLoader(如AppClassLoader)读取MyClass.class二进制流,生成java.lang.Class对象 -
链接:分验证(确保字节码合规)、准备(为
static字段分配内存并设默认值)、解析(把符号引用转为直接引用) -
初始化:真正执行
static代码块和static变量赋值语句——这是main方法能被执行的前提
常见卡点:如果static块里有死循环或未捕获异常,JVM会停在初始化阶段,报ExceptionInInitializerError,而不是进入main。
立即学习“Java免费学习笔记(深入)”;
main 方法不是起点,却是唯一被JVM主动调用的入口
JVM启动后,会反射调用你指定类的public static void main(String[] args)方法。这个签名必须完全匹配——少个static、参数类型写成List、返回值改成int,都会导致NoSuchMethodError或启动失败。
- 参数名
args可改,但类型必须是String[]或String... - 类名大小写敏感:文件叫
hello.java,但类声明是public class Hello,编译后得用java Hello,不是java hello - JVM不关心你有没有
System.out.println,哪怕main里只写return;,也算成功执行
真正容易被忽略的是:JVM进程一旦退出(比如main结束、或遇到未捕获Throwable),所有用户线程都会被强制终止——哪怕你起了一个new Thread(() -> {...}).start(),只要没显式标记为守护线程(setDaemon(true)),JVM也不会等它跑完。









