场景
今天看jvm虚拟机初始化的阶段时候,发现有下面5种情况,会触发初始化
初始化阶段,虚拟机规范则是严格规定了有且只有5种情况必须立即对类进行“初始化”(而加载、验证、准备自然需要在此之前开始):
1)遇到new、getstatic、putstatic或invokestatic这4条字节码指令时,如果类没有进行过初始化,则需要先触发其初始化。生成这4条指令的最常见的Java代码场景是:使用new关键字实例化对象的时候、读取或设置一个类的静态字段(被final修饰、已在编译期把结果放入常量池的静态字段除外)的时候,以及调用一个类的静态方法的时候。
2)使用java.lang.reflect包的方法对类进行反射调用的时候,如果类没有进行过初始化,则需要先触发其初始化。
3)当初始化一个类的时候,如果发现其父类还没有进行过初始化,则需要先触发其父类的初始化。
4)当虚拟机启动时,用户需要指定一个要执行的主类(包含main()方法的那个类),虚拟机会先初始化这个主类。
5)当使用JDK 1.7的动态语言支持时,如果一个java.lang.invoke.MethodHandle实例最后的解析结果REF_getStatic、REF_putStatic、REF_invokeStatic的方法句柄,并且这个方法句柄所对应的类没有进行过初始化,则需要先触发其初始化。
其中第5条没接触过,饿补了一下MethodHandle的使用场景。
MethodHandle
JDK6之前我们会使用java反射来实现动态方法调用,多数框架用反射的比较多,例如mybatis、spring等。在JDK7中,新增了java.lang.invoke.MethodHandle(方法句柄),称之为“现代化反射”。其实反射和java.lang.invoke.MethodHandle都是间接调用方法的途径,但java.lang.invoke.MethodHandle比反射更简洁,用反射功能会写一大堆冗余代码。
官方api给出的解释:
A method handle is a typed, directly executable reference to an underlying method, constructor, field, or similar low-level operation, with optional transformations of arguments or return values.其实就是可以获取方法的句柄,类似方法的指针。
下面看一个例子,使用方法句柄(MethodHandle)调用toString()方法:
import java.lang.invoke.MethodHandle; import java.lang.invoke.MethodHandles; import java.lang.invoke.MethodType; /** * Created by hfl on 2020/12/11 */ public class MHTestDemo { public String toString(String s) { return "hello," + s + "MethodHandle"; } public static void main(String[] args) { MHTestDemo mhTest = new MHTestDemo(); MethodHandle mh = getToStringMH(); //获取方法句柄 try { // 1.调用方法: String result = (String) mh.invokeExact(mhTest, "ssssss"); //根据方法句柄调用方法----注意返回值必须强转 System.out.println(result); } catch (Throwable throwable) { throwable.printStackTrace(); } // 2.or like this: try { MethodHandle methodHandle2 = mh.bindTo(mhTest); String toString2 = (String) methodHandle2.invokeWithArguments("sssss"); System.out.println(toString2); } catch (Throwable throwable) { throwable.printStackTrace(); } // 得到当前Class的不同表示方法,最后一个最好。一般我们在静态上下文用SLF4J得到logger用。 System.out.println(MHTestDemo.class); System.out.println(mhTest.getClass()); System.out.println(MethodHandles.lookup().lookupClass()); // like getClass() } /** * 获取方法句柄 * @return */ public static MethodHandle getToStringMH() { //获取方法类型 参数为:1.返回值类型,2方法中参数类型 MethodType mt = MethodType.methodType(String.class, String.class); MethodHandle mh = null; try { //查找方法句柄 mh = MethodHandles.lookup().findVirtual(MHTestDemo.class, "toString", mt); } catch (NoSuchMethodException | IllegalAccessException e) { e.printStackTrace(); } return mh; } }
运行结果:
hello,ssssssMethodHandle hello,sssssMethodHandle class com.gupao.gpjvm.ch01.MHTestDemo class com.gupao.gpjvm.ch01.MHTestDemo class com.gupao.gpjvm.ch01.MHTestDemo
本文完。