场景

今天看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


本文完。