基于Mybatis-Plus进行查询封装

基于Mybatis-Plus进行查询封装

package com.test.common.orm.builder;import cn.hutool.core.util.IdUtil;
import com.baomidou.mybatisplus.core.conditions.query.QueryWrapper;
import org.apache.ibatis.cache.Cache;
import org.apache.ibatis.cache.impl.PerpetualCache;
import org.springframework.util.Assert;
import org.springframework.util.ReflectionUtils;
import org.springframework.util.StringUtils;import java.lang.annotation.*;
import java.lang.reflect.Field;
import java.lang.reflect.Method;
import java.util.*;
import java.util.concurrent.atomic.AtomicInteger;
import java.util.stream.Stream;public final class QueryWrapperBuilder {private static final Cache FILED_CACHE = new PerpetualCache(IdUtil.simpleUUID());private static final Cache METHOD_CACHE = new PerpetualCache(IdUtil.simpleUUID());private static final char[] UPPER_CASE_CHAR_ARRAY = new char[26];static {UPPER_CASE_CHAR_ARRAY[0]='A';UPPER_CASE_CHAR_ARRAY[1]='B';UPPER_CASE_CHAR_ARRAY[2]='C';UPPER_CASE_CHAR_ARRAY[3]='D';UPPER_CASE_CHAR_ARRAY[4]='E';UPPER_CASE_CHAR_ARRAY[5]='F';UPPER_CASE_CHAR_ARRAY[6]='G';UPPER_CASE_CHAR_ARRAY[7]='H';UPPER_CASE_CHAR_ARRAY[8]='I';UPPER_CASE_CHAR_ARRAY[9]='J';UPPER_CASE_CHAR_ARRAY[10]='K';UPPER_CASE_CHAR_ARRAY[11]='L';UPPER_CASE_CHAR_ARRAY[12]='M';UPPER_CASE_CHAR_ARRAY[13]='N';UPPER_CASE_CHAR_ARRAY[14]='O';UPPER_CASE_CHAR_ARRAY[15]='P';UPPER_CASE_CHAR_ARRAY[16]='Q';UPPER_CASE_CHAR_ARRAY[17]='R';UPPER_CASE_CHAR_ARRAY[18]='S';UPPER_CASE_CHAR_ARRAY[19]='T';UPPER_CASE_CHAR_ARRAY[20]='U';UPPER_CASE_CHAR_ARRAY[21]='V';UPPER_CASE_CHAR_ARRAY[22]='W';UPPER_CASE_CHAR_ARRAY[23]='X';UPPER_CASE_CHAR_ARRAY[24]='Y';UPPER_CASE_CHAR_ARRAY[25]='Z';}public static <T>QueryWrapper<T> buildListWrapper(final Class<T> entityClass, final Object voObject, Integer... group) {Assert.isTrue(null != entityClass,"entityClass cannot be null");Assert.isTrue(null != voObject,"voObject cannot be null");if(group != null && group.length > 1) {throw new IllegalArgumentException("group cannot greater than one");}AtomicInteger defaultGroup = new AtomicInteger(-1);if(null == group || group.length <= 0) {defaultGroup.set(1);} else {defaultGroup.set(group[0]);}QueryWrapper<T> queryWrapper = new QueryWrapper<>();Class<?> voObjectClass = voObject.getClass();String voClassName = voObjectClass.getName();Field[] fields = (Field[]) FILED_CACHE.getObject(voClassName);if(null == fields) {fields = voObjectClass.getDeclaredFields();if(fields.length > 0) {FILED_CACHE.putObject(voClassName, fields);} else {throw new IllegalArgumentException("voObject [" + voClassName + "] not has fields");}}Stream<Field> stream = Arrays.stream(fields);stream.forEach(field -> {if(field.isAnnotationPresent(QueryOperation.class) || field.isAnnotationPresent(QueryOperations.class)) {String fieldName = field.getName();String columnName = fieldToColumnName(fieldName);Annotation[] annotations = field.getAnnotations();List<QueryOperation> list = new ArrayList<>();for(Annotation annotation : annotations) {if(annotation instanceof QueryOperations) {QueryOperation[] queryOperations = ((QueryOperations) annotation).value();list.addAll(Arrays.asList(queryOperations));} else if(annotation instanceof QueryOperation){list.add((QueryOperation) annotation);}}QueryOperation queryOperation =list.stream().filter(annotation ->annotation.group() == defaultGroup.get()).findFirst().orElse(null);if(null != queryOperation) {String annoColumnName = queryOperation.columnName();if(StringUtils.hasText(annoColumnName)) {columnName = annoColumnName;}QueryOperationType queryOperationType = queryOperation.value();String methodName = fileNameToMethodName(fieldName,"get");String key = voClassName + "@" + methodName;Method method = (Method) METHOD_CACHE.getObject(key);if(null == method) {method = ReflectionUtils.findMethod(voObjectClass, methodName);METHOD_CACHE.putObject(key,method);}Assert.notNull(method,"method : [ " + methodName + " ] not exists in class : [ " + voObjectClass.getName() + "]");Object o = ReflectionUtils.invokeMethod(method, voObject);setQueryWrapper(queryWrapper,queryOperationType,columnName,o);}}});return queryWrapper;}public static String fieldToColumnName(String fieldName) {Assert.isTrue(StringUtils.hasText(fieldName), "fieldName must contains at least one character");char[] chars = fieldName.toCharArray();List<Integer> indexList = new ArrayList<>();for(int i=0;i<chars.length;i++) {char val = chars[i];for(char charVal : UPPER_CASE_CHAR_ARRAY) {if(val == charVal) {indexList.add(i);break;}}}fieldName = fieldName.toLowerCase(Locale.ENGLISH);StringBuilder columnName = new StringBuilder(fieldName);indexList.forEach(index->{columnName.insert(index,"_");});return columnName.toString();}public static String fileNameToMethodName(final String fieldName,String prefix) {return prefix + fieldName.substring(0,1).toUpperCase(Locale.ENGLISH) + fieldName.substring(1);}private static void setQueryWrapper(QueryWrapper<?> queryWrapper, QueryOperationType queryOperationType,String columnName, Object o) {boolean canSet = false;if(o instanceof String) {canSet = StringUtils.hasLength((CharSequence)o) ;} else {canSet = null != o ;}if(canSet) {if(queryOperationType == QueryOperationType.EQUAL) {queryWrapper.eq(columnName,o);} else if(queryOperationType == QueryOperationType.NOT_EQUAL) {queryWrapper.ne(columnName,o);} else if(queryOperationType == QueryOperationType.LIKE) {queryWrapper.like(columnName,o);} else if(queryOperationType == QueryOperationType.BETWEEN) {if(!(o instanceof Collection<?>)) {throw new UnsupportedOperationException("只对集合支持between操作");}if(((Collection<?>) o).size() < 2) {throw new IllegalArgumentException("集合中元素的数量异常");}Object[] objArr = ((Collection<?>) o).toArray();queryWrapper.between(columnName, objArr[0], objArr[1]);} else if(queryOperationType == QueryOperationType.ORDER_ASC) {queryWrapper.orderByAsc(columnName);} else if(queryOperationType == QueryOperationType.ORDER_DESC) {queryWrapper.orderByDesc(columnName);}}}/*** 执行查询的类型* */@Target(ElementType.FIELD)@Retention(RetentionPolicy.RUNTIME)@Repeatable(QueryOperations.class)public @interface QueryOperation {QueryOperationType value() default QueryOperationType.EQUAL;int group() default 1;String columnName() default "";}/*** 设置可以使用重复注解* */@Target(ElementType.FIELD)@Retention(RetentionPolicy.RUNTIME)public @interface QueryOperations {QueryOperation[] value();}public enum QueryOperationType {EQUAL("equal"),NOT_EQUAL("not_equal"),LIKE("like"),BETWEEN("between"),ORDER_DESC("order_desc"),ORDER_ASC("order_asc");private final String operationType;QueryOperationType(String operationType) {this.operationType = operationType;}}}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/240649.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

