成人国产在线小视频_日韩寡妇人妻调教在线播放_色成人www永久在线观看_2018国产精品久久_亚洲欧美高清在线30p_亚洲少妇综合一区_黄色在线播放国产_亚洲另类技巧小说校园_国产主播xx日韩_a级毛片在线免费

資訊專欄INFORMATION COLUMN

JSR 292 學(xué)習(xí)

mochixuan / 475人閱讀

摘要:學(xué)習(xí)添加一個字節(jié)碼。使用提供了字節(jié)碼的支持。這個啟動方法的返回值是類的對象。與區(qū)別在模擬字節(jié)碼層次的方法調(diào)用,因而可適用于所有語言在模擬層次的方法調(diào)用,僅可適用于。方法調(diào)用需要考慮到字節(jié)碼,而則不用考慮這些。

JSR 292 學(xué)習(xí)
Add a new bytecode, invokedynamic ,
that supports efficient and flexible execution of method invocations in the absence of static type information.
添加一個字節(jié)碼 invokedynamic 。用于提供在缺乏靜態(tài)類型信息時高效和靈活的執(zhí)行方法調(diào)用。
invokedynamic 是針對 JVM 的,用于更好的支持動態(tài) JVM 語言的函數(shù)調(diào)用; JSR292 實現(xiàn)則提供在 Java 語言層面上的 invokedynamic 調(diào)用。

JSR292 實現(xiàn)于 JDK7,以下為實現(xiàn)方案的類。

└─java
    └─lang
        ├─invoke
        |   ├─CallSite
        |   ├─ConstantCallSite
        |   ├─MethodHandle
        |   ├─MethodHandleProxies
        |   ├─MethodHandles
        |   ├─MethodHandles.Lookup
        |   ├─MethodType
        |   ├─MutableCallSite
        |   ├─SwitchPoint
        |   └─VolatileCallSite
        ├─ClassValue
        └─BootstrapMethodError
API 使用
JSR292 API 提供了 invokedynamic 字節(jié)碼的 API 支持。
使用基本思路

創(chuàng)建 MethodType 對象,指定方法的簽名

MethodHandles.Lookup 中查找類型為 MethodTypeMethodHandle

傳入方法參數(shù)并調(diào)用 MethodHandle.invoke 或者 MethodHandle.invokeExact 方法

MethodType

用于指定方法的類型,此處的方法類型與 Java 的方法判斷有點(diǎn)區(qū)別,Java 的方法簽名不包括 返回值 ,但是 MethodType 卻是通過 返回值 和 參數(shù)類型 來確定。并且該類型不包含 方法名 。

// 快速創(chuàng)建一個 MethodType 。
MethodType methodType = MethodType.methodType(String.class, Object.class);

// 也可以通過 MethodHandle 實例獲取它要調(diào)用的方法實例
// MethodHandle handle = ...
MethodType type = handle.type();

除了這兩種獲取 MethodType 的方法(實際上第二種不算,MethodHandle 對象的創(chuàng)建依賴于 MethodType)。下面還有兩種,它也是 MethodType 的靜態(tài)方法:

genericMethodType: 所有類型都是 Object 類型

fromMethodDescriptorString: 有兩個參數(shù),一個是方法描述符,該描述符是基于 字節(jié)碼 層面的描述符。之后就是一個類加載器。

當(dāng)然,創(chuàng)建完之后還可以修改返回值類型和參數(shù)類型如: changeParameterType 、 changeReturnType 等。

MethodHandles.Lookup

一個工廠類,用于創(chuàng)建 MethodHandle 類對象。它可通過普通的 findXxx() 方法得到相應(yīng)的 MethodHandle 實例,也可以通過 反射類 來創(chuàng)建實例。如 lookup.unreflect(Method) 等方法。

MethodHandle

MethodHandle 可看成是方法引用,它使得 Java 擁有了類似函數(shù)指針或委托的方法別名工具。

注意幾點(diǎn):

引用的方法必須和 MethodHandle 的 type 保持一致。

這里提到的 type 包括 返回值參數(shù)列表 。

MethodHandle 也是可執(zhí)行及可以進(jìn)行轉(zhuǎn)換。

幾個 MethodHandle 方法與字節(jié)碼的對應(yīng):

