精品欧美一区二区三区在线观看 _久久久久国色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技術棧公眾號

亚洲 欧美 日韩 国产综合 在线 | 久久狠狠婷婷| 国产视频久久久久| 中日韩av在线播放| 黄色小说在线播放| 国产午夜精品一区二区三区嫩草| 91精品国产自产在线老师啪| 精品视频在线观看免费| 国内精品久久久久久99蜜桃| 日韩欧美一级在线播放| 久久久久久久片| 第四色日韩影片| 中文字幕一区二区三| 久久久久成人精品免费播放动漫| 91精品国产乱码久久久久| 亚洲伦伦在线| 久久天天躁狠狠躁夜夜躁| 男男一级淫片免费播放| 久久青草免费| 欧美日韩精品在线| 日本a级片在线观看| 毛片网站在线| 成人福利电影精品一区二区在线观看| 国产精品久久久久久久久久久久久 | 97久久综合区小说区图片区| 欧美午夜精品免费| 免费在线a视频| 蜜桃传媒在线观看免费进入 | 深夜成人福利| 亚洲国产精品嫩草影院| 日本福利视频导航| 成人午夜影视| 国产主播中文字幕| 五月六月丁香婷婷| 性欧美又大又长又硬| 日韩一区欧美一区| 欧美日韩综合久久| 人妻91麻豆一区二区三区| 久久精品国产精品亚洲红杏| 国产91免费观看| 日本一区二区欧美| 国产精品多人| 久久成人国产精品| 国产精品精品软件男同| 精品国内自产拍在线观看视频 | 国产91精品入口17c| 成人性生交大免费看| 中文字幕区一区二区三| 欧美精品xxxxbbbb| 久久婷五月综合| 日韩欧美一区二区三区免费观看 | 99久久久久免费精品国产| 亚洲综合精品一区二区| 99久久国产热无码精品免费| 久久精品久久99精品久久| 国产精品久久久久久久久免费看| 国产精品欧美综合| 日韩av在线播放中文字幕| 国产成人高清激情视频在线观看 | 91超碰这里只有精品国产| 亚洲高清在线免费观看| 777午夜精品电影免费看| 在线看不卡av| 亚洲欧美自偷自拍另类| 国产日韩一区二区三免费高清| 欧美日韩国产精品自在自线| 欧美一级xxxx| 精品一区91| 精品欧美久久久| 亚洲天堂资源在线| 精品视频免费在线观看| 色青青草原桃花久久综合| 青青操在线视频观看| 欧美777四色影| 韩国视频理论视频久久| 中文字幕亚洲乱码熟女1区2区| 日本欧美加勒比视频| 成人欧美一区二区三区在线湿哒哒| 国产又粗又猛视频免费| 国产精品69毛片高清亚洲| 国产精品区二区三区日本| 飘雪影院手机免费高清版在线观看 | 国语精品一区| 日本亚洲欧美成人| 91久久国语露脸精品国产高跟| 国产精品亚洲人在线观看| 国产精品美女诱惑| 国产粉嫩一区二区三区在线观看| 亚洲欧洲美洲综合色网| 无码人妻少妇伦在线电影| 中文字幕一区久| 欧美精品视频www在线观看 | av免费中文字幕| 九色成人搞黄网站| 精品裸体舞一区二区三区| 蜜桃无码一区二区三区| 成年美女黄网站色大片不卡| 国模 一区 二区 三区| 欧美极品在线视频| 婷婷激情五月综合| 国产一区二区女| 久久久久九九九| 国产日产一区二区三区| 国产精品免费精品自在线观看| 欧美日韩一区在线| 第一页在线视频| 青草国产精品| 高清欧美电影在线| 中文字幕日韩国产| 99精品桃花视频在线观看| 久久免费看毛片| 欧美大胆性生话| 精品国产网站在线观看| jizz日本在线播放| 先锋亚洲精品| www.成人三级视频| 欧美成人性生活视频| 欧美日韩另类在线| 最新国产精品自拍| 久久亚洲精品中文字幕蜜潮电影| 日本成人免费在线| 成人无码一区二区三区| 亚洲欧美日韩在线播放| 91女神在线观看| 国产欧美日韩精品一区二区免费 | 日本中文不卡| 蜜臀久久精品| 欧美mv日韩mv国产| 成人免费精品动漫网站| 免费不卡在线观看| 奇米精品在线| 在线播放高清视频www| 亚洲国产精品小视频| 青青草成人免费| 久久97超碰色| 综合视频免费看| 亚洲精品三区| 精品国模在线视频| 一区二区三区亚洲视频| 亚洲一区在线不卡| 成人午夜精品| 亚洲乱亚洲乱妇无码| 欧美丰满艳妇bbwbbw| 精彩视频一区二区三区| 日韩视频在线观看国产| 亚洲永久av| 亚洲欧美一区二区三区四区| 国产农村妇女aaaaa视频| 9色porny自拍视频一区二区| 阿v天堂2018| 日本欧美高清| 欧美一级大片在线观看| 深夜福利在线观看直播| 欧美体内谢she精2性欧美| 久久福利小视频| 国产日韩欧美一区二区三区在线观看| 精品国产免费久久久久久尖叫 | 成年人在线观看视频免费| 亚洲调教一区| 国产va免费精品高清在线| 国产精品一区二区婷婷| 欧美图区在线视频| 免费国产羞羞网站美图| 丰满岳乱妇一区二区三区| 欧美人成在线观看| 亚洲黄页在线观看| 国产精品激情av在线播放| 日韩黄色影院| 精品国产伦一区二区三区观看方式 | aaa一级黄色片| 欧美体内she精视频在线观看| 国产精品大全| 欧美xxx性| 久久精品国产精品亚洲| 黑人乱码一区二区三区av| 欧美色另类天堂2015| a级在线免费观看| 国产在线精品免费av| 国产人妻777人伦精品hd| 奇米狠狠一区二区三区| 91午夜理伦私人影院| av成人 com a| 中文字幕日韩欧美精品在线观看| 国产成人免费看一级大黄| 精品久久中文字幕| 国产精品麻豆免费版现看视频| 国产成人午夜精品5599| 国产三区在线视频| 你懂的视频一区二区| 久久人人97超碰人人澡爱香蕉| 日本综合久久| 欧美国产在线视频| 成人在线观看黄色| 亚洲电影免费观看| 在线视频播放大全| 亚欧色一区w666天堂| 青青操在线播放| www.66久久| 狠狠干狠狠操视频| 亚洲专区一区二区三区| 国产免费一区二区三区四在线播放 | 老**午夜毛片一区二区三区| 好色先生视频污| 亚洲影院天堂中文av色| 91影院未满十八岁禁止入内| 欧美日韩不卡| 4k岛国日韩精品**专区| www在线免费观看视频| 亚洲男人7777| 欧美 日韩 国产 成人 在线 91| 欧美日韩电影在线| 国产免费一级视频| 亚洲在线视频一区| 午夜成人亚洲理伦片在线观看| 久久亚洲精华国产精华液 | 国产丝袜美腿一区二区三区| 中文字幕一区二区三区人妻在线视频| 奇米影视一区二区三区| 波多野结衣家庭教师在线| 欧美fxxxxxx另类| 99亚洲精品视频| 日韩dvd碟片| 日本欧洲国产一区二区| 女同久久另类99精品国产| 91超碰在线免费观看| 日韩在线激情| 国产精品永久免费| 日韩欧美看国产| 国产91精品青草社区| www.youjizz.com在线| 欧美高清一级大片| 怡红院在线播放| 欧美成人国产va精品日本一级| 毛片在线不卡| 色偷偷噜噜噜亚洲男人的天堂 | 欧美激情一区二区三区高清视频 | 91国产美女视频| 9999热视频在线观看| 国内外成人免费激情在线视频网站 | 久久人人爽爽爽人久久久| 99久久人妻精品免费二区| 成人免费毛片aaaaa**| 香蕉久久久久久av成人| 成人一级视频在线观看| 男人的天堂免费| 国产盗摄一区二区三区| 女同性αv亚洲女同志| 国产福利视频一区二区三区| 91精品人妻一区二区三区四区| 粉嫩一区二区三区性色av| 91成人在线观看喷潮蘑菇| 成人a免费在线看| 91丨porny丨对白| 92国产精品观看| 免费看91的网站| 中文在线一区二区 | 欧美午夜片在线免费观看| 欧美a视频在线观看| 欧美影院一区二区三区| 中文人妻熟女乱又乱精品| 欧美日韩精品一区二区三区| 国产精品久久久久久免费免熟| 制服丝袜成人动漫| 性欧美8khd高清极品| 亚洲第一精品夜夜躁人人躁| 日韩亚洲视频在线观看| 伊人青青综合网站| 黄色一级片在线观看| 欧美高清视频一区二区| 樱桃视频成人在线观看| 国产精品日韩在线| 欧美电影院免费观看| 国产区二精品视| 成人羞羞网站入口免费| 91大学生片黄在线观看| 中国女人久久久| 在线观看的毛片| 国产成人8x视频一区二区| 亚洲做受高潮无遮挡| 国产精品灌醉下药二区| 精品无码一区二区三区电影桃花| 日韩欧美中文第一页| 国产又大又黄的视频| 亚洲成人av片在线观看| av在线第一页| 欧美激情xxxxx| 亚洲不卡系列| 国产69精品久久久久9999apgf| 欧美日韩xxxx| 青春草国产视频| 蜜桃视频一区二区| 欧美激情 亚洲| 成人欧美一区二区三区小说| 日本学生初尝黑人巨免费视频| 欧美日韩在线免费视频| 色婷婷av一区二区三| 中文字幕亚洲一区| 女海盗2成人h版中文字幕| 成人精品网站在线观看| 亚洲第一论坛sis| 青青草综合视频| 日本色综合中文字幕| 成人在线视频免费播放| 亚洲视频一二三| 成人公开免费视频| 亚洲国产91精品在线观看| 日本美女高清在线观看免费| 69**夜色精品国产69乱| 国产 日韩 欧美| 日韩久久久久久久| 久久国产精品久久久久久电车| 毛毛毛毛毛毛毛片123| 中日韩免费视频中文字幕| 亚洲精品www久久久久久| 91精品国产高清一区二区三区蜜臀| 每日更新在线观看av| 性欧美xxxx| av日韩在线播放| 日韩人妻精品一区二区三区| 日韩精品一区第一页| 久久国产精品无码一级毛片| 一区二区三区在线播| 97人妻人人澡人人爽人人精品| 一本色道久久综合亚洲精品小说| 国产无遮挡裸体视频在线观看| 97se国产在线视频| 亚洲久久久久| 老司机久久精品| 亚洲欧洲精品一区二区三区 | 精品99999| 国产美女福利在线观看| 亚洲综合最新在线| 91精品国产福利在线观看麻豆| wwww.国产| 欧美韩日一区二区三区四区| www.欧美色| 一本一本久久a久久精品牛牛影视| 色黄视频在线观看| 精品中文字幕人| 免费日韩视频| 亚洲国产天堂av| 在线观看日韩精品| 麻豆导航在线观看| 日本乱人伦a精品| 九九精品在线| 宅男噜噜噜66国产免费观看| 国产偷国产偷亚洲高清人白洁| 乱子伦一区二区三区| 在线观看欧美视频| 欧美一级做a| 玖玖精品在线视频| 成人中文字幕电影| 欧美成人aaaaⅴ片在线看| 亚洲激情第一页| 丝袜美腿一区| 亚洲欧洲一区二区福利| 久久www免费人成看片高清| 亚洲天堂黄色片| 精品少妇一区二区三区| 九色porny丨入口在线| 蜜桃免费一区二区三区| 人人爽香蕉精品| 波多野结衣家庭教师| 亚洲精品一区二区三区在线观看 | 800av在线播放| 日本高清成人免费播放| 欧美成人hd| 国产精品久久亚洲7777| 亚洲中字黄色| 国产福利在线导航| 精品剧情在线观看| 欧美男女交配| 欧美xxxx吸乳| 91蝌蚪国产九色| 91tv国产成人福利| 久久男人资源视频| 国产真实有声精品录音| 日韩av加勒比| 天天色图综合网| av大全在线免费看| 国产经典一区二区三区| 日本视频一区二区| 久草福利资源在线观看| 国产亚洲精品日韩| 91精品啪在线观看国产爱臀| 久久久久久久久久久久久国产精品| 1024亚洲合集| 色综合成人av| 18成人在线| 日韩高清在线观看| 久久免费公开视频| 中文在线资源观看视频网站免费不卡 | 影音先锋亚洲电影| 国产一区在线观看免费| 亚洲国产精久久久久久| 玖玖精品在线| 日本三级免费网站| 亚洲品质自拍视频网站| 久久99久久| 激情伦成人综合小说|