java如何根据属性名这个字符串创建SFunction?(mybatis plus 中的LambdaQueryWrapper使用到的SFunction)
问题描述
比如我有一个Person类 里面有name属性 我怎么根据name这个字符串和Person这个class 获取SFunction
解决方案:
我搜了好久才搜到一个与我这个问题相关的,链接如下:
可以根据fieldname生成SFunction吗
这篇文章只给出了部分代码,让我参考着改,然后我就看了他给出的代码,自己写了一份完整的
以下是完整的代码:
/**
* 获取与实体类字段对应的 SFunction 对象。
* @param entityClass 实体类的 Class 对象。
* @param fieldName 实体类中的字段名。
* @return 返回找到的 SFunction 对象。
*/
public static SFunction getSFunction(Class<?> entityClass, String fieldName) {
// 检查缓存中是否已经有了对应的 SFunction 对象。
if (functionMap.containsKey(entityClass.getName() + fieldName)) {
return functionMap.get(entityClass.getName() + fieldName);
}
// 获取实体类中名为 fieldName 的字段。
Field field = getDeclaredField(entityClass, fieldName);
if (field == null) {
//如果字段不存在,使用 ExceptionUtils 抛出一个异常,指出实体类中没有找到该字段。
throw ExceptionUtils.mpe("This class %s is not have field %s ", entityClass.getName(), fieldName);
}
SFunction func = null;
// 获取 MethodHandles.Lookup 实例,用于反射操作。
final MethodHandles.Lookup lookup = MethodHandles.lookup();
// 定义方法类型,表示实体类的实例方法,该方法返回字段的类型。
MethodType methodType = MethodType.methodType(field.getType(), entityClass);
// 用于存储 LambdaMetafactory 创建的 CallSite 对象。
final CallSite site;
// 构造标准的 Java getter 方法名。
String getFunName = "get" + fieldName.substring(0, 1).toUpperCase() + fieldName.substring(1);
try {
// 使用 LambdaMetafactory 创建一个动态的 SFunction 实例。
site = LambdaMetafactory.altMetafactory(
lookup,
"invoke",
MethodType.methodType(SFunction.class),
methodType,
lookup.findVirtual(entityClass, getFunName, MethodType.methodType(field.getType())),
methodType,
FLAG_SERIALIZABLE
);
// 使用 CallSite 来获取 SFunction 实例。
func = (SFunction) site.getTarget().invokeExact();
// 将生成的 SFunction 实例存储到缓存中。
functionMap.put(entityClass.getName() + field.getName(), func);
return func;
} catch (Throwable e) {
// 如果在创建 SFunction 过程中发生异常,抛出异常,指出实体类中没有找到对应的 getter 方法。
throw ExceptionUtils.mpe("This class %s is not have method %s ", entityClass.getName(), getFunName);
}
}
/**
* 递归获取类中声明的字段,包括私有字段。
* @param clazz 要检查的类。
* @param fieldName 要查找的字段名。
* @return 返回找到的 Field 对象,如果没有找到则返回 null。
*/
public static Field getDeclaredField(Class<?> clazz, String fieldName) {
Field field = null;
// 遍历类及其父类,直到到达 Object 类。
for (; clazz != Object.class; clazz = clazz.getSuperclass()) {
try {
// 尝试获取声明的字段。
field = clazz.getDeclaredField(fieldName);
// 如果找到字段,返回该字段。
return field;
} catch (NoSuchFieldException e) {
// 如果没有找到字段,继续查找父类。
// 这里不处理异常,让其继续执行循环。
}
}
// 如果没有找到字段,返回 null。
return null;
}
使用:
//Person类中必须包含"name"字段.
SFunction sFunction = getSFunction(Person.class, "name");