久久福利_99r_国产日韩在线视频_直接看av的网站_中文欧美日韩_久久一

您的位置:首頁(yè)技術(shù)文章
文章詳情頁(yè)

JAVA深入探究之Method的Invoke方法

瀏覽:43日期:2022-08-18 15:40:40
前言

在寫代碼的時(shí)候,發(fā)現(xiàn)從父類class通過(guò)getDeclaredMethod獲取的Method可以調(diào)用子類的對(duì)象,而子類改寫了這個(gè)方法,從子類class通過(guò)getDeclaredMethod也能獲取到Method,這時(shí)去調(diào)用父類的對(duì)象也會(huì)報(bào)錯(cuò)。雖然這是很符合多態(tài)的現(xiàn)象,也符合java的動(dòng)態(tài)綁定規(guī)范,但還是想弄懂java是如何實(shí)現(xiàn)的,就學(xué)習(xí)了下Method的源代碼。

Method的invoke方法

1.先檢查 AccessibleObject的override屬性是否為true。

AccessibleObject是Method,Field,Constructor的父類,override屬性默認(rèn)為false,可調(diào)用setAccessible方法改變,如果設(shè)置為true,則表示可以忽略訪問(wèn)權(quán)限的限制,直接調(diào)用。

2.如果不是ture,則要進(jìn)行訪問(wèn)權(quán)限檢測(cè)。用Reflection的quickCheckMemberAccess方法先檢查是不是public的,如果不是再用Reflection.getCallerClass(1)方法獲

得到調(diào)用這個(gè)方法的Class,然后做是否有權(quán)限訪問(wèn)的校驗(yàn),校驗(yàn)之后緩存一次,以便下次如果還是這個(gè)類來(lái)調(diào)用就不用去做校驗(yàn)了,直接用上次的結(jié)果,(很奇怪用這種方式緩存,因?yàn)檫@種方式如果下次換個(gè)類來(lái)調(diào)用的話,就不用會(huì)緩存了,而再驗(yàn)證一遍,把這次的結(jié)果做為緩存,但上一次的緩存結(jié)果就被沖掉了。這是一個(gè)很簡(jiǎn)單的緩沖機(jī)制,只適用于一個(gè)類的重復(fù)調(diào)用)。

3.調(diào)用MethodAccessor的invoke方法。每個(gè)Method對(duì)象包含一個(gè)root對(duì)象,root對(duì)象里持有一個(gè)MethodAccessor對(duì)象。我們獲得的Method獨(dú)享相當(dāng)于一個(gè)root對(duì)象的鏡像,所有這類Method共享root里的MethodAccessor對(duì)象,(這個(gè)對(duì)象由ReflectionFactory方法生成,ReflectionFactory對(duì)象在Method類中是static final的由native方法實(shí)例化)。

ReflectionFactory生成MethodAccessor:如果noInflation的屬性為true則直接返回MethodAccessorGenerator創(chuàng)建的一個(gè)MethodAccessor。

否則返回DelegatingMethodAccessorImpl,并將他與一個(gè)NativeMethodAccessorImpl互相引用。但DelegatingMethodAccessorImpl執(zhí)行invoke方法的時(shí)候又委托給NativeMethodAccessorImpl了。

再一步深入

4.NativeMethodAccessorImpl的invkoe方法:

調(diào)用natiave方法invoke0執(zhí)行方法調(diào)用.

注意這里有一個(gè)計(jì)數(shù)器numInvocations,每調(diào)用一次方法+1,當(dāng)比 ReflectionFactory.inflationThreshold(15)大的時(shí)候,用MethodAccessorGenerator創(chuàng)建一個(gè)MethodAccessor,并把之前的DelegatingMethodAccessorImpl引用替換為現(xiàn)在新創(chuàng)建的。下一次DelegatingMethodAccessorImpl就不會(huì)再交給NativeMethodAccessorImpl執(zhí)行了,而是交給新生成的java字節(jié)碼的MethodAccessor。

MethodAccessorGenerator使用了asm字節(jié)碼動(dòng)態(tài)加載技術(shù),暫不深入研究。

總結(jié) 一個(gè)方法可以生成多個(gè)Method對(duì)象,但只有一個(gè)root對(duì)象,主要用于持有一個(gè)MethodAccessor對(duì)象,這個(gè)對(duì)象也可以認(rèn)為一個(gè)方法只有一個(gè),相當(dāng)于是static的。因?yàn)镸ethod的invoke是交給MethodAccessor執(zhí)行的,所以我所想要知道的答案在MethodAccessor的invoke中,深入MethodAccessor:

MethodAccessor

public class A {public void foo(String name) {System.out.println('Hello, ' + name);}}

可以編寫另外一個(gè)類來(lái)反射調(diào)用A上的方法:

import java.lang.reflect.Method;public class TestClassLoad {public static void main(String[] args) throws Exception {Class<?> clz = Class.forName('A');Object o = clz.newInstance();Method m = clz.getMethod('foo', String.class);for (int i = 0; i < 16; i++) {m.invoke(o, Integer.toString(i));}}}

注意到TestClassLoad類上不會(huì)有對(duì)類A的符號(hào)依賴——也就是說(shuō)在加載并初始化TestClassLoad類時(shí)不需要關(guān)心類A的存在與否,而是等到main()方法執(zhí)行到調(diào)用Class.forName()時(shí)才試圖對(duì)類A做動(dòng)態(tài)加載;這里用的是一個(gè)參數(shù)版的forName(),也就是使用當(dāng)前方法所在類的ClassLoader來(lái)加載,并且初始化新加載的類。……好吧這個(gè)細(xì)節(jié)跟主題沒(méi)啥關(guān)系。

回到主題。這次我的測(cè)試環(huán)境是Sun的JDK 1.6.0 update 13 build 03。編譯上述代碼,并在執(zhí)行TestClassLoad時(shí)加入-XX:+TraceClassLoading參數(shù)(或者-verbose:class或者直接-verbose都行),如下:

控制臺(tái)命令

java -XX:+TraTestClassLoad ceClassLoading

可以看到輸出了一大堆log,把其中相關(guān)的部分截取出來(lái)如下:

