public static void main(String[] args) {
A a = null;
a.doSth();//会报空指针异常吗?
}
public static class A {
public static void doSth() {
System.out.println("do something...");
}
}
如上述代码,a.doSth()会报空指针么?这段代码的具体执行过程是什么样的?
带着这个问题来分析。
doSth:类A中的一个static方法。
在JVM中它的执行过程是什么样的呢?
1、类加载:
如果类A还没有被加载到JVM中,那么首先就会加载类A:包括读取类的字节码(.class),并验证其完整性、安全性和其他约束。加载类时,还会解析类中的符号引用(如:对其他类或方法的引用)
2、方法解析:
在调用a.doSth()时,JVM需要解析方法引用。因为doSth()是静态的,JVM会在类A的静态方法表中查找这个方法。这涉及到将方法名(doSth)和描述符(方法的签名,包括返回类型和参数类型)映射到实际的方法字节码。
3、静态方法调用:
由于doSth()是静态的,JVM不会检查a是否为null。相反,它直接定位到类A中的静态方法doSth()的实现。这个过程不需要创建或引用任何对象实例,也不检查。
4、执行方法:
JVM跳转到doSth()方法的字节码,并开始执行它。这可能涉及到将局部变量加载到操作数栈中,执行各种字节码指令(如加法、条件跳转等),以及可能的方法调用或异常处理。
5、返回结果:
如果doSth()方法有返回值,JVM会将结果放置在操作数栈的顶部,并准备返回给调用者。如果doSth()是void方法(即没有返回值),则只需返回到调用者。
6、清理和结束:
在方法执行完毕后,JVM会执行任何必要的清理工作,如释放局部变量和释放栈帧。然后,控制流返回到调用a.doSth()的地方,继续执行后续的代码。
(有所简化,如果是面试前看到这篇,就再拓展学习下相关的知识点吧)
输出结果:(符合预期,不会报NPE)
最后,要强调的是,尽管静态方法可以通过对象实例来调用,但这并不是推荐的做法,因为它可能会造成混淆。更好的做法自然是直接通过类名来调用静态方法,如A.doSth()。这样做不仅更清晰,也避免了因对象实例为null而可能产生的误解。
简单归简单,但是在实际项目中也总会有类似的情况出现~~~