前端 JS 安全对抗原理与实践

作者&#xff1a;vivo 互联网安全团队- Luo Bingsong 前端代码都是公开的&#xff0c;为了提高代码的破解成本、保证JS代码里的一些重要逻辑不被居心叵测的人利用&#xff0c;需要使用一些加密和混淆的防护手段。 一、概念解析 1.1 什么是接口加密 如今这个时代&#xff0c;…

高德地图逆地理编码踩坑日志

本人是一枚Java小白&#xff0c;公司项目中用到根据经纬度反查该地址中文信息的场景&#xff0c;因为一开始调用的经纬度是能反查出区域编码的&#xff0c;以为towncode都是String返回结果&#xff0c;如下图&#xff1a; 没想到当没有名字任何一个城市区域的时候&#xff0c;…

管理 Jenkins 详细指南

目录 系统配置 安全 状态信息 故障 排除 工具和操作 系统配置 系统&#xff0c;配置全局设置和路径&#xff0c;端口更改&#xff0c;下载地址等。 工具&#xff0c;配置工具、其位置和自动安装程序。 插件&#xff0c;添加、删除、禁用或启用可以扩展 Jenkins 功能的插…

华为OD机试 - 精准核酸检测(Java JS Python C)

题目描述 为了达到新冠疫情精准防控的需要,为了避免全员核酸检测带来的浪费,需要精准圈定可能被感染的人群。 现在根据传染病流调以及大数据分析,得到了每个人之间在时间、空间上是否存在轨迹交叉。 现在给定一组确诊人员编号(X1,X2,X3,...,Xn),在所有人当中,找出哪些…

