日韩成人免费在线_国产成人一二_精品国产免费人成电影在线观..._日本一区二区三区久久久久久久久不

當前位置:首頁 > 科技  > 軟件

通過方法引用獲取屬性名的底層邏輯是什么?

來源: 責編: 時間:2024-04-11 09:04:24 228觀看
導讀很多小伙伴可能都用過 MyBatis-Plus,這里邊我們構造 where 條件的時候,可以直接通過方法引用的方式去指定屬性名:LambdaQueryWrapper<Book> qw = new LambdaQueryWrapper<>();qw.eq(Book::getId, 2);List<Book> list = b

很多小伙伴可能都用過 MyBatis-Plus,這里邊我們構造 where 條件的時候,可以直接通過方法引用的方式去指定屬性名:8pZ28資訊網——每日最新資訊28at.com

LambdaQueryWrapper<Book> qw = new LambdaQueryWrapper<>();qw.eq(Book::getId, 2);List<Book> list = bookMapper.selectList(qw);System.out.println("list = " + list);

Book::getId 這就是方法引用,松哥之前也專門寫過文章介紹相關內容,這里就不再多說。這里我們就單純來說說為什么 MP 通過 Book::getId 就可以識別出來這里的屬性名。8pZ28資訊網——每日最新資訊28at.com

1. 源碼分析

這個問題其實好解決,我們順著 qw.eq 這個方法往下看就可以了,這個方法在執行的過程中幾經輾轉會來到 getColumnCache 方法中,這個方法就是解析出來屬性值的地方。8pZ28資訊網——每日最新資訊28at.com

protected ColumnCache getColumnCache(SFunction<T, ?> column) {    LambdaMeta meta = LambdaUtils.extract(column);    String fieldName = PropertyNamer.methodToProperty(meta.getImplMethodName());    Class<?> instantiatedClass = meta.getInstantiatedClass();    tryInitCache(instantiatedClass);    return getColumnCache(fieldName, instantiatedClass);}

首先這里先將我們傳入的 Lambda 表達式通過 LambdaUtils.extract 方法解析出來一個 LambdaMeta 對象。8pZ28資訊網——每日最新資訊28at.com