[Loaded TestClassLoad from file:/D:/temp_code/test_java_classload/][Loaded A from file:/D:/temp_code/test_java_classload/][Loaded sun.reflect.NativeMethodAccessorImpl from shared objects file][Loaded sun.reflect.DelegatingMethodAccessorImpl from shared objects file]Hello, 0Hello, 1Hello, 2Hello, 3Hello, 4Hello, 5Hello, 6Hello, 7Hello, 8Hello, 9Hello, 10Hello, 11Hello, 12Hello, 13Hello, 14[Loaded sun.reflect.ClassFileConstants from shared objects file][Loaded sun.reflect.AccessorGenerator from shared objects file][Loaded sun.reflect.MethodAccessorGenerator from shared objects file][Loaded sun.reflect.ByteVectorFactory from shared objects file][Loaded sun.reflect.ByteVector from shared objects file][Loaded sun.reflect.ByteVectorImpl from shared objects file][Loaded sun.reflect.ClassFileAssembler from shared objects file][Loaded sun.reflect.UTF8 from shared objects file][Loaded java.lang.Void from shared objects file][Loaded sun.reflect.Label from shared objects file][Loaded sun.reflect.Label$PatchInfo from shared objects file][Loaded java.util.AbstractList$Itr from shared objects file][Loaded sun.reflect.MethodAccessorGenerator$1 from shared objects file][Loaded sun.reflect.ClassDefiner from shared objects file][Loaded sun.reflect.ClassDefiner$1 from shared objects file][Loaded sun.reflect.GeneratedMethodAccessor1 from __JVM_DefineClass__]Hello, 15

可以看到前15次反射調(diào)用A.foo()方法并沒(méi)有什么稀奇的地方,但在第16次反射調(diào)用時(shí)似乎有什么東西被觸發(fā)了,導(dǎo)致JVM新加載了一堆類,其中就包括[Loaded sun.reflect.GeneratedMethodAccessor1 from __JVM_DefineClass__]這么一行。這是哪里來(lái)的呢?

先來(lái)看看JDK里Method.invoke()是怎么實(shí)現(xiàn)的。

java.lang.reflect.Method:

public final class Method extends AccessibleObject implements GenericDeclaration, Member { // ... private volatile MethodAccessor methodAccessor; // For sharing of MethodAccessors. This branching structure is // currently only two levels deep (i.e., one root Method and // potentially many Method objects pointing to it.) private Method root; // ... public Object invoke(Object obj, Object... args) throws IllegalAccessException, IllegalArgumentException, InvocationTargetException { if (!override) { if (!Reflection.quickCheckMemberAccess(clazz, modifiers)) { Class caller = Reflection.getCallerClass(1); Class targetClass = ((obj == null || !Modifier.isProtected(modifiers)) ? clazz : obj.getClass()); boolean cached; synchronized (this) { cached = (securityCheckCache == caller) && (securityCheckTargetClassCache == targetClass); } if (!cached) { Reflection.ensureMemberAccess(caller, clazz, obj, modifiers); synchronized (this) { securityCheckCache = caller; securityCheckTargetClassCache = targetClass; } } } } if (methodAccessor == null) acquireMethodAccessor(); return methodAccessor.invoke(obj, args); } // NOTE that there is no synchronization used here. It is correct // (though not efficient) to generate more than one MethodAccessor // for a given Method. However, avoiding synchronization will // probably make the implementation more scalable. private void acquireMethodAccessor() { // First check to see if one has been created yet, and take it // if so MethodAccessor tmp = null; if (root != null) tmp = root.getMethodAccessor(); if (tmp != null) { methodAccessor = tmp; return; } // Otherwise fabricate one and propagate it up to the root tmp = reflectionFactory.newMethodAccessor(this); setMethodAccessor(tmp); } // ...}

可以看到Method.invoke()實(shí)際上并不是自己實(shí)現(xiàn)的反射調(diào)用邏輯,而是委托給sun.reflect.MethodAccessor來(lái)處理。 每個(gè)實(shí)際的Java方法只有一個(gè)對(duì)應(yīng)的Method對(duì)象作為root,。這個(gè)root是不會(huì)暴露給用戶的,而是每次在通過(guò)反射獲取Method對(duì)象時(shí)新創(chuàng)建Method對(duì)象把root包裝起來(lái)再給用戶。在第一次調(diào)用一個(gè)實(shí)際Java方法對(duì)應(yīng)得Method對(duì)象的invoke()方法之前,實(shí)現(xiàn)調(diào)用邏輯的MethodAccessor對(duì)象還沒(méi)創(chuàng)建;等第一次調(diào)用時(shí)才新創(chuàng)建MethodAccessor并更新給root,然后調(diào)用MethodAccessor.invoke()真正完成反射調(diào)用。

那么MethodAccessor是啥呢?

sun.reflect.MethodAccessor:

public interface MethodAccessor { /** Matches specification in {@link java.lang.reflect.Method} */ public Object invoke(Object obj, Object[] args) throws IllegalArgumentException, InvocationTargetException;}

可以看到它只是一個(gè)單方法接口,其invoke()方法與Method.invoke()的對(duì)應(yīng)。

創(chuàng)建MethodAccessor實(shí)例的是ReflectionFactory。

sun.reflect.ReflectionFactory:

public class ReflectionFactory { private static boolean initted = false; // ... // // 'Inflation' mechanism. Loading bytecodes to implement // Method.invoke() and Constructor.newInstance() currently costs // 3-4x more than an invocation via native code for the first // invocation (though subsequent invocations have been benchmarked // to be over 20x faster). Unfortunately this cost increases // startup time for certain applications that use reflection // intensively (but only once per class) to bootstrap themselves. // To avoid this penalty we reuse the existing JVM entry points // for the first few invocations of Methods and Constructors and // then switch to the bytecode-based implementations. // // Package-private to be accessible to NativeMethodAccessorImpl // and NativeConstructorAccessorImpl private static boolean noInflation = false; private static int inflationThreshold = 15; // ... /** We have to defer full initialization of this class until after the static initializer is run since java.lang.reflect.Method’s static initializer (more properly, that for java.lang.reflect.AccessibleObject) causes this class’s to be run, before the system properties are set up. */ private static void checkInitted() { if (initted) return; AccessController.doPrivileged(new PrivilegedAction() { public Object run() { // Tests to ensure the system properties table is fully // initialized. This is needed because reflection code is // called very early in the initialization process (before // command-line arguments have been parsed and therefore // these user-settable properties installed.) We assume that // if System.out is non-null then the System class has been // fully initialized and that the bulk of the startup code // has been run. if (System.out == null) { // java.lang.System not yet fully initialized return null; } String val = System.getProperty('sun.reflect.noInflation'); if (val != null && val.equals('true')) { noInflation = true; } val = System.getProperty('sun.reflect.inflationThreshold'); if (val != null) { try { inflationThreshold = Integer.parseInt(val); } catch (NumberFormatException e) { throw (RuntimeException) new RuntimeException('Unable to parse property sun.reflect.inflationThreshold'). initCause(e); } } initted = true; return null; } }); } // ... public MethodAccessor newMethodAccessor(Method method) { checkInitted(); if (noInflation) { return new MethodAccessorGenerator(). generateMethod(method.getDeclaringClass(), method.getName(), method.getParameterTypes(), method.getReturnType(), method.getExceptionTypes(), method.getModifiers()); } else { NativeMethodAccessorImpl acc = new NativeMethodAccessorImpl(method); DelegatingMethodAccessorImpl res = new DelegatingMethodAccessorImpl(acc); acc.setParent(res); return res; } }}

這里就可以看到有趣的地方了。如注釋所述,實(shí)際的MethodAccessor實(shí)現(xiàn)有兩個(gè)版本,一個(gè)是Java實(shí)現(xiàn)的,另一個(gè)是native code實(shí)現(xiàn)的。Java實(shí)現(xiàn)的版本在初始化時(shí)需要較多時(shí)間,但長(zhǎng)久來(lái)說(shuō)性能較好;native版本正好相反,啟動(dòng)時(shí)相對(duì)較快,但運(yùn)行時(shí)間長(zhǎng)了之后速度就比不過(guò)Java版了。這是HotSpot的優(yōu)化方式帶來(lái)的性能特性,同時(shí)也是許多虛擬機(jī)的共同點(diǎn):跨越native邊界會(huì)對(duì)優(yōu)化有阻礙作用,它就像個(gè)黑箱一樣讓虛擬機(jī)難以分析也將其內(nèi)聯(lián),于是運(yùn)行時(shí)間長(zhǎng)了之后反而是托管版本的代碼更快些。

為了權(quán)衡兩個(gè)版本的性能,Sun的JDK使用了“inflation”的技巧:讓Java方法在被反射調(diào)用時(shí),開頭若干次使用native版,等反射調(diào)用次數(shù)超過(guò)閾值時(shí)則生成一個(gè)專用的MethodAccessor實(shí)現(xiàn)類,生成其中的invoke()方法的字節(jié)碼,以后對(duì)該Java方法的反射調(diào)用就會(huì)使用Java版。

Sun的JDK是從1.4系開始采用這種優(yōu)化的。

PS.可以在啟動(dòng)命令里加上-Dsun.reflect.noInflation=true,就會(huì)RefactionFactory的noInflation屬性就變成true了,這樣不用等到15調(diào)用后,程序一開始就會(huì)用java版的MethodAccessor了。

上面看到了ReflectionFactory.newMethodAccessor()生產(chǎn)MethodAccessor的邏輯,在“開頭若干次”時(shí)用到的DelegatingMethodAccessorImpl代碼如下:

sun.reflect.DelegatingMethodAccessorImpl:

/** Delegates its invocation to another MethodAccessorImpl and can change its delegate at run time. */class DelegatingMethodAccessorImpl extends MethodAccessorImpl { private MethodAccessorImpl delegate; DelegatingMethodAccessorImpl(MethodAccessorImpl delegate) { setDelegate(delegate); } public Object invoke(Object obj, Object[] args) throws IllegalArgumentException, InvocationTargetException { return delegate.invoke(obj, args); } void setDelegate(MethodAccessorImpl delegate) { this.delegate = delegate; }}

這是一個(gè)間接層,方便在native與Java版的MethodAccessor之間實(shí)現(xiàn)切換。

然后下面就是native版MethodAccessor的Java一側(cè)的聲明:

sun.reflect.NativeMethodAccessorImpl:

/** Used only for the first few invocations of a Method; afterward, switches to bytecode-based implementation */class NativeMethodAccessorImpl extends MethodAccessorImpl { private Method method; private DelegatingMethodAccessorImpl parent; private int numInvocations; NativeMethodAccessorImpl(Method method) { this.method = method; } public Object invoke(Object obj, Object[] args) throws IllegalArgumentException, InvocationTargetException { if (++numInvocations > ReflectionFactory.inflationThreshold()) { MethodAccessorImpl acc = (MethodAccessorImpl) new MethodAccessorGenerator(). generateMethod(method.getDeclaringClass(), method.getName(), method.getParameterTypes(), method.getReturnType(), method.getExceptionTypes(), method.getModifiers()); parent.setDelegate(acc); } return invoke0(method, obj, args); } void setParent(DelegatingMethodAccessorImpl parent) { this.parent = parent; } private static native Object invoke0(Method m, Object obj, Object[] args);}

每次NativeMethodAccessorImpl.invoke()方法被調(diào)用時(shí),都會(huì)增加一個(gè)調(diào)用次數(shù)計(jì)數(shù)器,看超過(guò)閾值沒(méi)有;一旦超過(guò),則調(diào)用MethodAccessorGenerator.generateMethod()來(lái)生成Java版的MethodAccessor的實(shí)現(xiàn)類,并且改變DelegatingMethodAccessorImpl所引用的MethodAccessor為Java版。后續(xù)經(jīng)由DelegatingMethodAccessorImpl.invoke()調(diào)用到的就是Java版的實(shí)現(xiàn)了。

注意到關(guān)鍵的invoke0()方法是個(gè)native方法。它在HotSpot VM里是由JVM_InvokeMethod()函數(shù)所支持的:

由C編寫

JNIEXPORT jobject JNICALL Java_sun_reflect_NativeMethodAccessorImpl_invoke0(JNIEnv *env, jclass unused, jobject m, jobject obj, jobjectArray args){ return JVM_InvokeMethod(env, m, obj, args);}

JVM_ENTRY(jobject, JVM_InvokeMethod(JNIEnv *env, jobject method, jobject obj, jobjectArray args0)) JVMWrapper('JVM_InvokeMethod'); Handle method_handle; if (thread->stack_available((address) &method_handle) >= JVMInvokeMethodSlack) { method_handle = Handle(THREAD, JNIHandles::resolve(method)); Handle receiver(THREAD, JNIHandles::resolve(obj)); objArrayHandle args(THREAD, objArrayOop(JNIHandles::resolve(args0))); oop result = Reflection::invoke_method(method_handle(), receiver, args, CHECK_NULL); jobject res = JNIHandles::make_local(env, result); if (JvmtiExport::should_post_vm_object_alloc()) { oop ret_type = java_lang_reflect_Method::return_type(method_handle()); assert(ret_type != NULL, 'sanity check: ret_type oop must not be NULL!'); if (java_lang_Class::is_primitive(ret_type)) { // Only for primitive type vm allocates memory for java object. // See box() method. JvmtiExport::post_vm_object_alloc(JavaThread::current(), result); } } return res; } else { THROW_0(vmSymbols::java_lang_StackOverflowError()); }JVM_END

其中的關(guān)鍵又是Reflection::invoke_method():

// This would be nicer if, say, java.lang.reflect.Method was a subclass// of java.lang.reflect.Constructoroop Reflection::invoke_method(oop method_mirror, Handle receiver, objArrayHandle args, TRAPS) { oop mirror = java_lang_reflect_Method::clazz(method_mirror); int slot = java_lang_reflect_Method::slot(method_mirror); bool override = java_lang_reflect_Method::override(method_mirror) != 0; objArrayHandle ptypes(THREAD, objArrayOop(java_lang_reflect_Method::parameter_types(method_mirror))); oop return_type_mirror = java_lang_reflect_Method::return_type(method_mirror); BasicType rtype; if (java_lang_Class::is_primitive(return_type_mirror)) { rtype = basic_type_mirror_to_basic_type(return_type_mirror, CHECK_NULL); } else { rtype = T_OBJECT; } instanceKlassHandle klass(THREAD, java_lang_Class::as_klassOop(mirror)); methodOop m = klass->method_with_idnum(slot); if (m == NULL) { THROW_MSG_0(vmSymbols::java_lang_InternalError(), 'invoke'); } methodHandle method(THREAD, m); return invoke(klass, method, receiver, override, ptypes, rtype, args, true, THREAD);}

再下去就深入到HotSpot VM的內(nèi)部了,本文就在這里打住吧。有同學(xué)有興趣深究的話以后可以再寫一篇討論native版的實(shí)現(xiàn)。

回到Java的一側(cè)。MethodAccessorGenerator長(zhǎng)啥樣呢?由于代碼太長(zhǎng),這里就不完整貼了,有興趣的可以到OpenJDK 6的Mercurial倉(cāng)庫(kù)看:OpenJDK 6 build 17的MethodAccessorGenerator。它的基本工作就是在內(nèi)存里生成新的專用Java類,并將其加載。就貼這么一個(gè)方法:

private static synchronized String generateName(boolean isConstructor, boolean forSerialization){ if (isConstructor) { if (forSerialization) { int num = ++serializationConstructorSymnum; return 'sun/reflect/GeneratedSerializationConstructorAccessor' + num; } else { int num = ++constructorSymnum; return 'sun/reflect/GeneratedConstructorAccessor' + num; } } else { int num = ++methodSymnum; return 'sun/reflect/GeneratedMethodAccessor' + num; }}

去閱讀源碼的話,可以看到MethodAccessorGenerator是如何一點(diǎn)點(diǎn)把Java版的MethodAccessor實(shí)現(xiàn)類生產(chǎn)出來(lái)的。也可以看到GeneratedMethodAccessor+數(shù)字這種名字是從哪里來(lái)的了,就在上面的generateName()方法里。 對(duì)本文開頭的例子的A.foo(),生成的Java版MethodAccessor大致如下:

package sun.reflect;public class GeneratedMethodAccessor1 extends MethodAccessorImpl { public GeneratedMethodAccessor1() { super(); } public Object invoke(Object obj, Object[] args) throws IllegalArgumentException, InvocationTargetException { // prepare the target and parameters if (obj == null) throw new NullPointerException(); try { A target = (A) obj; if (args.length != 1) throw new IllegalArgumentException(); String arg0 = (String) args[0]; } catch (ClassCastException e) { throw new IllegalArgumentException(e.toString()); } catch (NullPointerException e) { throw new IllegalArgumentException(e.toString()); } // make the invocation try { target.foo(arg0); } catch (Throwable t) { throw new InvocationTargetException(t); } }}

就反射調(diào)用而言,這個(gè)invoke()方法非常干凈(然而就“正常調(diào)用”而言這額外開銷還是明顯的)。注意到參數(shù)數(shù)組被拆開了,把每個(gè)參數(shù)都恢復(fù)到原本沒(méi)有被Object[]包裝前的樣子,然后對(duì)目標(biāo)方法做正常的invokevirtual調(diào)用。由于在生成代碼時(shí)已經(jīng)循環(huán)遍歷過(guò)參數(shù)類型的數(shù)組,生成出來(lái)的代碼里就不再包含循環(huán)了。

至此找到我的答案了,因?yàn)镸ethodAccessor會(huì)做強(qiáng)制類型轉(zhuǎn)換再進(jìn)行方法調(diào)用,但父類強(qiáng)制轉(zhuǎn)化成子類的的時(shí)候就會(huì)報(bào)錯(cuò)類型不匹配錯(cuò)誤了,所以如果變量的引用聲明是父但實(shí)際指向的對(duì)象是子,那么這種調(diào)用也是可以的。

題外話

當(dāng)該反射調(diào)用成為熱點(diǎn)時(shí),它甚至可以被內(nèi)聯(lián)到靠近Method.invoke()的一側(cè),大大降低了反射調(diào)用的開銷。而native版的反射調(diào)用則無(wú)法被有效內(nèi)聯(lián),因而調(diào)用開銷無(wú)法隨程序的運(yùn)行而降低。

雖說(shuō)Sun的JDK這種實(shí)現(xiàn)方式使得反射調(diào)用方法成本比以前降低了很多,但Method.invoke()本身要用數(shù)組包裝參數(shù);而且每次調(diào)用都必須檢查方法的可見(jiàn)性(在Method.invoke()里),也必須檢查每個(gè)實(shí)際參數(shù)與形式參數(shù)的類型匹配性(在NativeMethodAccessorImpl.invoke0()里或者生成的Java版MethodAccessor.invoke()里);而且Method.invoke()就像是個(gè)獨(dú)木橋一樣,各處的反射調(diào)用都要擠過(guò)去,在調(diào)用點(diǎn)上收集到的類型信息就會(huì)很亂,影響內(nèi)聯(lián)程序的判斷,使得Method.invoke()自身難以被內(nèi)聯(lián)到調(diào)用方。

相比之下JDK7里新的MethodHandler則更有潛力,在其功能完全實(shí)現(xiàn)后能達(dá)到比普通反射調(diào)用方法更高的性能。在使用MethodHandle來(lái)做反射調(diào)用時(shí),MethodHandle.invoke()的形式參數(shù)與返回值類型都是準(zhǔn)確的,所以只需要在鏈接方法的時(shí)候才需要檢查類型的匹配性,而不必在每次調(diào)用時(shí)都檢查。而且MethodHandle是不可變值,在創(chuàng)建后其內(nèi)部狀態(tài)就不會(huì)再改變了;JVM可以利用這個(gè)知識(shí)而放心的對(duì)它做激進(jìn)優(yōu)化,例如將實(shí)際的調(diào)用目標(biāo)內(nèi)聯(lián)到做反射調(diào)用的一側(cè)。

本來(lái)Java的安全機(jī)制使得不同類之間不是任意信息都可見(jiàn),但Sun的JDK里開了個(gè)口,有一個(gè)標(biāo)記類專門用于開后門:

package sun.reflect;/** <P> MagicAccessorImpl (named for parity with FieldAccessorImpl and others, not because it actually implements an interface) is a marker class in the hierarchy. All subclasses of this class are 'magically' granted access by the VM to otherwise inaccessible fields and methods of other classes. It is used to hold the code for dynamically-generated FieldAccessorImpl and MethodAccessorImpl subclasses. (Use of the word 'unsafe' was avoided in this class’s name to avoid confusion with {@link sun.misc.Unsafe}.) </P> <P> The bug fix for 4486457 also necessitated disabling verification for this class and all subclasses, as opposed to just SerializationConstructorAccessorImpl and subclasses, to avoid having to indicate to the VM which of these dynamically-generated stub classes were known to be able to pass the verifier. </P> <P> Do not change the name of this class without also changing the VM’s code. </P> */class MagicAccessorImpl {}

那個(gè)'__JVM_DefineClass__'的來(lái)源是這里:

src/share/vm/prims/jvm.cpp

// common code for JVM_DefineClass() and JVM_DefineClassWithSource()// and JVM_DefineClassWithSourceCond()static jclass jvm_define_class_common(JNIEnv *env, const char *name, jobject loader, const jbyte *buf, jsize len, jobject pd, const char *source, jboolean verify, TRAPS) { if (source == NULL) source = '__JVM_DefineClass__';

P.S. log里的'shared objects file',其實(shí)就是rt.jar,為什么要這么顯示,Stack OverFlow上有這樣的回答:

This is Class Data Sharing. When running the Sun/Oracle Client HotSpot and sharing enable (either -Xshare:auto which is the default, or -Xshare:on), the classes.jsa file is memory mapped. This file contains a number of classes (listed in the classlist file) in internal representation suitable for the exact configuration of the machine running it. The idea is that the classes can be loaded quickly, getting the the JVM up faster. Soon enough a class not covered will be hit, and rt.jar will need to be opened and classes loaded conventionally as required.

不能很好理解,大概理解就是所有jvm共享,并可以快速加載里面的class.有英文好的朋友可以留言幫助下。

P.S java內(nèi)聯(lián)函數(shù)

C++是否為內(nèi)聯(lián)函數(shù)由自己決定,Java由編譯器決定。內(nèi)聯(lián)函數(shù)就是指函數(shù)在被調(diào)用的地方直接展開,編譯器在調(diào)用時(shí)不用像一般函數(shù)那樣,參數(shù)壓棧,返回時(shí)參數(shù)出棧以及資源釋放等,這樣提高了程序執(zhí)行速度。

Java不支持直接聲明為內(nèi)聯(lián)函數(shù)的,如果想讓他內(nèi)聯(lián),則是由編譯器說(shuō)了算,你只能夠向編譯器提出請(qǐng)求。

final除了不能被override外,還可能實(shí)現(xiàn)內(nèi)聯(lián)。如果函數(shù)為private,則也可能是內(nèi)聯(lián)的。

總的來(lái)說(shuō),一般的函數(shù)都不會(huì)被當(dāng)做內(nèi)聯(lián)函數(shù),只有聲明了final后,編譯器才會(huì)考慮是不是要把你的函數(shù)變成內(nèi)聯(lián)函數(shù)。

內(nèi)聯(lián)不一定好,當(dāng)被指定為內(nèi)聯(lián)的方法體很大時(shí),展開的開銷可能就已經(jīng)超過(guò)了普通函數(shù)調(diào)用調(diào)用的時(shí)間,引入了內(nèi)聯(lián)反而降低了性能,因?yàn)樵谶x擇這個(gè)關(guān)鍵字應(yīng)該慎重些,不過(guò),在以后高版本的JVM中,在處理內(nèi)聯(lián)時(shí)做出了優(yōu)化,它會(huì)根據(jù)方法的規(guī)模來(lái)確定是否展開調(diào)用。

總結(jié)

到此這篇關(guān)于JAVA深入探究之Method的Invoke方法的文章就介紹到這了,更多相關(guān)JAVA Method的Invoke方法內(nèi)容請(qǐng)搜索好吧啦網(wǎng)以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持好吧啦網(wǎng)!

標(biāo)簽: Java
相關(guān)文章:
主站蜘蛛池模板: 成人精品在线观看 | 在线区| 国产一区二区视频在线观看 | 亚洲国产精品久久久男人的天堂 | 国产一区二区三区四区在线观看 | 国产精品一区在线看 | av一区二区在线观看 | 精品二区 | 97av在线 | 激情自拍偷拍 | 欧美日韩亚洲视频 | 欧美日韩亚洲国产 | 国外成人在线视频网站 | 亚洲一区二区三区高清 | av网站免费看 | 亚洲综合色视频在线观看 | 午夜影院在线 | 91免费观看在线 | 国产日韩欧美 | 狠狠色综合久久丁香婷婷 | 国产精品久久久久久久一区探花 | 91精品国产欧美一区二区成人 | 日韩1区3区4区第一页 | 中文字幕观看 | 亚洲www视频 | 日韩欧美网 | 欧美成人免费在线视频 | 久久国产精品电影 | 欧美成人精品一区二区男人看 | 精品久久久久久 | 一本色道久久综合狠狠躁篇的优点 | 国产精品久久久久久亚洲调教 | 国产成人精品久久二区二区 | 激情欧美一区二区三区中文字幕 | √8天堂资源地址中文在线 成人欧美一区二区三区白人 | 国产美女精品人人做人人爽 | 国产欧美精品一区二区三区四区 | 亚洲免费视频大全 | 欧美激情国产日韩精品一区18 | 91av视频在线| 欧美激情一区二区三级高清视频 | 欧美区在线 | 国内自拍第一页 | www久 | 日韩一区二区三区在线 | 一区二区免费看 | 欧美精品一 | 欧美日韩三级 | 国产日韩一区二区三区 | 蜜桃臀一区二区三区 | 日本一区二区不卡视频 | 久久精品91 | 亚洲国产精品99久久久久久久久 | 在线观看国产wwwa级羞羞视频 | 天天看夜夜爽 | 国产精品视频一区二区三区四 | 一区二区三区在线 | 91极品在线 | 骚视频在线观看 | 国产精品1区2区3区 欧美 中文字幕 | 精品久久久久久久久久久久久久久久久久 | 自拍偷拍专区 | 亚洲一区二区在线 | 欧美国产在线观看 | 欧洲一区在线 | 高清久久| 欧美黄色一区 | 日本狠狠色 | 久草精品视频在线播放 | 国产色99精品9i | 午夜亚洲福利 | 国产网站在线播放 | www.国产精品 | 精品精品久久 | 欧美激情欧美激情在线五月 | 欧美一区二区免费在线 | www.久久精品 | 国产精品一码二码三码在线 | 亚洲一区二区三区四区 | 能在线观看的黄色网址 | 情趣视频在线免费观看 | 国产乱码久久久久久一区二区 | 成人免费一区二区三区视频网站 | 久热官网| 国产精品久久久久久久久免费 | 精品国产91久久久久久久 | 国产精品一区二区三区四区 | 日韩精品在线免费观看视频 | 曰批免费视频播放免费 | 亚洲中字幕 | 欧美一区二区三区 | 国产精品视频入口 | 欧美成人免费视频 | 日韩av免费在线观看 | 一区二区三区的视频 | 日本99精品| 国产色在线 | 一区二区三区在线免费观看 | 日韩国产一区 | 成人高清av| 国产免费av一区二区三区 | 91在线免费看 | 在线国产一区二区 | 99久久婷婷国产精品综合 | 国产日韩欧美一区二区 | 精品一区二区三区国产 | 一级片在线观看 | 成人亚洲免费 | 亚洲欧美日韩另类一区二区 | 天天插天天操天天干 | 国产大片久久久 | www.91av在线| 亚洲第一网站 | 亚洲高清视频在线观看 | 青青草人人 | 国产免费一区 | 色在线播放 | 亚洲欧美激情精品一区二区 | 亚洲视频在线看 | 亚洲欧美激情在线 | 国产精品无码久久久久 | 国产精品1区2区 | 色中色综合 | 在线91 | 国产美女在线观看 | 久久人人爽人人爽人人片av不 | 99re6热只有精品免费观看 | 亚洲成人看片 | 国产精品久久久久久久久久99 | 国产人久久人人人人爽 | 国产成人精品在线观看 | 国产精品成人在线观看 | 免费成人av| 欧美日韩中文在线观看 | 国产一区二区三区四区 | 色花av| 在线日韩中文字幕 | 国内精品国产成人国产三级粉色 | 天天干天天曰天天操 | 日韩中文字幕在线免费 | 91精品国产欧美一区二区成人 | 亚洲免费资源 | 精品久久久久久久久久久久久久 | 久久在线 | 欧美精品久久久 | 久久久久一区二区 | 日本一本在线 | 久久爱电影 | 久久91av | 国产精品久久久久久福利一牛影视 | 亚洲这里只有精品 | 欧洲一级毛片 | a欧美| ririsao久久精品一区 | www视频在线观看 | 一区二区精品视频在线观看 | 色小妹三区 | 欧美午夜精品久久久久久人妖 | 国产成人精品一区二区三区视频 | 在线视频 亚洲 | 精品无码久久久久久国产 | 中文字幕亚洲精品 | 亚洲激情一区 | 91精品久久 | 红桃成人少妇网站 | 曰韩在线| 午夜视频在线观看网站 | 精品视频在线观看 | 日韩电影专区 | 在线亚洲一区 | 在线观看国产日韩欧美 | 欧美国产日韩一区 | 日韩欧美高清视频 | 久久精品视频网站 | 99福利视频 | 日韩欧美在线中文字幕 | 欧美一级二级片 | 日本不卡免费新一二三区 | 亚洲伊人久久综合 | 免费黄色网止 | 久久国产精品99久久久久久牛牛 | 精品国产仑片一区二区三区 | 欧美成人h版在线观看 | 国产精品久久久久免费 | 精品久久久久久久久久 | 欧美一区二区 | 亚洲三级免费观看 | 国产精品久久久久久久久久 | 国产精久久一区二区三区 | 国产成人av在线 | 午夜免费片 | 久久99精品久久久久久噜噜 | 嫩草视频网 | 色网站在线观看 | 日韩福利视频 | 自拍偷拍99 | 成人国产在线 | www,99热 | 免费成人在线网站 | 在线观看国产wwwa级羞羞视频 | 男女激情网址 | 国产精品电影 | 麻豆一区二区三区 | 久久久久久久成人 | 国产精品视频播放 | 国产激情视频 | 青青草免费在线 | 国产一区二区黑人欧美xxxx | 范冰冰一级做a爰片久久毛片 | 一区二区三区精品 | 黄色在线观看 | 国产乱码精品一区二区三区av | 中文字幕亚洲欧美日韩在线不卡 | 久久久亚洲精品中文字幕 | 欧美精品欧美精品系列 | 97国产在线视频 | 欧美精品三区 | 一区二区三区在线看 | 欧美11一13sex性hd | 国产精品久久久久久久午夜 | 国产精品无码专区在线观看 | 高清视频一区二区三区 | 免费久久网站 | 91成人精品| 色婷婷久久久swag精品 | 性色av一区二区三区 | 一区二区免费视频 | 91在线播 | 午夜免费| 久久久久国产精品免费免费搜索 | 黄色国产 | 精品视频在线播放 | 国产精品久久久久久久岛一牛影视 | 成人av网页 | 91精品国产综合久久久久久丝袜 | segui88久久综合9999 | 在线视频这里只有精品 | 亚洲三区视频 | 国产精品1区2区3区 欧美 中文字幕 | 天天天综合网 | 在线观看的av | 日韩1区 | 久久精品极品 | 亚洲毛片在线 | 国产精品综合 | 欧美在线观看视频 | 久久久精品久久久久久 | 欧美精品一区二区三区四区 | 日韩视频在线观看一区 | 久久99久久99精品免视看婷婷 | 欧美激情视频久久 | 日韩激情欧美 | 久久伊人成人网 | 91se在线 | 成人在线视频一区 | 欧美亚洲综合久久 | 中文字幕日韩专区 | 蜜桃精品久久久久久久免费影院 | а天堂中文最新一区二区三区 | 精品无码久久久久久国产 | 亚洲精品亚洲人成人网 | 日韩在线观看视频免费 | 亚洲好看站 | 亚洲精品一二区 | 美女在线国产 | 999国产在线 | av免费网站在线观看 | 国产片一区二区三区 | 国产视频99| 中文字幕免费在线观看 | 日韩成人黄色 | 欧美高清一区 | 国产精品国产精品国产 | 奇米色777欧美一区二区 | 美女主播精品视频一二三四 | 国产在线1| 国产伦精品一区二区三区四区视频_ | 久久精品国产免费 | 一区二区欧美视频 | 欧美a视频 | 一区二区不卡 | 一区二区三区免费 | 久草视频首页 | 国产精品欧美久久久久一区二区 | 在线成人av观看 | 久久久久一区二区三区 | 亚洲综合在线播放 | 男女视频在线观看 | 免费国产一区二区 | 中文字幕精品视频在线观看 | 亚洲免费网站 | 精品国产不卡一区二区三区 | 国产成人精品免费视频大全 | 欧美日韩一区二区视频在线观看 | 精品一区久久 | 欧美日韩二区三区 | 欧美一区二区三区电影 | 99视频在线 | 中文字幕在线亚洲 | a视频在线 | 欧美日韩中文在线 | 免费观看一级毛片 | 精品欧美乱码久久久久久 | 香蕉视频成人在线观看 | 国产a级大片 | 99re6在线视频精品免费 | 亚洲一区中文 | 日韩成人免费中文字幕 | 99久久精品免费 | 日韩成人 | 欧美精品一区二区久久 | 91精品国产综合久久国产大片 | 久久久久久成人 | 日本不卡高字幕在线2019 | 一区二区在线看 | 91综合在线观看 | av网址在线播放 | 国产精品一区二区三区在线播放 | 国产精品国产精品 | 黄色免费视频 | 亚洲精品一区二区三区中文字幕 | a∨色狠狠一区二区三区 | 91色乱码一区二区三区 | 国产精品人人做人人爽人人添 | 色婷婷欧美 | 精品久久99 | 精品99在线 | 国产一区二区视频免费看 | 欧美jizzhd精品欧美巨大免费 | 中文字幕一区在线观看视频 | 九色在线 | 亚洲永久免费视频 | 午夜精品在线观看 | 华丽的挑战在线观看 | 韩国一区二区视频 | 亚洲成人1区 | 国产精品久久久久久久午夜片 | 亚洲成人精品在线观看 | 欧美亚洲91 | 男人天堂999 | 欧美精品一区二区三区蜜桃视频 | 嫩草视频入口 | 精品在线一区二区 | 亚洲国产中文字幕 | 国产福利一区二区三区视频 | 成人国产精品久久 | 在线观看日韩av | 狠狠人人 | 久久成人一区 | 亚洲综合色自拍一区 | 亚洲精品一区久久久久久 | 亚洲成人精品久久 | 国产做a爰片久久毛片a我的朋友 | 精品国产依人香蕉在线精品 | 可以免费看黄视频的网站 | 亚洲欧美中文日韩在线v日本 | 亚洲福利在线播放 | 亚洲视频在线观看 | 午夜精品久久久久久久久久久久久 | 毛片网在线观看 | 中文字幕在线电影 | 99精品99 | 亚洲免费网 | 欧美综合久久 | 欧美成人h版在线观看 | 亚洲一区精品视频 | www.fefe66.com | youjizz国产 | 亚洲精品中文视频 | 日本成人午夜影院 | 国产伦精品一区二区三区四区视频 | 五月激情综合网 | 久久国产精品系列 | 国产激情在线视频 | 在线观看免费视频日韩 | 欧美一级片在线观看 | 天天操天天碰 | 久久精品久久久久久久久久久久久 | 欧美久久精品 | 国产高清一二三区 | 亚洲麻豆精品 | 欧美日韩中文字幕 | 色999国产 | 国产在线网| 一级毛片在线看aaaa | 亚洲视频在线播放 | 久草天堂 | 精品国产一区二区三区日日嗨 | 欧美成年网站 | 久久精品国产91精品亚洲高清 | 久久成人精品视频 | 国产精品二区一区二区aⅴ污介绍 | 久久综合九色综合欧美狠狠 | 免费观看电视在线高清视频 | 色av综合 | 97色在线视频 | 久久综合一区二区三区 | 一区二区精品视频 | 成人性大片免费观看网站 | 成人午夜激情 | 最新久久精品 | 91精品国产综合久久久久久软件 | 久久国产精品无码网站 | 久久久久久久国产 | 色婷婷网 | 亚洲久草| 免费日韩成人 | 狠狠干天天干 | 亚洲视频精品一区 | 亚洲精品成人在线 | 成人午夜视频在线观看 | 亚洲国产aⅴ成人精品无吗 国产精品永久在线观看 | 一区二区三区四区精品 | 久久91精品 | 一区二区三区成人久久爱 | 免费成人av | 91久草视频| 999国产| 中文字幕av一区二区三区 | 色玖玖综合| 国产精品高潮呻吟久久av黑人 | 欧美一区二区三区在线观看视频 | 国产乱码一区二区三区 | 国产精品亚洲一区二区三区 | 午夜专区 | 国产精品久久影院 | 中文字幕自拍偷拍 | 中文字幕日韩在线 | 成人爽a毛片一区二区免费 美女高潮久久久 | av午夜电影 | 99久久久久| 日本亚洲精品成人欧美一区 | av网站免费在线 | 日本三级中国三级99人妇网站 | 欧美成人精品在线视频 | 久久国产精品久久久久久 | 国产成人午夜精品影院游乐网 | 午夜激情在线播放 | 欧美精品久久久久久久久久丰满 | 四虎免费在线播放 | 国产九九九 | 国产黄色网址在线观看 | 日韩激情综合网 | 日韩手机在线 | 日韩精品观看 | 国产精品视频免费播放 | 精品一区二区三区久久久 | 国产在线一区二区 | 欧美国产一区二区 | 久久com | 国产精品一区二区在线 | 激情久久av一区av二区av三区 | 国产高清精品一区 | 亚洲视频在线看 | 青青草网站 | 国产大学生情侣呻吟视频 | 成年人av网站 | 青青草人人 | 免费一区二区三区 | 欧美久久a | 国产亚洲精品久久久久久久 | 在线中文av| 激情五月婷婷综合 | 日韩视频在线免费观看 | 亚洲精品久久久一区二区三区 | 一区免费视频 | 国产免费一区二区 | 日本电影www | 91视频国内| 久久精品久久久 | a免费在线 | 久久成人免费视频 | 日韩免费一区 | 日韩精品一区二区三区在线观看 | 天天操网 | 免费成人在线网站 | 99国产精品久久久久久久 | 国产精品毛片久久久久久久 | 亚洲欧美激情在线 | 日韩视频中文 | 亚洲午夜精品视频 | 亚洲精品在线免费观看视频 | 日韩av免费在线播放 | 国产最新视频在线 | 日韩中文在线 | 日韩在线国产精品 | 四虎成人在线视频 | 亚洲国产精品成人 | 国产高清网站 | 国产精品成av人在线视午夜片 | 国产视频久久 | 欧美一区二区三区 | 日韩在线观看成人 | 自拍偷拍精品 | 黄色最新网站 | 亚洲 欧美 精品 | 婷婷午夜激情网 | 精品国产99 | 欧一区二区 | 欧美精品成人一区二区三区四区 | 日韩成人免费 | 91高清视频在线观看 | 日韩在线观看 | 欧美一级二级三级视频 | 亚洲视频免费在线 | 国产综合视频在线观看 | 日本在线视频一区二区 | 粉嫩av网站 | а_天堂中文最新版地址 | 亚洲第一成年免费网站 | 精品一区二区三区四区视频 | 日韩精品一区二区三区中文在线 | 人人干操 | 色综合久久久久久久久久久 | 国产人久久人人人人爽 | 国内精品视频一区二区三区 | 色综合天天天天做夜夜夜夜做 | 久久精品国产视频 | 精品影院 | 亚洲精彩视频 | 9久久精品| 国产欧美一区二区精品性色 | 国产婷婷 | 国产va| 精品亚洲一区二区三区四区五区 | 久久国产精彩视频 | 欧美综合一区二区三区 | 欧美精品网站 | 欧美成人免费在线视频 | 91午夜激情 | 亚洲aⅴ天堂av在线电影软件 | 在线精品国产一区二区三区 | 91精品国产欧美一区二区成人 | 亚洲精品一区中文字幕乱码 | 亚洲第一色片 | 日本免费一区二区在线观看 | 91中文字幕 | 日韩成人视屏 | 91在线免费观看 | av黄色在线免费观看 | 久色视频在线 | 五月免费视频 | 天天曰天天曰 | 亚洲午夜一区 | 中文字幕在线亚洲 | 久久久精品区 | 日韩视频在线观看一区 | 精品日本久久 | 韩国久久精品 | 日本久久精品 | 午夜精品影院 | 精品中文字幕在线 | 国产一二三视频 | 蜜桃视频精品 | 中文字幕7777| 日韩国产欧美一区 | 国产丝袜在线 | 在线一区二区三区做爰视频网站 | 黄色a在线观看 | 国产一区在线看 | 天天操夜夜干 | 国产黄色在线观看 | 国产 日韩 欧美 中文 在线播放 | 五月激情综合婷婷 | 国产另类一区 | 福利精品视频 | 手机看片在线 | 日韩成人在线播放 | 中文字幕乱码一区二区三区 | 国产综合亚洲精品一区二 | 欧美高清成人 | 色视频www在线播放国产人成 | 国产精品一区久久久久 | 精品婷婷 | 欧美日韩国产精品一区二区亚洲 | 麻豆亚洲 | 国产乱a视频在线 | 欧美成人免费 | 国产精品久久久久久久久久 | 草草久久久| 黄网在线免费观看 | 品久久久久久久久久96高清 | 欧美日韩精品综合 | 伊人午夜| 国产一区免费 | 中文字幕日韩欧美一区二区三区 | 自拍偷拍专区 | 欧洲一级毛片 | 亚洲视频三区 | 91麻豆精品国产91久久久久久久久 | 欧美黑人一级爽快片淫片高清 | 国产成人综合在线 | 97久久精品午夜一区二区 | 亚洲精品在线看 | 99国产视频 | 日韩精品免费看 | 一级做a爰片毛片 | 日本中文字幕视频 | 四虎影音 | 精品欧美视频 | 欧美日韩一区二区电影 | sis色中色| 久久人人爽人人爽人人片av不 | 欧美日韩亚洲一区 | 狠狠综合久久av一区二区老牛 | 日韩精品一区二区三区四区 | 亚洲在线 | 91在线观看 | 噜噜噜在线观看免费视频日本 | 精品一区二区三区免费毛片爱 | 午夜影视在线观看 | 一区二区亚洲视频 | 国产www视频 | 国产精品免费一区二区三区四区 | 欧美性一区二区三区 |