MethodHandle方法 字節(jié)碼 描述
findStatic invokestatic 調(diào)用靜態(tài)方法
findSpecial invokespecial 調(diào)用實例構(gòu)造方法,私有方法,父類方法。
findVirtual invokevirtual 調(diào)用所有的虛方法
findVirtual invokeinterface 調(diào)用接口方法,會在運(yùn)行時再確定一個實現(xiàn)此接口的對象。

示例

public void example() throws Throwable {
    // MethodHandles.Lookup 類似于 MethodHandle 工廠類,用于創(chuàng)建 MethodHandle
    MethodHandles.Lookup lookup = MethodHandles.lookup();
    MethodHandle mh = lookup.findStatic(String.class, "valueOf", MethodType.methodType(String.class, int.class));

    String result = (String) mh.invoke(1);
    System.out.println("1".equals(result));
}

三種調(diào)用方式:

invokeExact: 調(diào)用此方法與直接調(diào)用底層方法一樣,需要做到參數(shù)類型精確匹配

invoke: 參數(shù)類型松散匹配,通過asType自動適配

invokeWithArguments: 直接通過方法參數(shù)來調(diào)用

CallSite

當(dāng) JVM 執(zhí)行 invokedynamic 指令時,首先需要鏈接其對應(yīng)的 動態(tài)調(diào)用點(diǎn) 。在鏈接的時候,JVM會先調(diào)用一個啟動方法(bootstrap method)。這個啟動方法的返回值是 java.lang.invoke.CallSite 類的對象。

在通過啟動方法得到了 CallSite 之后,通過這個 CallSite 對象的 getTarget() 可以獲取到實際要調(diào)用的目標(biāo)方法句柄。
有了方法句柄之后,對這個 動態(tài)調(diào)用點(diǎn) 的調(diào)用,實際上是代理給方法句柄來完成的。
也就是說,對 invokedynamic 指令的調(diào)用實際上就等價于對方法句柄的調(diào)用,具體來說是被轉(zhuǎn)換成對方法句柄的invoke方法的調(diào)用。

JDK7 中提供了三種類型的動態(tài)調(diào)用點(diǎn)CallSite的實現(xiàn):java.lang.invoke.ConstantCallSite、java.lang.invoke.MutableCallSitejava.lang.invoke.VolatileCallSite。

ConstantCallSite

表示的調(diào)用點(diǎn)綁定的是一個固定的方法句柄,一旦鏈接之后,就無法修改。示例如下:

public void constantCallSite() throws Throwable {
    MethodType type = MethodType.methodType(String.class, int.class, int.class);

    MethodHandles.Lookup lookup = MethodHandles.lookup();
    MethodHandle handle = lookup.findVirtual(String.class, "substring", type);

    ConstantCallSite callSite = new ConstantCallSite(handle);
    MethodHandle invoker = callSite.dynamicInvoker();
    String result = (String) invoker.invoke("Hello", 2, 3);
    System.out.println(result);
}
MutableCallSite

表示的調(diào)用點(diǎn)則允許在運(yùn)行時動態(tài)修改其目標(biāo)方法句柄,即可以重新鏈接到新的方法句柄上。示例如下:

/**
 * MutableCallSite 允許對其所關(guān)聯(lián)的目標(biāo)方法句柄通過setTarget方法來進(jìn)行修改。
 * 以下為 創(chuàng)建一個 MutableCallSite,指定了方法句柄的類型,則設(shè)置的其他方法也必須是這種類型。
 */
public void useMutableCallSite() throws Throwable {
    MethodType type = MethodType.methodType(int.class, int.class, int.class);
    MutableCallSite callSite = new MutableCallSite(type);
    MethodHandle invoker = callSite.dynamicInvoker();

    MethodHandles.Lookup lookup = MethodHandles.lookup();
    MethodHandle maxHandle = lookup.findStatic(Math.class, "max", type);
    callSite.setTarget(maxHandle);
    int result = (int) invoker.invoke(3, 5);
    System.out.println(result == 5);

    MethodHandle minHandle = lookup.findStatic(Math.class, "min", type);
    callSite.setTarget(minHandle);
    result = (int) invoker.invoke(3, 5);
    System.out.println(result == 3);
}

MutableCallSite.syncAll() 提供了方法來強(qiáng)制要求各個線程中 MutableCallSite 的使用者立即獲取最新的目標(biāo)方法句柄。
但這個時候也可以選擇使用 VolatileCallSite。

VolatileCallSite

