




main方法必须是public static void,因为JVM通过反射调用且不创建对象:public保证外部可访问,static支持无需实例执行,void因JVM不处理返回值;其他签名或修饰符会导致运行错误。
Java虚拟机在启动时,会通过反射机制查找并调用类中的main方法,这个过程不创建对象、不经过任何实例化。因此它必须满足三个硬性条件:
public:JVM从外部调用,访问权限不能受限static:无需实例即可执行,否则JVM得先知道构造参数、父类初始化顺序等,无法统一约定void:JVM不处理返回值,定义为int或其它类型会导致NoSuchMethodError
常见错误:static public void main(String[] args)语法合法但可读性差;public static void main(string[] args)因string大小写错直接编译失败;public static void main(String... args)虽能编译,但部分旧版JVM(如Java 5早期)可能不识别。
args本质是JVM传递给程序的命令行字符串数组,每个元素对应一个空格分隔的原始输入(不解析引号、不处理转义)。它不是配置中心,也不是自动类型转换器。
java MyApp
hello world → args = {"hello", "world"}
java MyApp "a b" c → args = {"a b", "c"}(shell完成引号剥离,JVM只接收切分后结果)args为空数组[],不是null;判空应写args.length == 0,而非args == null
容易踩的坑:有人试图在main里直接解析args做复杂配置,结果遇到路径含空格、特殊字符时出错;更稳妥的做法是用Apache Commons CLI或picocli这类库接管解析逻辑。
可以,但只有签名严格匹配public static void main(String[] args)的才会被JVM识别为入口点。其他变体属于普通静态方法,不会冲突,也不会被自动调用。
public static void main(String... args):重载,不作为入口public static int main(String[] args):返回类型不符,JVM忽略static void main(String[] args):缺少public,JVM抛IllegalAccessError
实际开发中,常利用这点写测试型main——比如在工具类里加一个临时main验证逻辑,只要签名正确,就能直接右键运行,不影响主流程。
Java 21引入main快捷写法(如void main()),但这只是编译器层面的语法糖,底层仍生成标准public static void main(String[] args)字节码。JVM本身没有任何改动。
void main() { ... } → 编译后仍是完整签名java HelloWorld.java直接运行源文件的场景.jar,或被其它类引用,就必须回归传统main签名,否则NoClassDefFoundError或NoSuchMethodError
真正容易被忽略的是:这个语法糖不解决args解析问题,也不影响类加载顺序、静态块执行时机,它只是省了一行声明——别把它当成“新入口模型”来理解。