java如何根据属性名这个字符串创建SFunction ---结合了我自己的项目来说的

问题描述

根据SFunction获取fieldname网上有很多,但怎么根据fieldname生成SFunction的方法呢,很难找得到。

比如我有一个Person类 里面有name属性 我怎么根据name这个字符串和Person这个class 获取SFunction

如果想直接看解决方案的话,可以直接看我的这篇文章
java如何根据属性名这个字符串创建SFunction(推荐)

遇到的问题

在我的做项目的过程中,有一个需求是根据查询条件获取满足条件的图表(Chart图表实体类)。
我写了一个专门构建LambdaQueryWrapper的方法getChartLambdaQueryWrapper(Chart
QueryRequest chart
QueryRequest)。
ChartQueryRequest类:包含了Chart类中所有可以作为查询条件的字段,同时继承了


PageRequest 类:

@Data
public class PageRequest {

    /**
     * 当前页号
     */
    private int currentPage = 1;

    /**
     * 页面大小
     */
    private int pageSize = 10;

    /**
     * 排序字段
     */
    private String sortField;

    /**
     * 排序顺序(默认升序)
     */
    private String sortOrder = CommonConstant.SORT_ORDER_ASC;
}

ChartQueryRequest 类:

/**
 * chart图表查询参数
 */
@Data
public class ChartQueryRequest extends PageRequest implements Serializable {

    private static final long serialVersionUID = 8878326647736641880L;

    /**
     * id
     */
    private Long id;

    /**
     * 图表名称
     */
    private String name;

    /**
     * 图表类型
     */
    private String chartType;

    /**
     * 创建用户id
     */
    private Long userId;

    /**
     * 创建时间
     */
    @DateTimeFormat(pattern = "yyyy-MM-dd HH:mm:ss")
    private LocalDateTime createTime;

    /**
     * 更新时间
     */
    @DateTimeFormat(pattern = "yyyy-MM-dd HH:mm:ss")
    private LocalDateTime updateTime;
}

前端发送请求,springboot将查询条件封装成ChartQueryRequest 对象,假设,我要根据createTime这个字段进行排序,即sortField = “createTime”
在构建查询时,ChartQueryRequest 对象有排序字段sortField,但是Chart类中没有。所有我无法通过Chart::getId(这种Lambda表达式)方式进行构建,也不能通过下面这种方式构建
在这里插入图片描述
因为chartLambdaQueryWrapper的类型是LambdaQueryWrapper,在构建时,Lambda表达式不能使用ChartQueryRequest的(也就是不能使用ChartQueryRequest::getSortField)
但是我又不想使用QueryWrapper来构建(因为这种方式不灵活,这个方式把代码写死了,实体类字段改变时,QueryWrapper的构建代码也需要改)。

所以这个时候,我们只知道排序的字段名(也就是sortField的值),如何根据这个字段名创建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;
    }

使用:

String sortField = "createTime"; 
//Chart类中必须包含createTime字段.
SFunction sFunction = getSFunction(Chart.class, sortField);
  • 15
    点赞
  • 11
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值