统一LambdaUtil#getRealClass方法,并提供测试用例,还对传入lambda进行了校验,且完善了注释...

This commit is contained in:
VampireAchao 2022-03-25 19:34:24 +08:00
parent e622296e64
commit 40bc89511e
3 changed files with 204 additions and 88 deletions

View File

@ -1,14 +1,14 @@
package cn.hutool.core.lang.func;
import java.io.Serializable;
import java.lang.invoke.SerializedLambda;
import cn.hutool.core.bean.BeanUtil;
import cn.hutool.core.lang.SimpleCache;
import cn.hutool.core.util.ClassUtil;
import cn.hutool.core.util.ReflectUtil;
import cn.hutool.core.util.StrUtil;
import java.io.Serializable;
import java.lang.invoke.SerializedLambda;
/**
* Lambda相关工具类
*
@ -19,6 +19,58 @@ public class LambdaUtil {
private static final SimpleCache<String, SerializedLambda> cache = new SimpleCache<>();
/**
* 通过对象的方法或类的静态方法引用获取lambda实现类
* 传入lambda无参数但含有返回值的情况能够匹配到此方法
* <pre>{@code
* @Data
* @EqualsAndHashCode(callSuper = true)
* static class MyTeacher extends Entity<MyTeacher> {
*
* public String age;
*
* public static String takeAge() {
* return new MyTeacher().getAge();
* }
*
* }
* }</pre>
* <ul>
* <li>引用特定对象的实例方法<pre>{@code
* MyTeacher myTeacher = new MyTeacher();
* Class<MyTeacher> supplierClass = LambdaUtil.getRealClass(myTeacher::getAge);
* Assert.assertEquals(MyTeacher.class, supplierClass);
* }</pre></li>
* <li>引用静态无参方法<pre>{@code
* Class<MyTeacher> staticSupplierClass = LambdaUtil.getRealClass(MyTeacher::takeAge);
* Assert.assertEquals(MyTeacher.class, staticSupplierClass);
* }</pre></li>
* </ul>
* 在以下场景无法获取到正确类型
* <pre>{@code
* // 枚举测试只能获取到枚举类型
* Class<Enum<?>> enumSupplierClass = LambdaUtil.getRealClass(LambdaUtil.LambdaKindEnum.REF_NONE::ordinal);
* Assert.assertEquals(Enum.class, enumSupplierClass);
* // 调用父类方法只能获取到父类类型
* Class<Entity<?>> superSupplierClass = LambdaUtil.getRealClass(myTeacher::getId);
* Assert.assertEquals(Entity.class, superSupplierClass);
* // 引用父类静态带参方法只能获取到父类类型
* Class<Entity<?>> staticSuperFunctionClass = LambdaUtil.getRealClass(MyTeacher::takeId);
* Assert.assertEquals(Entity.class, staticSuperFunctionClass);
* }</pre>
*
* @param func lambda
* @param <R> 类型
* @return lambda实现类
* @throws IllegalArgumentException 如果是不支持的方法引用抛出该异常{@link LambdaUtil#checkLambdaTypeCanGetClass}
* @since 5.8.0
*/
public static <R> Class<R> getRealClass(Func0<?> func) {
SerializedLambda lambda = resolve(func);
checkLambdaTypeCanGetClass(lambda.getImplMethodKind());
return ClassUtil.loadClass(lambda.getImplClass());
}
/**
* 解析lambda表达式,加了缓存
* 该缓存可能会在任意不定的时间被清除
@ -68,72 +120,57 @@ public class LambdaUtil {
}
/**
* 通过对象的方法或类的静态方法引用获取lambda实现类两种情况匹配到此方法
* 通过对象的方法或类的静态方法引用然后根据{@link SerializedLambda#getInstantiatedMethodType()}获取lambda实现类<br>
* 传入lambda有参数且含有返回值的情况能够匹配到此方法
* <pre>{@code
* @Data
* @EqualsAndHashCode(callSuper = true)
* static class MyTeacher extends Entity<MyTeacher> {
*
* public String age;
*
* public static String takeAgeBy(MyTeacher myTeacher) {
* return myTeacher.getAge();
* }
*
* }
* }</pre>
* <ul>
* <li>对象方法引用myTeacher::getAge</li>
* <li>类静态方法引用MyTeacher::takeAge</li>
* <li>引用特定类型的任意对象的实例方法<pre>{@code
* Class<MyTeacher> functionClass = LambdaUtil.getRealClass(MyTeacher::getAge);
* Assert.assertEquals(MyTeacher.class, functionClass);
* }</pre></li>
* <li>引用静态带参方法<pre>{@code
* Class<MyTeacher> staticFunctionClass = LambdaUtil.getRealClass(MyTeacher::takeAgeBy);
* Assert.assertEquals(MyTeacher.class, staticFunctionClass);
* }</pre></li>
* </ul>
* 如想获取调用的方法引用所在类可以
* <pre>
* // 返回MyTeacher.class
* LambdaUtil.getImplClass(myTeacher::getAge);
* </pre>
*
* @param func lambda
* @param <R> 类型
* @param <P> 方法调用方类型
* @param <R> 返回值类型
* @return lambda实现类
* @throws IllegalArgumentException 如果是不支持的方法引用抛出该异常{@link LambdaUtil#checkLambdaTypeCanGetClass}
* @since 5.8.0
*/
public static <R> Class<R> getImplClass(Func0<?> func) {
return ClassUtil.loadClass(resolve(func).getImplClass());
}
/**
* 通过类的方法引用获取lambda实现类<br>
* 类方法引用相当于获取的方法引用是MyTeacher.getAge(this)
* 如想获取调用的方法引用所在类可以
* <pre>
* // 返回MyTeacher.class
* LambdaUtil.getImplClass(MyTeacher::getAge);
* </pre>
*
* @param func lambda
* @param <T> 类型
* @return lambda实现类
* @since 5.8.0
*/
public static <T> Class<T> getImplClass(Func1<T, ?> func) {
return ClassUtil.loadClass(resolve(func).getImplClass());
}
/**
* 通过{@link SerializedLambda#getInstantiatedMethodType()}获取lambda实现类<br>
* 在使用{@link #getImplClass(Func0)}获取实现类的时候如果传入的是父类方法引用会返回父类导致问题<br>
* 此类通过方法的名称截取出类名
*
* @param func lambda
* @param <P> 类型
* @return lambda实现类
* @since 5.8.0
*/
public static <P> Class<P> getInstantiatedClass(Func0<?> func) {
final String instantiatedMethodType = resolve(func).getInstantiatedMethodType();
public static <P, R> Class<P> getRealClass(Func1<P, R> func) {
SerializedLambda lambda = resolve(func);
checkLambdaTypeCanGetClass(lambda.getImplMethodKind());
String instantiatedMethodType = lambda.getInstantiatedMethodType();
return ClassUtil.loadClass(StrUtil.sub(instantiatedMethodType, 2, StrUtil.indexOf(instantiatedMethodType, ';')));
}
/**
* 通过{@link SerializedLambda#getInstantiatedMethodType()}获取lambda实现类<br>
* 在使用{@link #getImplClass(Func1)}获取实现类的时候如果传入的是父类方法引用会返回父类导致问题<br>
* 此类通过方法的名称截取出类名
* 检查是否为支持的类型
*
* @param func lambda
* @param <P> 类型
* @return lambda实现类
* @since 5.8.0
* @param implMethodKind 支持的lambda类型
* @throws IllegalArgumentException 如果是不支持的方法引用抛出该异常
*/
public static <P> Class<P> getInstantiatedClass(Func1<P, ?> func) {
final String instantiatedMethodType = resolve(func).getInstantiatedMethodType();
return ClassUtil.loadClass(StrUtil.sub(instantiatedMethodType, 2, StrUtil.indexOf(instantiatedMethodType, ';')));
private static void checkLambdaTypeCanGetClass(int implMethodKind) {
if (implMethodKind != LambdaKindEnum.REF_invokeVirtual.ordinal() &&
implMethodKind != LambdaKindEnum.REF_invokeStatic.ordinal()) {
throw new IllegalArgumentException("该lambda不是合适的方法引用");
}
}
/**
@ -184,4 +221,20 @@ public class LambdaUtil {
private static SerializedLambda _resolve(Serializable func) {
return cache.get(func.getClass().getName(), () -> ReflectUtil.invoke(func, "writeReplace"));
}
/**
* Lambda类型枚举
*/
public enum LambdaKindEnum {
REF_NONE,
REF_getField,
REF_getStatic,
REF_putField,
REF_putStatic,
REF_invokeVirtual,
REF_invokeStatic,
REF_invokeSpecial,
REF_newInvokeSpecial,
}
}

View File

@ -1,10 +1,5 @@
package cn.hutool.core.util;
import cn.hutool.core.lang.Assert;
import cn.hutool.core.lang.func.Func1;
import cn.hutool.core.lang.func.LambdaUtil;
import cn.hutool.core.map.MapUtil;
import java.lang.reflect.Field;
import java.util.ArrayList;
import java.util.Arrays;
@ -14,6 +9,11 @@ import java.util.Map;
import java.util.function.Function;
import java.util.function.Predicate;
import cn.hutool.core.lang.Assert;
import cn.hutool.core.lang.func.Func1;
import cn.hutool.core.lang.func.LambdaUtil;
import cn.hutool.core.map.MapUtil;
/**
* 枚举工具类
*
@ -239,9 +239,9 @@ public class EnumUtil {
* @return 对应枚举 获取不到时为 {@code null}
*/
public static <E extends Enum<E>, C> E getBy(Func1<E, C> condition, C value) {
Class<E> implClass = LambdaUtil.getImplClass(condition);
Class<E> implClass = LambdaUtil.getRealClass(condition);
if (Enum.class.equals(implClass)) {
implClass = LambdaUtil.getInstantiatedClass(condition);
implClass = LambdaUtil.getRealClass(condition);
}
return Arrays.stream(implClass.getEnumConstants()).filter(e -> condition.callWithRuntimeException(e).equals(value)).findAny().orElse(null);
}
@ -260,9 +260,9 @@ public class EnumUtil {
*/
public static <E extends Enum<E>, F, C> F getFieldBy(Func1<E, F> field,
Function<E, C> condition, C value) {
Class<E> implClass = LambdaUtil.getImplClass(field);
Class<E> implClass = LambdaUtil.getRealClass(field);
if (Enum.class.equals(implClass)) {
implClass = LambdaUtil.getInstantiatedClass(field);
implClass = LambdaUtil.getRealClass(field);
}
return Arrays.stream(implClass.getEnumConstants())
// 过滤

View File

@ -1,9 +1,12 @@
package cn.hutool.core.lang.func;
import lombok.Data;
import org.junit.Assert;
import org.junit.Test;
import lombok.AllArgsConstructor;
import lombok.Data;
import lombok.EqualsAndHashCode;
public class LambdaUtilTest {
@Test
@ -19,38 +22,98 @@ public class LambdaUtilTest {
}
@Test
public void getImplClassTest() {
// 类方法引用相当于获取的方法引用是MyTeacher.getAge(this)
// 因此此处会匹配到Func1其参数就是this
Class<MyTeacher> aClass = LambdaUtil.getImplClass(MyTeacher::getAge);
Assert.assertEquals(MyTeacher.class, aClass);
MyTeacher myTeacher = new MyTeacher();
// 对象方法引用因为已经有了对象因此此处引用相当于获取myTeacher.getAge()
aClass = LambdaUtil.getImplClass(myTeacher::getAge);
Assert.assertEquals(MyTeacher.class, aClass);
// 静态方法引用相当于获取MyTeader.takeAge
aClass = LambdaUtil.getImplClass(MyTeacher::takeAge);
Assert.assertEquals(MyTeacher.class, aClass);
public void resolveTest() {
// 引用构造函数
Assert.assertEquals(LambdaUtil.LambdaKindEnum.REF_newInvokeSpecial.ordinal(),
LambdaUtil.resolve(MyTeacher::new).getImplMethodKind());
// 数组构造函数引用
Assert.assertEquals(LambdaUtil.LambdaKindEnum.REF_invokeStatic.ordinal(),
LambdaUtil.resolve(MyTeacher[]::new).getImplMethodKind());
// 引用静态方法
Assert.assertEquals(LambdaUtil.LambdaKindEnum.REF_invokeStatic.ordinal(),
LambdaUtil.resolve(MyTeacher::takeAge).getImplMethodKind());
// 引用特定对象的实例方法
Assert.assertEquals(LambdaUtil.LambdaKindEnum.REF_invokeVirtual.ordinal(),
LambdaUtil.resolve(new MyTeacher()::getAge).getImplMethodKind());
// 引用特定类型的任意对象的实例方法
Assert.assertEquals(LambdaUtil.LambdaKindEnum.REF_invokeVirtual.ordinal(),
LambdaUtil.resolve(MyTeacher::getAge).getImplMethodKind());
}
@Test
public void getInstantiatedClassTest() {
// 类方法引用相当于获取的方法引用是MyTeacher.getAge(this)
// 因此此处会匹配到Func1其参数就是this
Class<MyTeacher> aClass = LambdaUtil.getInstantiatedClass(MyTeacher::getAge);
Assert.assertEquals(MyTeacher.class, aClass);
public void getRealClassTest() {
// 引用特定类型的任意对象的实例方法
Class<MyTeacher> functionClass = LambdaUtil.getRealClass(MyTeacher::getAge);
Assert.assertEquals(MyTeacher.class, functionClass);
// 枚举测试不会导致类型擦除
Class<LambdaUtil.LambdaKindEnum> enumFunctionClass = LambdaUtil.getRealClass(LambdaUtil.LambdaKindEnum::ordinal);
Assert.assertEquals(LambdaUtil.LambdaKindEnum.class, enumFunctionClass);
// 调用父类方法能获取到正确的子类类型
Class<MyTeacher> superFunctionClass = LambdaUtil.getRealClass(MyTeacher::getId);
Assert.assertEquals(MyTeacher.class, superFunctionClass);
MyTeacher myTeacher = new MyTeacher();
// 引用特定对象的实例方法
Class<MyTeacher> supplierClass = LambdaUtil.getRealClass(myTeacher::getAge);
Assert.assertEquals(MyTeacher.class, supplierClass);
// 枚举测试只能获取到枚举类型
Class<Enum<?>> enumSupplierClass = LambdaUtil.getRealClass(LambdaUtil.LambdaKindEnum.REF_NONE::ordinal);
Assert.assertEquals(Enum.class, enumSupplierClass);
// 调用父类方法只能获取到父类类型
Class<Entity<?>> superSupplierClass = LambdaUtil.getRealClass(myTeacher::getId);
Assert.assertEquals(Entity.class, superSupplierClass);
// 引用静态带参方法能够获取到正确的参数类型
Class<MyTeacher> staticFunctionClass = LambdaUtil.getRealClass(MyTeacher::takeAgeBy);
Assert.assertEquals(MyTeacher.class, staticFunctionClass);
// 引用父类静态带参方法只能获取到父类类型
Class<Entity<?>> staticSuperFunctionClass = LambdaUtil.getRealClass(MyTeacher::takeId);
Assert.assertEquals(Entity.class, staticSuperFunctionClass);
// 引用静态无参方法能够获取到正确的类型
Class<MyTeacher> staticSupplierClass = LambdaUtil.getRealClass(MyTeacher::takeAge);
Assert.assertEquals(MyTeacher.class, staticSupplierClass);
// 引用父类静态无参方法能够获取到正确的参数类型
Class<MyTeacher> staticSuperSupplierClass = LambdaUtil.getRealClass(MyTeacher::takeIdBy);
Assert.assertEquals(MyTeacher.class, staticSuperSupplierClass);
}
@Data
static class MyTeacher {
@AllArgsConstructor
static class MyStudent {
public static String takeAge(){
private String name;
}
@Data
public static class Entity<T> {
private T id;
public static <T> T takeId() {
return new Entity<T>().getId();
}
public static <T> T takeIdBy(Entity<T> entity) {
return entity.getId();
}
}
@Data
@EqualsAndHashCode(callSuper = true)
static class MyTeacher extends Entity<MyTeacher> {
public static String takeAge() {
return new MyTeacher().getAge();
}
public static String takeAgeBy(MyTeacher myTeacher) {
return myTeacher.getAge();
}
public String age;
}
}