精品欧美一区二区三区在线观看 _久久久久国色av免费观看性色_国产精品久久在线观看_亚洲第一综合网站_91精品又粗又猛又爽_小泽玛利亚一区二区免费_91亚洲精品国偷拍自产在线观看 _久久精品视频在线播放_美女精品久久久_欧美日韩国产成人在线

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

開發 前端
有的小伙伴注意到,在 qw.eq(Book::getId, 2); 方法中,第一個參數是一個 SFunction 的實例,那就說我直接給一個 SFunction 的實例,不用 Lambda。大家注意,這種寫法不對!

很多小伙伴可能都用過 MyBatis-Plus,這里邊我們構造 where 條件的時候,可以直接通過方法引用的方式去指定屬性名:

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 就可以識別出來這里的屬性名。

1. 源碼分析

這個問題其實好解決,我們順著 qw.eq 這個方法往下看就可以了,這個方法在執行的過程中幾經輾轉會來到 getColumnCache 方法中,這個方法就是解析出來屬性值的地方。

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 對象。

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 對象返回。

接下來回到 getColumnCache 方法中,繼續通過 String fieldName = PropertyNamer.methodToProperty(meta.getImplMethodName()); 獲取到屬性名稱。

這里有一個 meta.getImplMethodName() 方法,這個方法的拿到的其實就是我們 Lambda 表達式中的方法名,也就是 getId,然后再通過 PropertyNamer.methodToProperty 對這個方法名進行處理,最終拿到屬性名:

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 這些去掉,然后剩余的字符串首字母小寫之后返回。

這就是我們傳入 Book::getId,最終能夠拿到 id 這個名稱的原因。

現在的問題變成了 writeReplace 方法究竟是個什么方法?

2. writeReplace

這個方法其實是系統底層自動生成的。我們可以將 Lambda 表達式在運行時生成的字節碼保存下來,然后進行反編譯,這樣就能夠看到 writeReplace 方法了。

如果需要將 Lambda 運行時生成的字節碼保存,需要在啟動參數中添加如下內容:

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

等于號后面的部分是指定生成的字節碼的保存位置,大家可以根據自己的實際情況去配置。

以本文一開頭的 Lambda 表達式為例,最終生成的字節碼反編譯之后,內容如下:

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 方法。

然后大家看到,反編譯之后多了一個 writeReplace 方法,這個方法的返回值是一個 SerializedLambda,這個 SerializedLambda 對象其實就是對 Lambda 表達式的描述。基本上每個參數都能做到見名知意,我這里說一下第七個參數,值是 getId,這個參數的變量名是 implMethodName,這就是我們 Lambda 表達式中給出來的變量名。這也是第一小節中,meta.getImplMethodName() 所獲取到的值。

這下就清楚了,為什么寫了 Book::getId 就能拿到屬性名了。

3. 擴展知識

有的小伙伴注意到,在 qw.eq(Book::getId, 2); 方法中,第一個參數是一個 SFunction 的實例,那就說我直接給一個 SFunction 的實例,不用 Lambda。大家注意,這種寫法不對!

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

還有小伙伴說,既然是 Lambda,那么我不用方法引用行不行?我像下面這樣寫行不行?

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

這也是一個 Lambda,但是如果你這樣寫了,運行之后就會報錯。為什么呢?我們來看下這個 Lambda 生成的字節碼反編譯之后是什么樣的:

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 了。回到本文一開始的源碼分析中,你會發現這樣的方法名就無法提取出來我們想要的屬性名。所以這種寫法也不對。

從這里大家也可以看到,類似于 b -> b.getId() 這樣的 Lambda,和方法引用 Book::getId 在底層是不同的。

再給小伙伴們舉個例子,比如下面一段代碼:

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 表達式。這兩個執行起來效果是一致的,但是底層原理不同。

先來看第一個底層生成的 Lambda 字節碼:

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 方法將字符串輸出。

對于第二個底層生成的 Lambda 字節碼如下:

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)。

3. 小結

好啦,一篇小文,和小伙伴們探討下 MP 中 qw.eq(Book::getId, 2); 方法的底層邏輯。

責任編輯:武曉燕 來源: 江南一點雨
相關推薦

2023-09-25 15:13:57

數字化轉型

2021-08-16 11:58:15

CSS顏色屬性前端

2019-10-28 10:41:13

數據庫POLARDB底層邏輯

2021-12-24 10:39:33

軟件開發 技術

2021-04-15 18:44:15

2023-12-14 15:01:04