public static <T> LambdaMeta extract(SFunction<T, ?> func) {    // 1. IDEA 調試模式下 lambda 表達式是一個代理    if (func instanceof Proxy) {        return new IdeaProxyLambdaMeta((Proxy) func);    }    // 2. 反射讀取    try {        Method method = func.getClass().getDeclaredMethod("writeReplace");        method.setAccessible(true);        return new ReflectLambdaMeta((SerializedLambda) method.invoke(func), func.getClass().getClassLoader());    } catch (Throwable e) {        // 3. 反射失敗使用序列化的方式讀取        return new ShadowLambdaMeta(com.baomidou.mybatisplus.core.toolkit.support.SerializedLambda.extract(func));    }}

這塊的重點其實就在反射讀取這塊,這是從我們傳入的 Lambda 中找到了一個名為 writeReplace 的方法,并且通過反射執行了這個方法,然后將執行結果封裝為一個 ReflectLambdaMeta 對象返回。8pZ28資訊網——每日最新資訊28at.com

接下來回到 getColumnCache 方法中,繼續通過 String fieldName = PropertyNamer.methodToProperty(meta.getImplMethodName()); 獲取到屬性名稱。8pZ28資訊網——每日最新資訊28at.com

這里有一個 meta.getImplMethodName() 方法,這個方法的拿到的其實就是我們 Lambda 表達式中的方法名,也就是 getId,然后再通過 PropertyNamer.methodToProperty 對這個方法名進行處理,最終拿到屬性名:8pZ28資訊網——每日最新資訊28at.com

public static String methodToProperty(String name) {  if (name.startsWith("is")) {    name = name.substring(2);  } else if (name.startsWith("get") || name.startsWith("set")) {    name = name.substring(3);  } else {    throw new ReflectionException(        "Error parsing property name '" + name + "'.  Didn't start with 'is', 'get' or 'set'.");  }  if (name.length() == 1 || name.length() > 1 && !Character.isUpperCase(name.charAt(1))) {    name = name.substring(0, 1).toLowerCase(Locale.ENGLISH) + name.substring(1);  }  return name;}

大家看到,這個解析的過程其實就是把方法名的前綴 get/set/is 這些去掉,然后剩余的字符串首字母小寫之后返回。8pZ28資訊網——每日最新資訊28at.com

這就是我們傳入 Book::getId,最終能夠拿到 id 這個名稱的原因。8pZ28資訊網——每日最新資訊28at.com

現在的問題變成了 writeReplace 方法究竟是個什么方法?8pZ28資訊網——每日最新資訊28at.com

2. writeReplace

這個方法其實是系統底層自動生成的。我們可以將 Lambda 表達式在運行時生成的字節碼保存下來,然后進行反編譯,這樣就能夠看到 writeReplace 方法了。8pZ28資訊網——每日最新資訊28at.com

如果需要將 Lambda 運行時生成的字節碼保存,需要在啟動參數中添加如下內容:8pZ28資訊網——每日最新資訊28at.com

-Djdk.internal.lambda.dumpProxyClasses=/Users/sang/workspace/code/mp_demo/lambda/

等于號后面的部分是指定生成的字節碼的保存位置,大家可以根據自己的實際情況去配置。8pZ28資訊網——每日最新資訊28at.com

以本文一開頭的 Lambda 表達式為例,最終生成的字節碼反編譯之后,內容如下:8pZ28資訊網——每日最新資訊28at.com

final class MpDemo02ApplicationTests$$Lambda$1164 implements SFunction {    private MpDemo02ApplicationTests$$Lambda$1164() {    }    public Object apply(Object var1) {        return ((Book)var1).getId();    }    private final Object writeReplace() {        return new SerializedLambda(MpDemo02ApplicationTests.class, "com/baomidou/mybatisplus/core/toolkit/support/SFunction", "apply", "(Ljava/lang/Object;)Ljava/lang/Object;", 5, "org/javaboy/mp_demo02/model/Book", "getId", "()Ljava/lang/Integer;", "(Lorg/javaboy/mp_demo02/model/Book;)Ljava/lang/Object;", new Object[0]);    }}

大家可以看到,apply 方法實際上是重寫的接口的方法,在這個方法中將傳入的對象強轉為 Book 類型,然后調用其 getId 方法。8pZ28資訊網——每日最新資訊28at.com

然后大家看到,反編譯之后多了一個 writeReplace 方法,這個方法的返回值是一個 SerializedLambda,這個 SerializedLambda 對象其實就是對 Lambda 表達式的描述?;旧厦總€參數都能做到見名知意,我這里說一下第七個參數,值是 getId,這個參數的變量名是 implMethodName,這就是我們 Lambda 表達式中給出來的變量名。這也是第一小節中,meta.getImplMethodName() 所獲取到的值。8pZ28資訊網——每日最新資訊28at.com

這下就清楚了,為什么寫了 Book::getId 就能拿到屬性名了。8pZ28資訊網——每日最新資訊28at.com

3. 擴展知識

有的小伙伴注意到,在 qw.eq(Book::getId, 2); 方法中,第一個參數是一個 SFunction 的實例,那就說我直接給一個 SFunction 的實例,不用 Lambda。大家注意,這種寫法不對!8pZ28資訊網——每日最新資訊28at.com

原因在于經過前面的源碼分析之后,我們發現,MP 中根據 Book::getId 去獲取屬性名稱,一個關鍵點是利用 Lambda 在執行的時候生成的字節碼去獲取,如果你都沒有用 Lambda,那也就不會生成所謂的 Lambda 字節碼,也就不存在 writeReplace 方法,按照前文所分析的源碼,就無法獲取到屬性名稱。8pZ28資訊網——每日最新資訊28at.com

還有小伙伴說,既然是 Lambda,那么我不用方法引用行不行?我像下面這樣寫行不行?8pZ28資訊網——每日最新資訊28at.com

LambdaQueryWrapper<Book> qw = new LambdaQueryWrapper<>();qw.eq(b -> b.getId(), 2);List<Book> list = bookMapper.selectList(qw);System.out.println("list = " + list);

這也是一個 Lambda,但是如果你這樣寫了,運行之后就會報錯。為什么呢?我們來看下這個 Lambda 生成的字節碼反編譯之后是什么樣的:8pZ28資訊網——每日最新資訊28at.com

final class MpDemo02ApplicationTests$$Lambda$1164 implements SFunction {    private MpDemo02ApplicationTests$$Lambda$1164() {    }    public Object apply(Object var1) {        return MpDemo02ApplicationTests.lambda$test18$3fed5817$1((Book)var1);    }    private final Object writeReplace() {        return new SerializedLambda(MpDemo02ApplicationTests.class, "com/baomidou/mybatisplus/core/toolkit/support/SFunction", "apply", "(Ljava/lang/Object;)Ljava/lang/Object;", 6, "org/javaboy/mp_demo02/MpDemo02ApplicationTests", "lambda$test18$3fed5817$1", "(Lorg/javaboy/mp_demo02/model/Book;)Ljava/lang/Object;", "(Lorg/javaboy/mp_demo02/model/Book;)Ljava/lang/Object;", new Object[0]);    }}

首先大家注意到 apply 方法生成的就不一樣,apply 里邊調用了 MpDemo02ApplicationTests.lambda$test18$3fed5817$1 方法,傳入了 Book 對象作為參數。這個方法內容相當于就是 return book.getId();。然后在 writeReplace 方法中,返回 SerializedLambda 對象的時候,implMethodName 的值就是 lambda$test18$3fed5817$1 了。回到本文一開始的源碼分析中,你會發現這樣的方法名就無法提取出來我們想要的屬性名。所以這種寫法也不對。8pZ28資訊網——每日最新資訊28at.com

從這里大家也可以看到,類似于 b -> b.getId() 這樣的 Lambda,和方法引用 Book::getId 在底層是不同的。8pZ28資訊網——每日最新資訊28at.com

再給小伙伴們舉個例子,比如下面一段代碼:8pZ28資訊網——每日最新資訊28at.com

public class Demo01 {    public static void main(String[] args) {        Consumer<String> out1 = System.out::println;        out1.accept("javaboy");        Consumer<String> out2 = s -> System.out.println(s);        out2.accept("江南一點雨");    }}

這里有兩個輸出,第一個是一個方法引用,第二個則是一個常規的 Lambda 表達式。這兩個執行起來效果是一致的,但是底層原理不同。8pZ28資訊網——每日最新資訊28at.com

先來看第一個底層生成的 Lambda 字節碼:8pZ28資訊網——每日最新資訊28at.com

final class Demo01$$Lambda$14 implements Consumer {    private final PrintStream arg$1;    private Demo01$$Lambda$14(PrintStream var1) {        this.arg$1 = var1;    }    public void accept(Object var1) {        this.arg$1.println((String)var1);    }}

可以看到,這里把 System.out 的值 PrintStream 作為構造函數的參數傳進來賦值給 arg變量,當調用方法的時候,再調用1.println 方法將字符串輸出。8pZ28資訊網——每日最新資訊28at.com

對于第二個底層生成的 Lambda 字節碼如下:8pZ28資訊網——每日最新資訊28at.com

final class Demo01$$Lambda$16 implements Consumer {    private Demo01$$Lambda$16() {    }    public void accept(Object var1) {        Demo01.lambda$main$0((String)var1);    }}

可以看到,這里有一個新的 lambda$main$0 方法,這個方法的底層邏輯其實就是我們自定義 Lambda 的時候寫的 System.out.println(s)。8pZ28資訊網——每日最新資訊28at.com

3. 小結

好啦,一篇小文,和小伙伴們探討下 MP 中 qw.eq(Book::getId, 2); 方法的底層邏輯。8pZ28資訊網——每日最新資訊28at.com

本文鏈接:http://m.www897cc.com/showinfo-26-82751-0.html通過方法引用獲取屬性名的底層邏輯是什么?

聲明:本網頁內容旨在傳播知識,若有侵權等問題請及時與本網聯系,我們將在第一時間刪除處理。郵件:2376512515@qq.com

上一篇: 深入理解C/C++指針的算術運算

下一篇: 一文讀懂Cache一致性原理

標簽:
  • 熱門焦點
Top 日韩成人免费在线_国产成人一二_精品国产免费人成电影在线观..._日本一区二区三区久久久久久久久不
国内精品久久久久久影视8| 久久精品日韩一区二区三区| 国内精品久久久久久| 影音先锋日韩资源| 亚洲精品国产精品国自产观看浪潮 | 国产精品大全| 国产欧美在线观看一区| 在线成人黄色| 日韩视频在线一区二区| 午夜国产精品影院在线观看| 久久亚洲精品网站| 欧美日韩亚洲激情| 国外成人在线视频| 亚洲福利视频专区| 亚洲一区二区伦理| 久热re这里精品视频在线6| 欧美理论大片| 国产日韩欧美综合在线| 亚洲精品国偷自产在线99热| 午夜精品国产更新| 久久先锋影音av| 欧美三级欧美一级| 在线观看欧美激情| 亚洲男人第一av网站| 欧美成人免费在线| 国产日韩一区二区三区| 亚洲精品一区二区在线| 久久国产成人| 国产精品vvv| 亚洲国产精品成人综合| 亚洲欧美在线一区二区| 欧美国产先锋| 国内揄拍国内精品久久| 亚洲一区二区免费视频| 欧美大色视频| 国内精品国语自产拍在线观看| 99精品99久久久久久宅男| 久久er99精品| 国产精品国产馆在线真实露脸| 亚洲成人资源| 久久9热精品视频| 欧美日韩一区二区三区高清| 亚洲国产成人精品久久久国产成人一区 | 亚洲日本va午夜在线影院| 午夜精品一区二区三区四区| 欧美精品在线免费播放| 激情欧美丁香| 午夜亚洲精品| 国产精品电影在线观看| 亚洲日本欧美天堂| 久久一二三区| 国产一区日韩欧美| 欧美亚洲三级| 国产精品视频精品| 亚洲视频在线免费观看| 欧美成人激情视频| 在线成人激情黄色| 亚洲欧美清纯在线制服| 欧美美女福利视频| 亚洲精品美女在线观看| 女人色偷偷aa久久天堂| 精品动漫3d一区二区三区免费 | 久久成人免费网| 国产精品极品美女粉嫩高清在线 | 激情久久综艺| 一区二区三区蜜桃网| 男男成人高潮片免费网站| 国产精品久久久久久久午夜| 日韩午夜在线电影| 狠狠v欧美v日韩v亚洲ⅴ| 国产精品久久久久7777婷婷| 欧美国产欧美亚洲国产日韩mv天天看完整 | 国产精品久久网站| 久久一区二区三区av| 久久久久国色av免费观看性色| 国产真实乱子伦精品视频| 夜夜嗨av一区二区三区四季av| 亚洲国产高清一区二区三区| 亚洲黄一区二区三区| 在线看视频不卡| 亚洲欧洲另类国产综合| 一区二区久久久久| 午夜影视日本亚洲欧洲精品| 在线午夜精品自拍| 亚洲视频一区二区| 黄色成人在线网站| 国产视频久久久久久久| 一色屋精品视频在线观看网站| 亚洲电影免费观看高清完整版在线 | 亚洲天堂网站在线观看视频| 久久精品1区| 欧美成人免费视频| 欧美性猛片xxxx免费看久爱| 欧美激情中文字幕乱码免费| 好看的av在线不卡观看| 久久精品91| 红桃视频亚洲| 美女福利精品视频| 亚洲欧洲日本国产| 欧美日韩精品欧美日韩精品| 亚洲手机在线| 国产精品一区二区三区乱码| 久久精品国产一区二区三区| 精品动漫3d一区二区三区免费版| 乱人伦精品视频在线观看| 亚洲欧洲精品一区二区三区 | 欧美日韩免费一区二区三区| 亚洲一区二区精品视频| 国产情侣久久| 老司机午夜免费精品视频| 亚洲精品五月天| 国产精品国产三级国产aⅴ浪潮| 午夜久久美女| 伊人久久婷婷| 欧美另类极品videosbest最新版本| 一本色道久久综合一区| 国产精品久久久久久影视| 久久精品国产综合| 亚洲精品一二区| 国产精品露脸自拍| 久久免费国产精品| 亚洲精品日韩在线观看| 国产精品乱码妇女bbbb| 久久伊人免费视频| 一本久久精品一区二区| 国产欧美日韩伦理| 女人香蕉久久**毛片精品| 亚洲一区三区电影在线观看| 韩国在线一区| 欧美日韩国产不卡| 欧美一区二区三区久久精品茉莉花 | 欧美va亚洲va国产综合| 夜夜嗨网站十八久久| 国产欧美亚洲视频| 欧美高清成人| 午夜精品福利电影| 91久久精品国产91久久性色tv | 伊人男人综合视频网| 欧美日韩国产在线播放网站| 亚洲男人的天堂在线| 亚洲成色777777女色窝| 欧美午夜免费| 久久综合色播五月| 亚洲永久免费观看| 亚洲国产成人精品女人久久久| 国产精品视频内| 免费一级欧美片在线播放| 午夜精品久久久久久久99黑人| 亚洲国产成人午夜在线一区| 国产精品永久在线| 欧美日韩ab片| 久久永久免费| 性娇小13――14欧美| 亚洲看片一区| 尹人成人综合网| 国产精品久久国产三级国电话系列| 美国成人毛片| 欧美一区二区精美| 99精品久久久| 在线看片欧美| 国产午夜精品在线观看| 欧美三级午夜理伦三级中视频| 久久只精品国产| 性欧美在线看片a免费观看| 亚洲精品专区| 1769国产精品| 国产手机视频一区二区| 欧美视频日韩| 欧美人与禽性xxxxx杂性| 久久偷看各类wc女厕嘘嘘偷窃| 亚洲一区二区三区免费观看| 亚洲黄色天堂| 国产伦精品一区二区| 欧美精品免费播放| 免费成人毛片| 久久久成人精品| 亚洲欧美激情四射在线日| 日韩午夜在线电影| 亚洲精品久久久一区二区三区| 在线免费观看日本一区| 国产日韩一区二区三区在线播放| 欧美日韩精品免费观看视一区二区 | 欧美国产乱视频| 久久精品视频在线观看| 99国产精品| 欧美一区日本一区韩国一区| 亚洲午夜精品久久久久久app| 亚洲欧洲另类| 国产偷国产偷精品高清尤物| 国产精品免费在线| 欧美精品在线一区二区| 免播放器亚洲| 新67194成人永久网站| 亚洲在线视频网站| 亚洲一区免费观看| 亚洲午夜视频在线观看| 亚洲香蕉视频| 亚洲综合电影一区二区三区| 亚洲夜晚福利在线观看| 亚洲图片在线观看| 亚洲永久免费精品| 亚洲欧美日韩国产成人精品影院| 一区二区高清在线观看| 一本色道久久精品|