From 822948d3017586b3d8dd9aab23d3897c555ba8f4 Mon Sep 17 00:00:00 2001 From: Omooo <869759698@qq.com> Date: Wed, 7 Aug 2019 15:45:33 +0800 Subject: [PATCH] =?UTF-8?q?Create=20JVM=20=E6=98=AF=E5=A6=82=E4=BD=95?= =?UTF-8?q?=E5=AE=9E=E7=8E=B0=E5=8F=8D=E5=B0=84=E7=9A=84=EF=BC=9F.md?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- blogs/JVM/JVM 是如何实现反射的?.md | 171 +++++++++++++++++++ 1 file changed, 171 insertions(+) create mode 100644 blogs/JVM/JVM 是如何实现反射的?.md diff --git a/blogs/JVM/JVM 是如何实现反射的?.md b/blogs/JVM/JVM 是如何实现反射的?.md new file mode 100644 index 0000000..5818058 --- /dev/null +++ b/blogs/JVM/JVM 是如何实现反射的?.md @@ -0,0 +1,171 @@ +--- +JVM 是如何实现反射的? +--- + +#### 目录 + +1. 前言 +2. 反射调用的实现 +3. 反射调用的开销 +4. 总结 + +#### 前言 + +反射是 Java 语言中一个相当重要的特性,它允许正在运行的 Java 程序观测,甚至是修改程序的动态行为。 + +举例来说,我们可以通过 Class 对象枚举该类中的所有方法,我们还可以通过 Method.setAccessible(位于 java.lang.reflect 包,该方法继承自 AccessibleObject)绕过 Java 语言的访问权限,在私有方法所在类之外的地方调用该方法。 + +今天我们便来了解一下反射的实现机制,以及它性能糟糕的原因。 + +#### 反射调用的实现 + +首先,我们来看看方法的反射调用,也就是 Method.invoke,是怎么实现的。 + +```java +public final class Method extends Executable { + ... + public Object invoke(Object obj, Object... args) throws ... { + ... // 权限检查 + MethodAccessor ma = methodAccessor; + if (ma == null) { + ma = acquireMethodAccessor(); + } + return ma.invoke(obj, args); + } +} +``` + +如果你查阅 Method.invoke 的源代码,那么你会发现,它实际上委派给 MethodAccessor 来处理。MethodAccessor 是一个接口,它有两个已有的具体实现:一个通过本地方法来实现反射调用(NativeMethodAccessorImpl),另一个则使用了委派模式(DelegatingMethodAccessorImpl)。为了方便记忆,我便用 “本地实现” 和 “委派实现” 来指代这两者。 + +每个 Method 实例的第一次反射调用都会生成一个委派实现,它所委派的具体实现便是一个本地实现。本地实现非常容易理解,当进入了 Java 虚拟机内部之后,我们便拥有了 Method 实例所指向方法的具体地址。这时候,反射调用无非就是将传入的参数准备好,然后调用进入目标方法。 + +```java +public class Main { + + public static void main(String[] args) throws Exception { + Class clazz = Class.forName("Main"); + Method method = clazz.getMethod("target", int.class); + method.invoke(null, 233); + } + + public static void target(int i) { + new Exception("#" + i).printStackTrace(); + } +} + +java.lang.Exception: #233 + at Main.target(Main.java:16) + at java.base/jdk.internal.reflect.NativeMethodAccessorImpl.invoke0(Native Method) + at java.base/jdk.internal.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:62) + at java.base/jdk.internal.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43) + at java.base/java.lang.reflect.Method.invoke(Method.java:566) + at Main.main(Main.java:12) +``` + +为了方便理解,我们可以打印一下反射调用到目标方法时的栈轨迹。 + +可以看到,反射调用先是调用了 Method.invoke,然后进去委派实现(DelegatingMethodAccessorImpl),再然后进入本地实现(NativeMethodAccessorImpl),最后到达目标方法。 + +这里你可能会疑问,为什么反射调用还要采取委派实现作为中间层?直接交给本地实现不可以么? + +其实,Java 的反射调用机制还设立了另外一种动态生成字节码的实现(下称动态实现),直接使用 invoke 指令来调用目标方法。之所以采取委派实现,便是为了能够在本地实现以及动态实现中切换。 + +```java +// 动态实现的伪代码,这里只列举了关键的调用逻辑,其实它还包括调用者检测、参数检测的字节码。 +package jdk.internal.reflect; + +public class GeneratedMethodAccessor1 extends ... { + @Overrides + public Object invoke(Object obj, Object[] args) throws ... { + Test.target((int) args[0]); + return null; + } +} +``` + +动态实现和本地实现相比,其运行效率要快上 20 倍。这是因为动态实现无需经过 Java 到 C++ 再到 Java 的切换,但由于生成字节码十分耗时,仅调用一次的话,反而是本地实现要快上 3 到 4 倍。 + +考虑到很多反射调用仅会执行一次,Java 虚拟机设置了一个阈值 15,当某个反射调用的调用次数在 15 之下时,采用本地实现;当达到 15 时,便开始动态生成字节码,并将委派实现的委派对象切换至动态实现,这个过程我们称之为 Inflation。 + +```java +public class Main { + public static void main(String[] args) throws Exception { + Class clazz = Class.forName("Main"); + Method method = clazz.getMethod("target", int.class); + for (int i = 0; i < 20; i++) { + method.invoke(null, i); + } + } + + public static void target(int i) { + new Exception("#" + i).printStackTrace(); + } +} + +# 使用 -verbose:class 打印加载的类 +$ java -verbose:class Main + + +[0.216s][info][class,load] java.util.IdentityHashMap$KeySet source: jrt:/java.base +java.lang.Exception: #0 +[0.216s][info][class,load] java.lang.StackTraceElement$HashedModules source: jrt:/java.base + at Main.target(Main.java:17) + at java.base/jdk.internal.reflect.NativeMethodAccessorImpl.invoke0(Native Method) + at java.base/jdk.internal.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:62) + at java.base/jdk.internal.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43) + at java.base/java.lang.reflect.Method.invoke(Method.java:566) + at Main.main(Main.java:12) + + +... //省略 + +[0.234s][info][class,load] sun.reflect.misc.ReflectUtil source: jrt:/java.base +[0.234s][info][class,load] jdk.internal.reflect.ClassFileConstants source: jrt:/java.base +[0.234s][info][class,load] jdk.internal.reflect.AccessorGenerator source: jrt:/java.base +[0.235s][info][class,load] jdk.internal.reflect.MethodAccessorGenerator source: jrt:/java.base +[0.235s][info][class,load] jdk.internal.reflect.ByteVectorFactory source: jrt:/java.base +[0.235s][info][class,load] jdk.internal.reflect.ByteVector source: jrt:/java.base +[0.236s][info][class,load] jdk.internal.reflect.ByteVectorImpl source: jrt:/java.base +[0.236s][info][class,load] jdk.internal.reflect.ClassFileAssembler source: jrt:/java.base +[0.236s][info][class,load] jdk.internal.reflect.UTF8 source: jrt:/java.base +[0.236s][info][class,load] jdk.internal.reflect.Label source: jrt:/java.base +[0.237s][info][class,load] jdk.internal.reflect.Label$PatchInfo source: jrt:/java.base +[0.238s][info][class,load] jdk.internal.reflect.MethodAccessorGenerator$1 source: jrt:/java.base +[0.238s][info][class,load] jdk.internal.reflect.ClassDefiner source: jrt:/java.base +[0.238s][info][class,load] jdk.internal.reflect.ClassDefiner$1 source: jrt:/java.base +[0.239s][info][class,load] jdk.internal.reflect.GeneratedMethodAccessor1 source: __JVM_DefineClass__ +java.lang.Exception: #15 + at Main.target(Main.java:17) + at java.base/jdk.internal.reflect.NativeMethodAccessorImpl.invoke0(Native Method) + at java.base/jdk.internal.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:62) + at java.base/jdk.internal.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43) + at java.base/java.lang.reflect.Method.invoke(Method.java:566) + at Main.main(Main.java:12) +java.lang.Exception: #16 + at Main.target(Main.java:17) + at jdk.internal.reflect.GeneratedMethodAccessor1.invoke(Unknown Source) + at java.base/jdk.internal.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43) + at java.base/java.lang.reflect.Method.invoke(Method.java:566) + at Main.main(Main.java:12) +``` + +可以看到,在第 15 次(从 0 开始数)反射调用时,我们便触发了动态实现的生成。这时候,Java 虚拟机额外加载了不少类。其中,最重要的当属 GeneratedMethodAccessor1。并且,从第 16 次反射调用开始,我们便切换至这个刚刚生成的动态实现。 + +反射调用的 Inflation 机制是可以通过参数来关闭的。这样一来,在反射调用一开始便会直接生成动态实现,而不会使用委派实现或者本地实现。 + +#### 反射调用的开销 + +下面,我们便来拆解反射调用的性能开销。 + +在刚才的例子中,我们先后进行了 Class.forName,Class.getMethod 以及 Method.invoke 三个操作。其中,Class.forName 会调用本地方法,Class.getMethod 则会遍历该类的公有方法。如果没有匹配到,它还将遍历父类的公有方法。可想而知,这两个操作都非常耗时。 + +值得注意的是,以 getMethod 为代表的查找方法操作,会返回查找得到结果的一份拷贝。因此,我们应当避免在热点代码中使用返回 Method 数组的 getMethods 或者 getDeclaredMethods 方法,以减少不必要的堆空间消耗。 + +在实践中,我们往往会在应用程序中缓存 Class.forName 和 Class.getMethod 的结果。因此,下面我们就只关注反射调用本身的性能开销。 + +#### 总结 + +在默认情况下,方法的反射调用为委派实现,委派给本地实现来进行方法调用。在调用超过 15 次之后,委派实现便会将委派对象切换至动态实现。这个动态实现的字节码是自动生成的,它将直接使用 invoke 指令来调用目标方法。 + +方法的反射调用会带来不少性能开销,原因主要有三个:变成参数方法导致的 Object 数组,基于类型的自动装箱、拆箱,还有最重要的方法内联。 +