數字化轉型數據智能數字化

2025-05-28 04:00:00

AI人工智能大數據

2025-03-27 04:00:00

2021-04-26 07:51:00

JavaScript方法函數

2023-10-31 15:08:56

WorkBoxServiceWorker

2021-09-10 06:50:03

HashMapHash方法

2010-09-26 16:46:05

2022-03-03 10:20:02

編譯器橋接Java

2023-03-17 16:47:23

索引開發大數據

2022-08-26 08:35:59

對象設計底層

2022-01-13 13:24:16

工具底層邏輯

2023-12-13 10:11:14

數據庫ACID數據

2020-03-23 10:09:27

云安全云計算

2013-11-28 14:21:31

百度
點贊
收藏

51CTO技術棧公眾號

三级网站在线播放 | 亚洲精品1区| 宅男噜噜噜66一区二区66| 亚洲一卡二卡三卡| 性猛交xxxx乱大交孕妇印度| 亚洲精品护士| 这里只有精品丝袜| 天天爽夜夜爽视频| а√天堂中文在线资源8| 26uuu国产日韩综合| 成人国产精品色哟哟| 亚洲一区二区91| 精品国产一区探花在线观看| 日韩一区二区三区免费观看| 97国产精东麻豆人妻电影| 岛国大片在线观看| 国产成人综合亚洲91猫咪| 热久久这里只有精品| 亚洲伦理一区二区三区| 婷婷综合电影| 欧美成人一区二区三区片免费| 日韩精品视频一区二区在线观看| 91.xxx.高清在线| eeuss鲁片一区二区三区在线观看| 国产精品三级美女白浆呻吟 | 国产成人精品在线观看| 91 在线视频| 国产成人精品免费视| 日韩女优av电影| 午夜激情av在线| 在线一区av| 亚洲一区中文在线| 亚洲欧洲一区二区福利| 青青草观看免费视频在线| 国产自产视频一区二区三区| 国产精品狠色婷| 日韩在线视频不卡| 黄色免费成人| 欧美成人精品激情在线观看| 亚洲黄色网址大全| 亚洲国产合集| 亚洲国产欧美一区二区丝袜黑人| 亚洲男人天堂2021| 欧美综合影院| 欧美自拍丝袜亚洲| 农村妇女精品一二区| 91探花在线观看| 一区二区三区丝袜| 性做爰过程免费播放| 99re热久久这里只有精品34| 久久久久久日产精品| 久久国产精品亚洲va麻豆| 国产黄色美女视频| 国产一区二区免费在线| 成人羞羞国产免费| 亚洲一区二区影视| 美国毛片一区二区| 国产精品自产拍在线观看中文| 老熟妇一区二区三区| 亚洲日本视频| 欧美亚洲第一页| 国产超碰人人爽人人做人人爱| 亚洲日本免费| 浅井舞香一区二区| 国产成人精品777777| 亚洲欧美久久久| 欧美中在线观看| 69亚洲精品久久久蜜桃小说| 久久精品盗摄| 国产精品美女久久| 一级黄色大毛片| 国产一区在线不卡| 91久久爱成人| 日本美女一级视频| 91在线观看污| 日韩一区免费观看| 日本中文字幕在线看| 亚洲特级片在线| 青青视频免费在线| 国产高清在线a视频大全| 亚洲午夜免费电影| 亚洲 高清 成人 动漫| 粉嫩一区二区三区| 欧美日韩成人综合在线一区二区| 中文 日韩 欧美| 国产一区二区三区不卡av| 亚洲精品v欧美精品v日韩精品| 亚洲做受高潮无遮挡| 成人嫩草影院| 欧美另类99xxxxx| 亚洲黄色小说图片| 久草这里只有精品视频| 成人欧美一区二区三区视频 | 日韩三级成人| 欧美一区二区免费视频| 国产又黄又粗又猛又爽的视频| 国产欧美日韩在线观看视频| 久久人人爽人人爽人人片亚洲| 久久中文字幕无码| 日韩精品电影在线| 成人羞羞视频免费| 国产专区在线播放| 亚洲精品五月天| 免费av网址在线| 麻豆精品久久| 国产亚洲精品久久久优势| 欧美黑人猛猛猛| 免费亚洲视频| 99热在线国产| jzzjzzjzz亚洲成熟少妇| 亚洲综合色视频| 欧美成人黄色网址| 久久悠悠精品综合网| 综合久久五月天| 国产成人亚洲精品自产在线| 石原莉奈在线亚洲二区| 不卡一区二区三区视频| 国产精品ⅴa有声小说| 亚洲综合男人的天堂| 波多结衣在线观看| 亚洲妇女av| 午夜精品免费视频| 国产精品美女一区| 国产拍揄自揄精品视频麻豆| 日韩欧美猛交xxxxx无码| 日韩精品一级毛片在线播放| 亚洲人成在线免费观看| 日本三级片在线观看| 极品少妇一区二区| 日韩精品一线二线三线| 欧美激情网站| 亚洲第一二三四五区| 欧美h片在线观看| 青草av.久久免费一区| 久久精品第九区免费观看| 亚洲羞羞网站| 91麻豆精品国产91久久久久久| 亚洲天堂成人av| 国产主播一区| 亚洲最大福利网站| 国产激情在线视频| 欧美二区在线观看| 网站永久看片免费| 男人的天堂久久精品| 欧美一级爱爱| 一二区成人影院电影网| 亚洲欧美国产va在线影院| 五月天综合激情| 99久久伊人精品| 成年人网站国产| 精品福利网址导航| 国内精品模特av私拍在线观看| 亚洲国产成人在线观看| 亚洲精品日韩一| 国产亚洲色婷婷久久| 女主播福利一区| 亚洲自拍偷拍一区| 在线你懂的视频| 精品国产伦一区二区三区免费| avtt天堂在线| 风间由美性色一区二区三区| 国产成人艳妇aa视频在线 | 久中文字幕一区| 小视频免费在线观看| 日韩成人中文电影| 色av性av丰满av| 欧美高清在线精品一区| 亚洲精品视频导航| 国产韩国精品一区二区三区| 成人福利在线视频| 在线观看wwwxxxx| 精品国内二区三区| 毛片在线免费视频| 国产丝袜欧美中文另类| 91 在线视频观看| 婷婷久久综合| 国产精品免费一区二区三区| 日韩伦理在线| 这里只有精品在线播放| 国产又粗又黄视频| 亚洲午夜在线视频| 国产精品815.cc红桃| 美美哒免费高清在线观看视频一区二区| 欧美日韩一级在线| 国产成人澳门| 欧美中文字幕视频| 欧美黄色激情| 亚洲电影天堂av| 天天干,天天干| 日韩理论片在线| 少妇被狂c下部羞羞漫画| 首页欧美精品中文字幕| 亚洲欧美一二三| 日韩成人午夜| 国产中文字幕日韩| 搞黄网站在线看| 亚洲天堂av高清| 国产不卡av在线播放| 欧美日韩激情小视频| 2017亚洲天堂| 91丨porny丨国产| 午夜大片在线观看| 亚洲影院免费| 91精品国产毛片武则天| 伊人久久综合影院| 97se在线视频| 精品国模一区二区三区| 欧美激情视频一区| av在线资源网| 亚洲精品美女久久| 亚洲在线观看av| 黑人巨大精品欧美一区二区| 欧美老熟妇一区二区三区| 久久久久久久综合狠狠综合| 中文字幕av一区二区三区人妻少妇| 亚洲在线观看| 成人免费在线视频播放| 日本久久黄色| 久久精品欧美| 99久久人爽人人添人人澡| 国产精品美女久久久久久免费| 蜜臀久久精品| 久久久久久亚洲精品| 求av网址在线观看| 国产亚洲精品久久久久动| 免费观看毛片网站| 4438成人网| 久久久久久亚洲av无码专区| 精品成人av一区| 劲爆欧美第一页| 一区二区中文视频| 日韩不卡av在线| 久久久亚洲精品石原莉奈| 少妇被狂c下部羞羞漫画| 国产寡妇亲子伦一区二区| 日本免费色视频| 秋霞午夜鲁丝一区二区老狼| 欧美视频第一区| 国产日韩亚洲| 国产精品裸体瑜伽视频| 亚洲手机视频| 亚洲精品久久久久久久蜜桃臀| 一区二区电影| 日韩中文在线字幕| 国产精品久久久乱弄| 一区二区免费电影| 久久五月天小说| 中文字幕在线亚洲三区| 久久密一区二区三区| 亚洲精品一区二区三| 日韩在线综合| 一区二区三区的久久的视频| 色135综合网| 在线观看日韩羞羞视频| 婷婷久久国产对白刺激五月99| 在线免费一区| 欧美91大片| 欧美又粗又长又爽做受| 亚洲东热激情| 亚洲 高清 成人 动漫| 噜噜噜久久亚洲精品国产品小说| 黄色片久久久久| 日韩电影免费在线观看网站| 国产一级特黄a大片免费| 蜜桃视频第一区免费观看| 91视频这里只有精品| 国产一区二区三区日韩 | av在线国产精品| 91高跟黑色丝袜呻吟在线观看| 亚洲国产高清在线观看| 不卡视频一区| 中文字幕中文字幕精品| 污视频在线免费观看一区二区三区 | 亚洲精品少妇| 韩国一区二区av| 久久国产人妖系列| 人妻精油按摩bd高清中文字幕| 岛国一区二区在线观看| 成人免费看aa片| 欧美国产禁国产网站cc| 国产高潮国产高潮久久久91| 亚洲国产一区二区三区青草影视| 欧美特黄aaaaaa| 欧美美女喷水视频| 亚洲精品成av人片天堂无码| 精品亚洲一区二区三区四区五区| 成黄免费在线| 欧美激情a在线| 欧美自拍电影| 亚洲最大福利网站| 要久久爱电视剧全集完整观看| 深田咏美在线x99av| 欧美激情视频一区二区三区免费| 丰满少妇被猛烈进入高清播放| 捆绑紧缚一区二区三区视频| 一区二区三区四区影院| 亚洲国产精品精华液ab| 欧美日韩在线观看成人| 色偷偷成人一区二区三区91| 国产探花精品一区二区| 精品无码久久久久久国产| av电影在线观看| 国内精品久久久| 欧美成人家庭影院| 久久国产精品99久久久久久丝袜| 久久精品影视| 免费激情视频在线观看| 高清在线成人网| 日日操免费视频| 欧美日韩国产一中文字不卡| 国产精品九九九九| 亚洲欧洲日韩国产| 丁香花高清在线观看完整版| 国产精品视频xxx| 婷婷综合成人| 国产一二三在线视频| 久久国产精品一区二区| 李宗瑞91在线正在播放| 亚洲综合无码一区二区| 国产精品久久久久久免费 | 欧美日韩在线亚洲一区蜜芽| 内射无码专区久久亚洲| 欧美成人合集magnet| 成人在线不卡| 日本成人黄色| 亚洲在线电影| 91视频啊啊啊| 亚洲福利国产精品| 99热这里只有精品3| 色av吧综合网| 国产成人免费9x9x人网站视频 | 蜜桃av鲁一鲁一鲁一鲁俄罗斯的| www.亚洲男人天堂| 成人四虎影院| 亚洲成人网上| 日韩不卡一区二区| 亚洲a v网站| 一本色道亚洲精品aⅴ| 婷婷av一区二区三区| 久久全球大尺度高清视频| 日韩黄色av| 免费看日本黄色| 国产suv精品一区二区883| 日韩欧美123区| 91精品国模一区二区三区| 成年人视频在线看| 国产欧美精品在线| 日韩精品欧美| 极品粉嫩美女露脸啪啪| 国产精品伦一区二区三级视频| 久久国产香蕉视频| 在线亚洲欧美视频| 日日狠狠久久| 国产日韩欧美大片| 国产精品99久久久久久久女警| 欧美日韩偷拍视频| 精品少妇一区二区三区在线播放| 怡红院在线观看| 国产日韩欧美综合精品| 亚洲日产国产精品| 久久中文字幕人妻| 欧美伊人久久大香线蕉综合69| 川上优的av在线一区二区| 成人a级免费视频| 女生裸体视频一区二区三区| 黄色国产在线视频| 精品久久久国产| 国产黄色片在线观看| 国产精品永久在线| 欧美99在线视频观看| 中国xxxx性xxxx产国| 日韩欧美国产免费播放| 你懂的在线播放| 国产在线播放91| 黄色欧美日韩| 国产男女猛烈无遮挡a片漫画 | 国产精品国产精品国产专区不片| 国产一区二区三区在线观看| 欧美国产极速在线| 日韩高清一级| 伊人网在线综合| 香蕉av福利精品导航| 国模吧精品人体gogo| 成人综合网网址| 99视频精品免费观看| wwwww黄色| 精品国产99国产精品| 香蕉成人av| 法国空姐在线观看免费| 97成人超碰视| 国产又粗又黄又爽的视频| 久久人人爽人人爽人人片av高请| 国产一区二区在线| 波多野结衣免费观看| 欧美视频在线看| 日本高清中文字幕在线| 国内外成人免费视频| 麻豆精品视频在线观看| 五月婷婷开心网| 久久久国产视频|