作用與 MutableCallSite 類似,不同的是它適用于多線程情況,用來保證對于目標(biāo)方法句柄所做的修改能夠被其他線程看到。
此處便不再提供示例,可參考 MutableCallSite。

MethodHandle 與 Method 區(qū)別

MethodHandle 在模擬 字節(jié)碼 層次的方法調(diào)用,因而可適用于所有 JVM 語言 ;Reflection 在模擬 Java 層次的方法調(diào)用,僅可適用于 Java。

MethodHandle 可進(jìn)行 JVM 的內(nèi)聯(lián)優(yōu)化,Reflection 屏蔽了 JVM ,所以完全沒有相應(yīng)的優(yōu)化。

MethodHandle 從 JVM 層次支持調(diào)用,只需要包含方法必要的信息,所以說是輕量級的,而 Reflection 是 Java Api 層次的反射調(diào)用,包含了方法的簽名、描述符以及方法屬性表中各種屬性的Java端表示方式,還包含有執(zhí)行權(quán)限等的運(yùn)行期信息,所以說是重量級的。

MethodHandle 方法調(diào)用需要考慮到 字節(jié)碼,而 Reflection 則不用考慮這些。

參考

JDK1.8下關(guān)于MethodHandle問題

Invokedynamic 和 MethodHandle的緣由

MethodHandle與反射Method區(qū)別

文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。

轉(zhuǎn)載請注明本文地址:http://systransis.cn/yun/72496.html

相關(guān)文章

  • 2014年Java值得期待的五大理由

    摘要:物聯(lián)網(wǎng)已經(jīng)成為現(xiàn)實只要去年單獨(dú)參加過會議或者搜索過專門的技術(shù)網(wǎng)站,你現(xiàn)在會越來越清楚地注意到物物互聯(lián)正在蓬勃發(fā)展。有大量的討論在口頭上承認(rèn)了物聯(lián)網(wǎng),年這方面的激烈討論只會更多。兩年過去了,期待已久的更新應(yīng)該在月日發(fā)布。 導(dǎo)讀 如果你還在為Oracle收購Sun公司給Java社區(qū)的變化所糾結(jié),請站在Oracle的角度替它想想吧。2013年大部分時間里,Oracle都在與遺留的Java安...

    tinyq 評論0 收藏0
  • Java8 新特性:Lambda表達(dá)式和虛擬擴(kuò)展方法標(biāo)注

    摘要:摘要添加了表達(dá)式閉包和特性支持,包括方法的引用,增強(qiáng)類型推斷,和虛擬擴(kuò)展方法。圍繞的語言功能支持包括虛擬擴(kuò)展方法,這將使接口的源代碼和二進(jìn)制兼容的方式演變升級。 Author:Joseph D. Darcy Organization:Oracle Owner:Brian Goetz Created:2011/11/1 Updated:2013/2/21 Type:Feature Sta...

    UsherChen 評論0 收藏0
  • Java7的新特性

    摘要:語言特性系列的新特性的新特性的新特性的新特性的新特性的新特性的新特性的新特性的新特性序本文主要講的新特性,相對于而言,增加了一些重要的特性,比如,不像那么雞肋,也算是一個重要的版本。其他支持的下劃線支持參考 Java語言特性系列 Java5的新特性 Java6的新特性 Java7的新特性 Java8的新特性 Java9的新特性 Java10的新特性 Java11的新特性 Java12...

    April 評論0 收藏0
  • JDK 9 變更一覽

    摘要:概述是一個主要版本的發(fā)布這里介紹的是對的特性和增強(qiáng)的實現(xiàn)是的增強(qiáng)提案,包括增強(qiáng)建議和路線圖流程規(guī)范請求,描述了針對平臺的建議和最終規(guī)范主要變更統(tǒng)一模塊化標(biāo)準(zhǔn)這是全新的編程組件模塊,是可命名的可自描述的代碼和數(shù)據(jù)集合。 概述 java9是一個主要版本的發(fā)布 這里介紹的是Oracle對JDK9的特性和增強(qiáng)的實現(xiàn) JEP是JDK的增強(qiáng)提案,包括增強(qiáng)建議和路線圖流程 JSR(Java規(guī)范請...

    sherlock221 評論0 收藏0

發(fā)表評論

0條評論

mochixuan

|高級講師

TA的文章

閱讀更多
最新活動
閱讀需要支付1元查看
<