java
2024-08-04
2024-08-04
家庭是用孜孜不倦的爱情的劳动建立起来的。——陀思妥耶夫斯基
介绍
LambdaUtil
是一个用于处理 Lambda 表达式的工具类,提供了解析、获取信息和构建 Lambda 方法的多种功能。
使用
方法介绍
Lambda 获取相关方法
getRealClass
获取 Lambda 实现类。resolve
解析 Lambda 表达式,并缓存结果。getMethodName
获取 Lambda 表达式的函数名称。getFieldName
获取 Lambda 表达式 Getter 或 Setter 对应的字段名称。buildGetter
构建 Getter 方法引用。buildSetter
构建 Setter 方法引用。build
构建指定方法的 Lambda 引用。toFunction
将BiFunction
转换为Function
。toPredicate
将BiPredicate
转换为Predicate
。toConsumer
将BiConsumer
转换为Consumer
。getInvokeMethod
获取函数的执行方法。
例子:
我们定义一个类:
1 | public class MyTeacher { |
获取 Lambda 实现类:
1 | MyTeacher myTeacher = new MyTeacher(); |
解析 Lambda 表达式:
1 | LambdaInfo lambdaInfo = LambdaUtil.resolve(myTeacher::getAge); |
获取 Lambda 表达式的函数名称:
1 | String methodName = LambdaUtil.getMethodName(myTeacher::getAge); |
获取 Lambda 表达式 Getter 对应的字段名称:
1 | String fieldName = LambdaUtil.getFieldName(myTeacher::getAge); |
构建 Getter 方法引用:
1 | Function<MyTeacher, Integer> getter = LambdaUtil.buildGetter(MyTeacher::getAge); |
支持全系列序列化lambda
这里是牺牲了部分易用性(主要是考虑到此功能一般用于工具类内部封装),需要手动指定传入的lambda类型,但保证了通用性,支持自定义可序列化函数式接口,改版后用法可以如下:
第一种,拆分为变量和赋值:
1 | SerFunction<MyTeacher, String> lambda = MyTeacher::getAge; |
第二种,强转指定序列化函数式接口类型:
1 | LambdaInfo lambdaInfo = LambdaUtil.resolve((SerFunction<Integer, MyTeacher[]>) MyTeacher[]::new); |
第三种,匿名序列化函数式接口(实际上也就是匿名内部类多实现接口):
1 | LambdaInfo lambdaInfo = LambdaUtil.resolve((Serializable & Function<Integer, MyTeacher[]>) MyTeacher[]::new); |
未来如果入参是泛型,还可以采取这种方式指定(暂定):
1 | final String fieldName = LambdaUtil.<SerFunction<MyTeacher, String>>getFieldName(MyTeacher::getAge); |
对于上述调整,避免了对每一种序列化的函数式接口都要对应实现一种重载方法,以外部指定的方式,来应万变。
通用执行入口
现在的LambdaUtil除了调writeReplace
拿到SerializedLambda
以外,还可以通过其信息+字段描述符,拿到反射的Executable
对象,相当于也有了通用的执行入口。
单元测试
单元测试覆盖了主要方法的使用场景,确保每个功能的正确性。
1 | public class LambdaUtilTest { |