分析一下mybatis plus中的构造器 ::
举例: 实体类Entity中 存在name字段
在构造器中 我们写法是 wrapper.lambda.select(Entity::getName)…
其中最核心的部分就是将Function序列化成流 再将流自定义反序列化成 SerializedLambda 对象,在SerializedLambda中定义一个implMethodName字段 存储字符串 “getName” ,并对String做处理 截取Name 替换首字母为小写 name
另外有一种更简单的获取方式:
/**
* Description: 函数式参数转回String属性名
* date: 2021/06/09 12:05
* @param sFunctionField
* @return
* @author qkj
*/
public static <T> String function2Str (SFunction<T, ?> sFunctionField) {
Method writeReplace = null;
try {
// 函数式方法可以直接调用writeReplace
writeReplace = sFunctionField.getClass().getDeclaredMethod("writeReplace");
} catch (NoSuchMethodException e) {
e.printStackTrace();
}
writeReplace.setAccessible(true);
String fieldName = "";
try {
// 序列化
Object object = writeReplace.invoke(sFunctionField);
// 反序列化
SerializedLambda serializedLambda = (SerializedLambda) object;
String methodName = serializedLambda.getImplMethodName();
String temp = methodName.substring(3);
fieldName = temp.replaceFirst(temp.substring(0,1),temp.substring(0,1).toLowerCase());
} catch (IllegalAccessException e) {
e.printStackTrace();
} catch (InvocationTargetException e) {
e.printStackTrace();
}
return fieldName;
}
关于function为什么可以调用writeReplace方法 ,这个真的是太冷门了,会用writeReplace方法的人不多,能理解原理的人少之又少。在时隔几年后 通过AI的帮助下 配合博主自己的理解 ,终于有所解惑:
我们将 Entity::getName 作为function2Str 方法的参数传入
(Entity::getName是个方法的引用 函数式编程中被视为Function的实现 所以可以传参)
(方法引用因为和lambda表达式高度相似 jvm处理也是一样的 所以一般不会区分)
方法引用和Lambda表达式都可以生成包含invokedynamic指令的字节码
通过invokedynamic字节码会生成一个新的类,这个新类的实例 在LambdaMetafactory中生产, 且这个新类实现了Function(的子类)。为了支持序列化,JVM在字节码层面 自动往新类中添加writeReplace方法,这个方法在序列化时被调用