- 先看题目
package 子类静态方法不会执行; class Father{ static String fatherString = "abc"; static{ System.out.println("father static "); } } class Son extends Father{ static{ System.out.println("son static"); } } public class Test { public static void main(String[] args) throws Exception{ System.out.println(Son.fatherString); } }
输出的是
father static abc
-
问题是: 为什么会这样呢 ? 这牵扯到类的加载和类的初始化,类加载是指将编译后的 class文件加载进虚拟机,初始化是指静态代码块执行,静态变量的赋值
-
使用 -XX:+TraceClassLoading 来看Java程序执行的时候加载了什么类(省略了一些),可以看到父类和子类都被加载进了虚拟机
[Loaded sun.misc.FloatingDecimal$ASCIIToBinaryConverter from D:\jdk1.8.0_20\jre\lib\rt.jar] [Loaded sun.misc.FloatingDecimal$PreparedASCIIToBinaryBuffer from D:\jdk1.8.0_20\jre\lib\rt.jar] [Loaded 子类静态方法不会执行.Test from file:/K:/%e9%9d%a2%e8%af%95%e8%b5%84%e6%96%99/%e7%9f%a5%e4%b9%8e%e9%9d%a2%e8%af%95%e9%a2%98/out/production/%e7%9f%a5%e4%b9%8e%e9%9d%a2%e8%af%95%e9%a2%98/] [Loaded sun.launcher.LauncherHelper$FXHelper from D:\jdk1.8.0_20\jre\lib\rt.jar] [Loaded java.lang.Class$MethodArray from D:\jdk1.8.0_20\jre\lib\rt.jar] [Loaded sun.misc.FloatingDecimal$ASCIIToBinaryBuffer from D:\jdk1.8.0_20\jre\lib\rt.jar] [Loaded java.net.DualStackPlainSocketImpl from D:\jdk1.8.0_20\jre\lib\rt.jar] [Loaded 子类静态方法不会执行.Father from file:/K:/%e9%9d%a2%e8%af%95%e8%b5%84%e6%96%99/%e7%9f%a5%e4%b9%8e%e9%9d%a2%e8%af%95%e9%a2%98/out/production/%e7%9f%a5%e4%b9%8e%e9%9d%a2%e8%af%95%e9%a2%98/] [Loaded 子类静态方法不会执行.Son from file:/K:/%e9%9d%a2%e8%af%95%e8%b5%84%e6%96%99/%e7%9f%a5%e4%b9%8e%e9%9d%a2%e8%af%95%e9%a2%98/out/production/%e7%9f%a5%e4%b9%8e%e9%9d%a2%e8%af%95%e9%a2%98/] [Loaded java.net.SocksSocketImpl$3 from D:\jdk1.8.0_20\jre\lib\rt.jar] father static [Loaded java.net.ProxySelector from D:\jdk1.8.0_20\jre\lib\rt.jar] abc [Loaded sun.net.spi.DefaultProxySelector from D:\jdk1.8.0_20\jre\lib\rt.jar] [Loaded sun.net.spi.DefaultProxySelector$1 from D:\jdk1.8.0_20\jre\lib\rt.jar] [Loaded java.lang.Shutdown from D:\jdk1.8.0_20\jre\lib\rt.jar] [Loaded java.lang.Shutdown$Lock from D:\jdk1.8.0_20\jre\lib\rt.jar] [Loaded sun.net.NetProperties from D:\jdk1.8.0_20\jre\lib\rt.jar]
-
在说一下类在什么时候会被初始化
1.遇到new、getstatic、putstatic、或invokestatic这4条字节码指令时,如果类没有进行过初始化,则需要先触发其初始化。
2.使用java.lang.reflect包的方法对类进行反射调用的时候,如果类么有进行过初始化,则需要先触发其初始化。 3.当初始化类的时候,如果发现其父类还没有初始化,则必须先触发其父类的初始化。 4.虚拟机启动时,用户需要指定一个要执行的主类(包含main()方法的那个类),虚拟机会先初始化这个主类 5.当使用JDK1.7的动态语言支持时,如果一个java.lang.invoke.MethodHandle实例最后的解析结果REF_getStatic、REF_putStatic、REF_invokeStatic的方法句柄,并且这个放啊放句柄所对应的类还没有进行初始化,则需要先触发其初始化。 所以,在执行以上程序的时候,也就是Son.fatherString 的时候,并不满足Son类的初始化条件,但是满足了Father 类初始化条件,所以父类的静态方法执行了。