ssh远程管理服务

什么是ssh SSH是一种加密的网络协议&#xff0c;用于在不安全的网络中安全地传输数据。它允许用户通过一个安全的通道连接到远程计算机&#xff0c;并在该通道上执行各种网络服务&#xff0c;例如远程登录和文件传输。 SSH使用公钥加密技术来验证远程计算机的身份&#xff0c;并…

初识Stable Diffusion

界面选项解读 这是在趋动云上部署的Stable Diffusion txt2img prompt &#xff08;1&#xff09;分割符号&#xff1a;使用逗号 , 用于分割词缀&#xff0c;且有一定权重排序功能&#xff0c;逗号前权重高&#xff0c;逗号后权重低 &#xff08;2&#xff09;建议的通用范式…

【C++11特性篇】玩转C++11中的包装器(function&bind)

前言 大家好吖&#xff0c;欢迎来到 YY 滴C系列 &#xff0c;热烈欢迎&#xff01; 本章主要内容面向接触过C的老铁 主要内容含&#xff1a; 欢迎订阅 YY滴C专栏&#xff01;更多干货持续更新&#xff01;以下是传送门&#xff01; 目录 一.为什么需要包装器function&#xff…

【Earth Engine】协同Sentinel-1/2使用随机森林回归实现高分辨率相对财富(贫困)制图

目录 1 简介与摘要2 思路3 效果预览4 代码思路5 完整代码6 后记 1 简介与摘要 最近在做一些课题&#xff0c;需要使用Sentinel-1/2进行机器学习制图。 然后想着总结一下相关数据和方法&#xff0c;就花半小时写了个代码。 然后再花半小时写下这篇博客记录一下。 因为基于多次拍…

通过windows cng api 实现rsa非对称加密

参考&#xff1a; 1,使用 CNG 加密数据 - Win32 apps | Microsoft Learn 2,不记得了 &#xff08;下文通过cng api演示rsa加密&#xff0c;不做原理性介绍&#xff09; 相对于aes等对称加密算法&#xff0c;rsa加密算法不可逆性更强。非对称加密在通常情况下&#xff0c;使…

前端传输formDate格式的数据,后端不能用@RequestBody接收

写了个接口&#xff0c;跟前端对接&#xff0c;前端说怎么一直415的报错 我寻思不对啊&#xff0c;我swagger都请求成功了&#xff0c;后来发现前端一直是以formdata格式提交的数据&#xff0c;这样我其实是可以不加RequestBody的&#xff1b; 知识点&#xff1a; RequestBody…

