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

2024-05-05 06:52

本文主要是介绍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);

这篇关于java如何根据属性名这个字符串创建SFunction ---结合了我自己的项目来说的的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/961052

相关文章

Spring事务传播机制最佳实践

《Spring事务传播机制最佳实践》Spring的事务传播机制为我们提供了优雅的解决方案,本文将带您深入理解这一机制,掌握不同场景下的最佳实践,感兴趣的朋友一起看看吧... 目录1. 什么是事务传播行为2. Spring支持的七种事务传播行为2.1 REQUIRED(默认)2.2 SUPPORTS2

怎样通过分析GC日志来定位Java进程的内存问题

《怎样通过分析GC日志来定位Java进程的内存问题》:本文主要介绍怎样通过分析GC日志来定位Java进程的内存问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录一、GC 日志基础配置1. 启用详细 GC 日志2. 不同收集器的日志格式二、关键指标与分析维度1.

Java进程异常故障定位及排查过程

《Java进程异常故障定位及排查过程》:本文主要介绍Java进程异常故障定位及排查过程,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录一、故障发现与初步判断1. 监控系统告警2. 日志初步分析二、核心排查工具与步骤1. 进程状态检查2. CPU 飙升问题3. 内存

java中新生代和老生代的关系说明

《java中新生代和老生代的关系说明》:本文主要介绍java中新生代和老生代的关系说明,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录一、内存区域划分新生代老年代二、对象生命周期与晋升流程三、新生代与老年代的协作机制1. 跨代引用处理2. 动态年龄判定3. 空间分

MySQL查询JSON数组字段包含特定字符串的方法

《MySQL查询JSON数组字段包含特定字符串的方法》在MySQL数据库中,当某个字段存储的是JSON数组,需要查询数组中包含特定字符串的记录时传统的LIKE语句无法直接使用,下面小编就为大家介绍两种... 目录问题背景解决方案对比1. 精确匹配方案(推荐)2. 模糊匹配方案参数化查询示例使用场景建议性能优

Java设计模式---迭代器模式(Iterator)解读

《Java设计模式---迭代器模式(Iterator)解读》:本文主要介绍Java设计模式---迭代器模式(Iterator),具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,... 目录1、迭代器(Iterator)1.1、结构1.2、常用方法1.3、本质1、解耦集合与遍历逻辑2、统一

Java内存分配与JVM参数详解(推荐)

《Java内存分配与JVM参数详解(推荐)》本文详解JVM内存结构与参数调整,涵盖堆分代、元空间、GC选择及优化策略,帮助开发者提升性能、避免内存泄漏,本文给大家介绍Java内存分配与JVM参数详解,... 目录引言JVM内存结构JVM参数概述堆内存分配年轻代与老年代调整堆内存大小调整年轻代与老年代比例元空

深度解析Java DTO(最新推荐)

《深度解析JavaDTO(最新推荐)》DTO(DataTransferObject)是一种用于在不同层(如Controller层、Service层)之间传输数据的对象设计模式,其核心目的是封装数据,... 目录一、什么是DTO?DTO的核心特点:二、为什么需要DTO?(对比Entity)三、实际应用场景解析

Java 线程安全与 volatile与单例模式问题及解决方案

《Java线程安全与volatile与单例模式问题及解决方案》文章主要讲解线程安全问题的五个成因(调度随机、变量修改、非原子操作、内存可见性、指令重排序)及解决方案,强调使用volatile关键字... 目录什么是线程安全线程安全问题的产生与解决方案线程的调度是随机的多个线程对同一个变量进行修改线程的修改操

从原理到实战深入理解Java 断言assert

《从原理到实战深入理解Java断言assert》本文深入解析Java断言机制,涵盖语法、工作原理、启用方式及与异常的区别,推荐用于开发阶段的条件检查与状态验证,并强调生产环境应使用参数验证工具类替代... 目录深入理解 Java 断言(assert):从原理到实战引言:为什么需要断言?一、断言基础1.1 语