类和对象

1 类定义&#xff1a; class ChecksumAccumulator {// class definition goes here } 你就能创建 ChecksumAccumulator 对象&#xff1a;new CheckSumAccumulator 注&#xff1a;1scala类中成员默认是public类型&#xff0c;若设为私有属性则必须加private关键字。在scala中是…

基于Springboot的留守儿童爱心网站(有报告)。Javaee项目,springboot项目。

演示视频&#xff1a; 基于Springboot的留守儿童爱心网站(有报告)。Javaee项目&#xff0c;springboot项目。 项目介绍&#xff1a; 采用M&#xff08;model&#xff09;V&#xff08;view&#xff09;C&#xff08;controller&#xff09;三层体系结构&#xff0c;通过Spring…

技术交底二维码的应用

二维码技术交底可以逐级落实、责任到人、有据可查、是目前最方便、实用的交底方式&#xff0c;下面我们讲解技术交底二维码的应用。 1、生成对应的技术交底二维码&#xff0c;将施工方案、技术资料、安全教育资料等内容上传到二维码里。打印出来现场粘贴&#xff0c;便于作业班…

微信小程序 上列表拉加载下拉刷新

上拉加载和下拉刷新是小程序开发的常见需求。本文将介绍如何在微信小程序中实现上拉加载和下拉刷新的功能&#xff0c;为用户带来更加流畅、便捷的使用体验。 1. 实现效果 微信小程序 上列表拉加载下拉刷新 2. 实现思路 (1) 首先需要在使用到的 json 文件下配置 “enablePull…

java中线程相关的面试题

什么是线程安全&#xff0c;造成线程安全的本质是什么&#xff1f; 什么是线程安全呢&#xff1f; 咱们初步去理解话记住一句话就行&#xff1a;如果一个对象可以安全地被多个线程同时使用&#xff0c;那它就是线程安全的。 为什么并发编程会导致线程不安全&#xff1f; 可见…

用友U8CRM系统help2 任意文件读取漏洞复现

用友U8CRM系统的help2文件中接口存在任意文件读取漏洞&#xff0c;攻击者在未登录情况下即可进行漏洞利用。 1.1 漏洞级别 高危 1.2 快速检索 fofa语法&#xff1a; title"用友U8CRM"1.3 漏洞复现 该漏洞利用非常简单&#xff0c;只需构造get请求 访问该地址即可…

密码学 Mod37 校验算法(Java和C#) ISO/IEC 7064 modulo 37-2

简介 ISO/IEC 7064:2003 信息技术 安全技术 校验字符系统 密码学 Mod37 校验算法(Java和C#) ISO/IEC 7064 modulo 37-2 适用范围 1.1 本标准规定了一组校验字符系统&#xff0c;它可以防止在复制或键人数据时产生的串的错误。串的长度 可以是固定的或是可变的&#xff0c;包…

青少年CTF-qsnctf-A1-Misc-签到

题目环境&#xff1a; 题目难度&#xff1a;★题目描述&#xff1a;有没有可能&#xff0c;这个平台就是个题目&#xff1f; 一道杂项题 题目说的是这个平台就是题目 那么也就是说flag就在这个平台里面1.从高层次向低层次逐一排查 2.首先对平台首页进行排查进平台首页 第一种解…

HTML---浮动

提示&#xff1a;文章写完后&#xff0c;目录可以自动生成&#xff0c;如何生成可参考右边的帮助文档 文章目录 前言一、pandas是什么&#xff1f;二、使用步骤 1.引入库2.读入数据总结 一.常见的网页布局 二.标准文档流 标准文档流常见标签 标准文档流的组成 块级元素<div…

二叉搜索树 --- C++实现

目录 1.二叉搜索树的概念 2.二叉搜索树的操作 3. 二叉树的实现 4.二叉搜索树的应用 5. 二叉树的性能分析 6. 二叉树进阶练习题 1.二叉搜索树的概念 二叉搜索树又称二叉排序树&#xff0c;它或者是一棵空树&#xff0c;或者是具有以下性质的二叉树&#xff1a; 若它的左…