feat: 新需求功能代码提交,新增MPJLambdaWrapper查询。

This commit is contained in:
tianyongbao
2025-10-16 00:27:38 +08:00
parent f54859f62d
commit ab3508d6c4
19 changed files with 192 additions and 534 deletions

17
pom.xml
View File

@@ -24,7 +24,7 @@
<fastexcel.version>1.3.0</fastexcel.version> <fastexcel.version>1.3.0</fastexcel.version>
<velocity.version>2.3</velocity.version> <velocity.version>2.3</velocity.version>
<satoken.version>1.44.0</satoken.version> <satoken.version>1.44.0</satoken.version>
<mybatis-plus.version>3.5.14</mybatis-plus.version> <mybatis-plus.version>3.5.7</mybatis-plus.version>
<p6spy.version>3.9.1</p6spy.version> <p6spy.version>3.9.1</p6spy.version>
<hutool.version>5.8.40</hutool.version> <hutool.version>5.8.40</hutool.version>
<spring-boot-admin.version>3.5.3</spring-boot-admin.version> <spring-boot-admin.version>3.5.3</spring-boot-admin.version>
@@ -50,6 +50,9 @@
<!-- 工作流配置 --> <!-- 工作流配置 -->
<warm-flow.version>1.8.1</warm-flow.version> <warm-flow.version>1.8.1</warm-flow.version>
<!-- mybatis-plus-join -->
<mybatis-plus-join.version>1.5.2</mybatis-plus-join.version>
<!-- 插件版本 --> <!-- 插件版本 -->
<maven-jar-plugin.version>3.4.2</maven-jar-plugin.version> <maven-jar-plugin.version>3.4.2</maven-jar-plugin.version>
<maven-war-plugin.version>3.4.0</maven-war-plugin.version> <maven-war-plugin.version>3.4.0</maven-war-plugin.version>
@@ -201,12 +204,16 @@
<version>${mybatis-plus.version}</version> <version>${mybatis-plus.version}</version>
</dependency> </dependency>
<!-- <dependency>-->
<!-- <groupId>com.baomidou</groupId>-->
<!-- <artifactId>mybatis-plus-jsqlparser</artifactId>-->
<!-- <version>${mybatis-plus.version}</version>-->
<!-- </dependency>-->
<dependency> <dependency>
<groupId>com.baomidou</groupId> <groupId>com.github.yulichang</groupId>
<artifactId>mybatis-plus-jsqlparser</artifactId> <artifactId>mybatis-plus-join-boot-starter</artifactId>
<version>${mybatis-plus.version}</version> <version>${mybatis-plus-join.version}</version>
</dependency> </dependency>
<dependency> <dependency>
<groupId>com.baomidou</groupId> <groupId>com.baomidou</groupId>
<artifactId>mybatis-plus-annotation</artifactId> <artifactId>mybatis-plus-annotation</artifactId>

View File

@@ -37,10 +37,17 @@
<artifactId>mybatis-plus-spring-boot3-starter</artifactId> <artifactId>mybatis-plus-spring-boot3-starter</artifactId>
</dependency> </dependency>
<!-- <dependency>-->
<!-- <groupId>com.baomidou</groupId>-->
<!-- <artifactId>mybatis-plus-jsqlparser</artifactId>-->
<!-- </dependency>-->
<dependency> <dependency>
<groupId>com.baomidou</groupId> <groupId>com.github.yulichang</groupId>
<artifactId>mybatis-plus-jsqlparser</artifactId> <artifactId>mybatis-plus-join-boot-starter</artifactId>
</dependency> </dependency>
<!-- sql性能分析插件 --> <!-- sql性能分析插件 -->
<dependency> <dependency>
<groupId>p6spy</groupId> <groupId>p6spy</groupId>

View File

@@ -18,7 +18,9 @@ import java.io.Serializable;
import java.util.Collection; import java.util.Collection;
import java.util.List; import java.util.List;
import java.util.Map; import java.util.Map;
import java.util.Objects;
import java.util.function.Function; import java.util.function.Function;
import java.util.stream.Collectors;
/** /**
* 自定义 Mapper 接口, 实现 自定义扩展 * 自定义 Mapper 接口, 实现 自定义扩展
@@ -33,113 +35,78 @@ public interface BaseMapperPlus<T, V> extends BaseMapper<T> {
Log log = LogFactory.getLog(BaseMapperPlus.class); Log log = LogFactory.getLog(BaseMapperPlus.class);
/**
* 获取当前实例对象关联的泛型类型 V 的 Class 对象
*
* @return 返回当前实例对象关联的泛型类型 V 的 Class 对象
*/
default Class<V> currentVoClass() { default Class<V> currentVoClass() {
return (Class<V>) GenericTypeUtils.resolveTypeArguments(this.getClass(), BaseMapperPlus.class)[1]; return (Class<V>) GenericTypeUtils.resolveTypeArguments(this.getClass(), BaseMapperPlus.class)[1];
} }
/**
* 获取当前实例对象关联的泛型类型 T 的 Class 对象
*
* @return 返回当前实例对象关联的泛型类型 T 的 Class 对象
*/
default Class<T> currentModelClass() { default Class<T> currentModelClass() {
return (Class<T>) GenericTypeUtils.resolveTypeArguments(this.getClass(), BaseMapperPlus.class)[0]; return (Class<T>) GenericTypeUtils.resolveTypeArguments(this.getClass(), BaseMapperPlus.class)[0];
} }
/**
* 使用默认的查询条件查询并返回结果列表
*
* @return 返回查询结果的列表
*/
default List<T> selectList() { default List<T> selectList() {
return this.selectList(new QueryWrapper<>()); return this.selectList(new QueryWrapper<>());
} }
/** /**
* 批量插入实体对象集合 * 批量插入
*
* @param entityList 实体对象集合
* @return 插入操作是否成功的布尔值
*/ */
default boolean insertBatch(Collection<T> entityList) { default boolean insertBatch(Collection<T> entityList) {
return Db.saveBatch(entityList); Db.saveBatch(entityList);
// 临时解决 新版本 mp 插入状态判断错误问题
return true;
} }
/** /**
* 批量根据ID更新实体对象集合 * 批量更新
*
* @param entityList 实体对象集合
* @return 更新操作是否成功的布尔值
*/ */
default boolean updateBatchById(Collection<T> entityList) { default boolean updateBatchById(Collection<T> entityList) {
return Db.updateBatchById(entityList); Db.updateBatchById(entityList);
// 临时解决 新版本 mp 插入状态判断错误问题
return true;
} }
/** /**
* 批量插入或更新实体对象集合 * 批量插入或更新
*
* @param entityList 实体对象集合
* @return 插入或更新操作是否成功的布尔值
*/ */
default boolean insertOrUpdateBatch(Collection<T> entityList) { default boolean insertOrUpdateBatch(Collection<T> entityList) {
return Db.saveOrUpdateBatch(entityList); Db.saveOrUpdateBatch(entityList);
// 临时解决 新版本 mp 插入状态判断错误问题
return true;
} }
/** /**
* 批量插入实体对象集合并指定批处理大小 * 批量插入(包含限制条数)
*
* @param entityList 实体对象集合
* @param batchSize 批处理大小
* @return 插入操作是否成功的布尔值
*/ */
default boolean insertBatch(Collection<T> entityList, int batchSize) { default boolean insertBatch(Collection<T> entityList, int batchSize) {
return Db.saveBatch(entityList, batchSize); Db.saveBatch(entityList, batchSize);
// 临时解决 新版本 mp 插入状态判断错误问题
return true;
} }
/** /**
* 批量根据ID更新实体对象集合并指定批处理大小 * 批量更新(包含限制条数)
*
* @param entityList 实体对象集合
* @param batchSize 批处理大小
* @return 更新操作是否成功的布尔值
*/ */
default boolean updateBatchById(Collection<T> entityList, int batchSize) { default boolean updateBatchById(Collection<T> entityList, int batchSize) {
return Db.updateBatchById(entityList, batchSize); Db.updateBatchById(entityList, batchSize);
// 临时解决 新版本 mp 插入状态判断错误问题
return true;
} }
/** /**
* 批量插入或更新实体对象集合并指定批处理大小 * 批量插入或更新(包含限制条数)
*
* @param entityList 实体对象集合
* @param batchSize 批处理大小
* @return 插入或更新操作是否成功的布尔值
*/ */
default boolean insertOrUpdateBatch(Collection<T> entityList, int batchSize) { default boolean insertOrUpdateBatch(Collection<T> entityList, int batchSize) {
return Db.saveOrUpdateBatch(entityList, batchSize); Db.saveOrUpdateBatch(entityList, batchSize);
// 临时解决 新版本 mp 插入状态判断错误问题
return true;
} }
/**
* 根据ID查询单个VO对象
*
* @param id 主键ID
* @return 查询到的单个VO对象
*/
default V selectVoById(Serializable id) { default V selectVoById(Serializable id) {
return selectVoById(id, this.currentVoClass()); return selectVoById(id, this.currentVoClass());
} }
/** /**
* 根据ID查询单个VO对象并将其转换为指定的VO类 * 根据 ID 查询
*
* @param id 主键ID
* @param voClass 要转换的VO类的Class对象
* @param <C> VO类的类型
* @return 查询到的单个VO对象经过转换为指定的VO类后返回
*/ */
default <C> C selectVoById(Serializable id, Class<C> voClass) { default <C> C selectVoById(Serializable id, Class<C> voClass) {
T obj = this.selectById(id); T obj = this.selectById(id);
@@ -149,49 +116,27 @@ public interface BaseMapperPlus<T, V> extends BaseMapper<T> {
return MapstructUtils.convert(obj, voClass); return MapstructUtils.convert(obj, voClass);
} }
/** default List<V> selectVoBatchIds(Collection<? extends Serializable> idList) {
* 根据ID集合批量查询VO对象列表 return selectVoBatchIds(idList, this.currentVoClass());
*
* @param idList 主键ID集合
* @return 查询到的VO对象列表
*/
default List<V> selectVoByIds(Collection<? extends Serializable> idList) {
return selectVoByIds(idList, this.currentVoClass());
} }
/** /**
* 根据ID集合批量查询实体对象列表并将其转换为指定的VO对象列表 * 查询(根据ID 批量查询
*
* @param idList 主键ID集合
* @param voClass 要转换的VO类的Class对象
* @param <C> VO类的类型
* @return 查询到的VO对象列表经过转换为指定的VO类后返回
*/ */
default <C> List<C> selectVoByIds(Collection<? extends Serializable> idList, Class<C> voClass) { default <C> List<C> selectVoBatchIds(Collection<? extends Serializable> idList, Class<C> voClass) {
List<T> list = this.selectByIds(idList); List<T> list = this.selectBatchIds(idList);
if (CollUtil.isEmpty(list)) { if (CollUtil.isEmpty(list)) {
return CollUtil.newArrayList(); return CollUtil.newArrayList();
} }
return MapstructUtils.convert(list, voClass); return MapstructUtils.convert(list, voClass);
} }
/**
* 根据查询条件Map查询VO对象列表
*
* @param map 查询条件Map
* @return 查询到的VO对象列表
*/
default List<V> selectVoByMap(Map<String, Object> map) { default List<V> selectVoByMap(Map<String, Object> map) {
return selectVoByMap(map, this.currentVoClass()); return selectVoByMap(map, this.currentVoClass());
} }
/** /**
* 根据查询条件Map查询实体对象列表并将其转换为指定的VO对象列表 * 查询(根据 columnMap 条件)
*
* @param map 查询条件Map
* @param voClass 要转换的VO类的Class对象
* @param <C> VO类的类型
* @return 查询到的VO对象列表经过转换为指定的VO类后返回
*/ */
default <C> List<C> selectVoByMap(Map<String, Object> map, Class<C> voClass) { default <C> List<C> selectVoByMap(Map<String, Object> map, Class<C> voClass) {
List<T> list = this.selectByMap(map); List<T> list = this.selectByMap(map);
@@ -201,47 +146,23 @@ public interface BaseMapperPlus<T, V> extends BaseMapper<T> {
return MapstructUtils.convert(list, voClass); return MapstructUtils.convert(list, voClass);
} }
/**
* 根据条件查询单个VO对象
*
* @param wrapper 查询条件Wrapper
* @return 查询到的单个VO对象
*/
default V selectVoOne(Wrapper<T> wrapper) { default V selectVoOne(Wrapper<T> wrapper) {
return selectVoOne(wrapper, this.currentVoClass()); return selectVoOne(wrapper, this.currentVoClass());
} }
/**
* 根据条件查询单个VO对象并根据需要决定是否抛出异常
*
* @param wrapper 查询条件Wrapper
* @param throwEx 是否抛出异常的标志
* @return 查询到的单个VO对象
*/
default V selectVoOne(Wrapper<T> wrapper, boolean throwEx) { default V selectVoOne(Wrapper<T> wrapper, boolean throwEx) {
return selectVoOne(wrapper, this.currentVoClass(), throwEx); return selectVoOne(wrapper, this.currentVoClass(), throwEx);
} }
/** /**
* 根据条件查询单个VO对象并指定返回的VO对象的类型 * 根据 entity 条件查询一条记录
*
* @param wrapper 查询条件Wrapper
* @param voClass 返回的VO对象的Class对象
* @param <C> 返回的VO对象的类型
* @return 查询到的单个VO对象经过类型转换为指定的VO类后返回
*/ */
default <C> C selectVoOne(Wrapper<T> wrapper, Class<C> voClass) { default <C> C selectVoOne(Wrapper<T> wrapper, Class<C> voClass) {
return selectVoOne(wrapper, voClass, true); return selectVoOne(wrapper, voClass, true);
} }
/** /**
* 根据条件查询单个实体对象并将其转换为指定的VO对象 * 根据 entity 条件查询一条记录
*
* @param wrapper 查询条件Wrapper
* @param voClass 要转换的VO类的Class对象
* @param throwEx 是否抛出异常的标志
* @param <C> VO类的类型
* @return 查询到的单个VO对象经过转换为指定的VO类后返回
*/ */
default <C> C selectVoOne(Wrapper<T> wrapper, Class<C> voClass, boolean throwEx) { default <C> C selectVoOne(Wrapper<T> wrapper, Class<C> voClass, boolean throwEx) {
T obj = this.selectOne(wrapper, throwEx); T obj = this.selectOne(wrapper, throwEx);
@@ -251,32 +172,16 @@ public interface BaseMapperPlus<T, V> extends BaseMapper<T> {
return MapstructUtils.convert(obj, voClass); return MapstructUtils.convert(obj, voClass);
} }
/**
* 查询所有VO对象列表
*
* @return 查询到的VO对象列表
*/
default List<V> selectVoList() { default List<V> selectVoList() {
return selectVoList(new QueryWrapper<>(), this.currentVoClass()); return selectVoList(new QueryWrapper<>(), this.currentVoClass());
} }
/**
* 根据条件查询VO对象列表
*
* @param wrapper 查询条件Wrapper
* @return 查询到的VO对象列表
*/
default List<V> selectVoList(Wrapper<T> wrapper) { default List<V> selectVoList(Wrapper<T> wrapper) {
return selectVoList(wrapper, this.currentVoClass()); return selectVoList(wrapper, this.currentVoClass());
} }
/** /**
* 根据条件查询实体对象列表并将其转换为指定的VO对象列表 * 根据 entity 条件查询全部记录
*
* @param wrapper 查询条件Wrapper
* @param voClass 要转换的VO类的Class对象
* @param <C> VO类的类型
* @return 查询到的VO对象列表经过转换为指定的VO类后返回
*/ */
default <C> List<C> selectVoList(Wrapper<T> wrapper, Class<C> voClass) { default <C> List<C> selectVoList(Wrapper<T> wrapper, Class<C> voClass) {
List<T> list = this.selectList(wrapper); List<T> list = this.selectList(wrapper);
@@ -286,31 +191,15 @@ public interface BaseMapperPlus<T, V> extends BaseMapper<T> {
return MapstructUtils.convert(list, voClass); return MapstructUtils.convert(list, voClass);
} }
/**
* 根据条件分页查询VO对象列表
*
* @param page 分页信息
* @param wrapper 查询条件Wrapper
* @return 查询到的VO对象分页列表
*/
default <P extends IPage<V>> P selectVoPage(IPage<T> page, Wrapper<T> wrapper) { default <P extends IPage<V>> P selectVoPage(IPage<T> page, Wrapper<T> wrapper) {
return selectVoPage(page, wrapper, this.currentVoClass()); return selectVoPage(page, wrapper, this.currentVoClass());
} }
/** /**
* 根据条件分页查询实体对象列表并将其转换为指定的VO对象分页列表 * 分页查询VO
*
* @param page 分页信息
* @param wrapper 查询条件Wrapper
* @param voClass 要转换的VO类的Class对象
* @param <C> VO类的类型
* @param <P> VO对象分页列表的类型
* @return 查询到的VO对象分页列表经过转换为指定的VO类后返回
*/ */
default <C, P extends IPage<C>> P selectVoPage(IPage<T> page, Wrapper<T> wrapper, Class<C> voClass) { default <C, P extends IPage<C>> P selectVoPage(IPage<T> page, Wrapper<T> wrapper, Class<C> voClass) {
// 根据条件分页查询实体对象列表
List<T> list = this.selectList(page, wrapper); List<T> list = this.selectList(page, wrapper);
// 创建一个新的VO对象分页列表并设置分页信息
IPage<C> voPage = new Page<>(page.getCurrent(), page.getSize(), page.getTotal()); IPage<C> voPage = new Page<>(page.getCurrent(), page.getSize(), page.getTotal());
if (CollUtil.isEmpty(list)) { if (CollUtil.isEmpty(list)) {
return (P) voPage; return (P) voPage;
@@ -319,16 +208,9 @@ public interface BaseMapperPlus<T, V> extends BaseMapper<T> {
return (P) voPage; return (P) voPage;
} }
/**
* 根据条件查询符合条件的对象,并将其转换为指定类型的对象列表
*
* @param wrapper 查询条件Wrapper
* @param mapper 转换函数,用于将查询到的对象转换为指定类型的对象
* @param <C> 要转换的对象的类型
* @return 查询到的符合条件的对象列表,经过转换为指定类型的对象后返回
*/
default <C> List<C> selectObjs(Wrapper<T> wrapper, Function<? super Object, C> mapper) { default <C> List<C> selectObjs(Wrapper<T> wrapper, Function<? super Object, C> mapper) {
return StreamUtils.toList(this.selectObjs(wrapper), mapper); return this.selectObjs(wrapper).stream().filter(Objects::nonNull).map(mapper).collect(Collectors.toList());
} }
} }

View File

@@ -52,7 +52,7 @@ public interface TestDemoMapper extends BaseMapperPlus<TestDemo, TestDemoVo> {
@DataColumn(key = "deptName", value = "dept_id"), @DataColumn(key = "deptName", value = "dept_id"),
@DataColumn(key = "userName", value = "user_id") @DataColumn(key = "userName", value = "user_id")
}, joinStr = "AND") }, joinStr = "AND")
List<TestDemo> selectByIds(@Param(Constants.COLL) Collection<? extends Serializable> idList); List<TestDemo> selectBatchIds(@Param(Constants.COLL) Collection<? extends Serializable> idList);
@Override @Override
@DataPermission({ @DataPermission({

View File

@@ -101,7 +101,7 @@ public class TestDemoServiceImpl implements ITestDemoService {
public Boolean deleteWithValidByIds(Collection<Long> ids, Boolean isValid) { public Boolean deleteWithValidByIds(Collection<Long> ids, Boolean isValid) {
if (isValid) { if (isValid) {
// 做一些业务上的校验,判断是否需要校验 // 做一些业务上的校验,判断是否需要校验
List<TestDemo> list = baseMapper.selectByIds(ids); List<TestDemo> list = baseMapper.selectBatchIds(ids);
if (list.size() != ids.size()) { if (list.size() != ids.size()) {
throw new ServiceException("您没有删除权限!"); throw new ServiceException("您没有删除权限!");
} }

View File

@@ -1,5 +1,6 @@
package org.dromara.fishery.mapper; package org.dromara.fishery.mapper;
import org.dromara.common.mybatis.core.mapper.BaseMapperPlusJoin;
import org.dromara.fishery.domain.DeviceBindRecord; import org.dromara.fishery.domain.DeviceBindRecord;
import org.dromara.fishery.domain.vo.DeviceBindRecordVo; import org.dromara.fishery.domain.vo.DeviceBindRecordVo;
import org.dromara.common.mybatis.core.mapper.BaseMapperPlus; import org.dromara.common.mybatis.core.mapper.BaseMapperPlus;
@@ -10,6 +11,6 @@ import org.dromara.common.mybatis.core.mapper.BaseMapperPlus;
* @author intc * @author intc
* @date 2025-10-14 * @date 2025-10-14
*/ */
public interface DeviceBindRecordMapper extends BaseMapperPlus<DeviceBindRecord, DeviceBindRecordVo> { public interface DeviceBindRecordMapper extends BaseMapperPlusJoin<DeviceBindRecord, DeviceBindRecordVo> {
} }

View File

@@ -1,6 +1,7 @@
package org.dromara.fishery.mapper; package org.dromara.fishery.mapper;
import org.dromara.common.mybatis.core.mapper.BaseMapperPlus; import org.dromara.common.mybatis.core.mapper.BaseMapperPlus;
import org.dromara.common.mybatis.core.mapper.BaseMapperPlusJoin;
import org.dromara.fishery.domain.Device; import org.dromara.fishery.domain.Device;
import org.dromara.fishery.domain.vo.DeviceVo; import org.dromara.fishery.domain.vo.DeviceVo;
import org.dromara.common.mybatis.core.mapper.BaseMapperPlus; import org.dromara.common.mybatis.core.mapper.BaseMapperPlus;
@@ -11,6 +12,6 @@ import org.dromara.common.mybatis.core.mapper.BaseMapperPlus;
* @author intc * @author intc
* @date 2025-10-14 * @date 2025-10-14
*/ */
public interface DeviceMapper extends BaseMapperPlus<Device, DeviceVo> { public interface DeviceMapper extends BaseMapperPlusJoin<Device, DeviceVo> {
} }

View File

@@ -1,5 +1,6 @@
package org.dromara.fishery.mapper; package org.dromara.fishery.mapper;
import org.dromara.common.mybatis.core.mapper.BaseMapperPlusJoin;
import org.dromara.fishery.domain.Pond; import org.dromara.fishery.domain.Pond;
import org.dromara.fishery.domain.vo.PondVo; import org.dromara.fishery.domain.vo.PondVo;
import org.dromara.common.mybatis.core.mapper.BaseMapperPlus; import org.dromara.common.mybatis.core.mapper.BaseMapperPlus;
@@ -10,6 +11,6 @@ import org.dromara.common.mybatis.core.mapper.BaseMapperPlus;
* @author intc * @author intc
* @date 2025-10-11 * @date 2025-10-11
*/ */
public interface PondMapper extends BaseMapperPlus<Pond, PondVo> { public interface PondMapper extends BaseMapperPlusJoin<Pond, PondVo> {
} }

View File

@@ -5,8 +5,7 @@ import org.dromara.common.core.utils.StringUtils;
import org.dromara.common.mybatis.core.page.TableDataInfo; import org.dromara.common.mybatis.core.page.TableDataInfo;
import org.dromara.common.mybatis.core.page.PageQuery; import org.dromara.common.mybatis.core.page.PageQuery;
import com.baomidou.mybatisplus.extension.plugins.pagination.Page; import com.baomidou.mybatisplus.extension.plugins.pagination.Page;
import com.baomidou.mybatisplus.core.conditions.query.LambdaQueryWrapper; import com.github.yulichang.wrapper.MPJLambdaWrapper;
import com.baomidou.mybatisplus.core.toolkit.Wrappers;
import lombok.RequiredArgsConstructor; import lombok.RequiredArgsConstructor;
import lombok.extern.slf4j.Slf4j; import lombok.extern.slf4j.Slf4j;
import org.springframework.stereotype.Service; import org.springframework.stereotype.Service;
@@ -14,15 +13,12 @@ import org.dromara.fishery.domain.AquUser;
import org.dromara.fishery.domain.bo.DeviceBindRecordBo; import org.dromara.fishery.domain.bo.DeviceBindRecordBo;
import org.dromara.fishery.domain.vo.DeviceBindRecordVo; import org.dromara.fishery.domain.vo.DeviceBindRecordVo;
import org.dromara.fishery.domain.DeviceBindRecord; import org.dromara.fishery.domain.DeviceBindRecord;
import org.dromara.fishery.mapper.AquUserMapper;
import org.dromara.fishery.mapper.DeviceBindRecordMapper; import org.dromara.fishery.mapper.DeviceBindRecordMapper;
import org.dromara.fishery.service.IDeviceBindRecordService; import org.dromara.fishery.service.IDeviceBindRecordService;
import java.util.Arrays;
import java.util.List; import java.util.List;
import java.util.Map; import java.util.Map;
import java.util.Collection; import java.util.Collection;
import java.util.stream.Collectors;
/** /**
* 设备绑定记录Service业务层处理 * 设备绑定记录Service业务层处理
@@ -36,7 +32,6 @@ import java.util.stream.Collectors;
public class DeviceBindRecordServiceImpl implements IDeviceBindRecordService { public class DeviceBindRecordServiceImpl implements IDeviceBindRecordService {
private final DeviceBindRecordMapper baseMapper; private final DeviceBindRecordMapper baseMapper;
private final AquUserMapper aquUserMapper;
/** /**
* 查询设备绑定记录 * 查询设备绑定记录
@@ -58,12 +53,8 @@ public class DeviceBindRecordServiceImpl implements IDeviceBindRecordService {
*/ */
@Override @Override
public TableDataInfo<DeviceBindRecordVo> queryPageList(DeviceBindRecordBo bo, PageQuery pageQuery) { public TableDataInfo<DeviceBindRecordVo> queryPageList(DeviceBindRecordBo bo, PageQuery pageQuery) {
LambdaQueryWrapper<DeviceBindRecord> lqw = buildQueryWrapper(bo); MPJLambdaWrapper<DeviceBindRecord> wrapper = buildQueryWrapper(bo);
Page<DeviceBindRecordVo> result = baseMapper.selectVoPage(pageQuery.build(), lqw); Page<DeviceBindRecordVo> result = baseMapper.selectJoinPage(pageQuery.build(), DeviceBindRecordVo.class, wrapper);
// 批量填充用户信息
enrichRecordListWithUserInfo(result.getRecords());
return TableDataInfo.build(result); return TableDataInfo.build(result);
} }
@@ -75,50 +66,61 @@ public class DeviceBindRecordServiceImpl implements IDeviceBindRecordService {
*/ */
@Override @Override
public List<DeviceBindRecordVo> queryList(DeviceBindRecordBo bo) { public List<DeviceBindRecordVo> queryList(DeviceBindRecordBo bo) {
LambdaQueryWrapper<DeviceBindRecord> lqw = buildQueryWrapper(bo); MPJLambdaWrapper<DeviceBindRecord> wrapper = buildQueryWrapper(bo);
List<DeviceBindRecordVo> recordList = baseMapper.selectVoList(lqw); return baseMapper.selectVoJoinList(wrapper);
// 批量填充用户信息
enrichRecordListWithUserInfo(recordList);
return recordList;
} }
private LambdaQueryWrapper<DeviceBindRecord> buildQueryWrapper(DeviceBindRecordBo bo) { /**
* 构建查询条件包装器(使用 MPJLambdaWrapper 支持联表查询)
*
* @param bo 查询条件业务对象
* @return 构建好的查询条件包装器
*/
private MPJLambdaWrapper<DeviceBindRecord> buildQueryWrapper(DeviceBindRecordBo bo) {
Map<String, Object> params = bo.getParams(); Map<String, Object> params = bo.getParams();
LambdaQueryWrapper<DeviceBindRecord> lqw = Wrappers.lambdaQuery(); MPJLambdaWrapper<DeviceBindRecord> wrapper = new MPJLambdaWrapper<>();
lqw.orderByDesc(DeviceBindRecord::getCreateTime);
lqw.like(StringUtils.isNotBlank(bo.getIotId()), DeviceBindRecord::getIotId, bo.getIotId()); // 选择主表所有字段
lqw.eq(bo.getDeviceType() != null, DeviceBindRecord::getDeviceType, bo.getDeviceType()); wrapper.selectAll(DeviceBindRecord.class);
lqw.eq(StringUtils.isNotBlank(bo.getSerialNum()), DeviceBindRecord::getSerialNum, bo.getSerialNum()); // 选择关联表的字段(使用 selectAs 映射到 VO
lqw.eq(bo.getUserId() != null, DeviceBindRecord::getUserId, bo.getUserId()); wrapper.selectAs(AquUser::getUserName, DeviceBindRecordVo::getUserName)
lqw.eq(bo.getIsBind() != null, DeviceBindRecord::getIsBind, bo.getIsBind()); .selectAs(AquUser::getMobilePhone, DeviceBindRecordVo::getMobilePhone);
// 关联查询用户表
wrapper.leftJoin(AquUser.class, AquUser::getId, DeviceBindRecord::getUserId);
// 默认按创建时间倒序排序
wrapper.orderByDesc(DeviceBindRecord::getCreateTime);
// 基础查询条件
wrapper.like(StringUtils.isNotBlank(bo.getIotId()), DeviceBindRecord::getIotId, bo.getIotId());
wrapper.eq(bo.getDeviceType() != null, DeviceBindRecord::getDeviceType, bo.getDeviceType());
wrapper.eq(StringUtils.isNotBlank(bo.getSerialNum()), DeviceBindRecord::getSerialNum, bo.getSerialNum());
wrapper.eq(bo.getUserId() != null, DeviceBindRecord::getUserId, bo.getUserId());
wrapper.eq(bo.getIsBind() != null, DeviceBindRecord::getIsBind, bo.getIsBind());
// 处理额外的查询参数 // 处理额外的查询参数
if (params != null && !params.isEmpty()) { if (params != null && !params.isEmpty()) {
handleExtraParams(lqw, params); handleExtraParams(wrapper, params);
} }
return lqw; return wrapper;
} }
/** /**
* 处理额外的查询参数 * 处理额外的查询参数
* *
* @param lqw 查询包装器 * @param wrapper 查询包装器
* @param params 额外参数 * @param params 额外参数
*/ */
private void handleExtraParams(LambdaQueryWrapper<DeviceBindRecord> lqw, Map<String, Object> params) { private void handleExtraParams(MPJLambdaWrapper<DeviceBindRecord> wrapper, Map<String, Object> params) {
// 处理用户搜索关键词(用户名或手机号) // 处理用户搜索关键词(用户名或手机号)
String userKeyword = (String) params.get("userKeyword"); String userKeyword = (String) params.get("userKeyword");
if (StringUtils.isNotBlank(userKeyword)) { if (StringUtils.isNotBlank(userKeyword)) {
List<Long> matchingUserIds = getMatchingUserIds(userKeyword); // 使用联表查询直接在 SQL 中过滤用户信息
if (matchingUserIds.isEmpty()) { wrapper.and(w -> w.like(AquUser::getUserName, userKeyword)
// 如果没有符合条件的用户,直接返回空结果 .or()
lqw.eq(DeviceBindRecord::getId, -1); // 使用不存在的ID来返回空结果 .like(AquUser::getMobilePhone, userKeyword));
return;
}
lqw.in(DeviceBindRecord::getUserId, matchingUserIds);
} }
} }
@@ -174,63 +176,5 @@ public class DeviceBindRecordServiceImpl implements IDeviceBindRecordService {
return baseMapper.deleteByIds(ids) > 0; return baseMapper.deleteByIds(ids) > 0;
} }
/**
* 为记录列表批量填充用户信息
*
* @param recordList 记录列表
*/
private void enrichRecordListWithUserInfo(List<DeviceBindRecordVo> recordList) {
if (recordList == null || recordList.isEmpty()) {
return;
}
// 收集所有的用户ID
List<Long> userIds = recordList.stream()
.map(DeviceBindRecordVo::getUserId)
.filter(userId -> userId != null)
.distinct()
.collect(Collectors.toList());
if (userIds.isEmpty()) {
return;
}
// 批量查询用户信息
List<AquUser> userList = aquUserMapper.selectBatchIds(userIds);
Map<Long, AquUser> userMap = userList.stream()
.collect(Collectors.toMap(AquUser::getId, user -> user, (existing, replacement) -> existing));
// 为每条记录设置用户信息
for (DeviceBindRecordVo record : recordList) {
if (record.getUserId() != null) {
AquUser user = userMap.get(record.getUserId());
if (user != null) {
record.setUserName(user.getUserName());
record.setMobilePhone(user.getMobilePhone());
}
}
}
}
/**
* 根据用户关键词查询符合条件的用户ID列表
*
* @param userKeyword 用户搜索关键词(用户名或手机号)
* @return 符合条件的用户ID列表
*/
private List<Long> getMatchingUserIds(String userKeyword) {
if (StringUtils.isBlank(userKeyword)) {
return Arrays.asList();
}
LambdaQueryWrapper<AquUser> userQuery = Wrappers.lambdaQuery();
userQuery.like(AquUser::getUserName, userKeyword)
.or()
.like(AquUser::getMobilePhone, userKeyword);
List<AquUser> matchingUsers = aquUserMapper.selectList(userQuery);
return matchingUsers.stream()
.map(AquUser::getId)
.collect(Collectors.toList());
}
} }

View File

@@ -1,11 +1,9 @@
package org.dromara.fishery.service.impl; package org.dromara.fishery.service.impl;
import java.util.Arrays;
import java.util.Collection; import java.util.Collection;
import java.util.Date; import java.util.Date;
import java.util.List; import java.util.List;
import java.util.Map; import java.util.Map;
import java.util.stream.Collectors;
import org.dromara.common.core.utils.MapstructUtils; import org.dromara.common.core.utils.MapstructUtils;
import org.dromara.common.core.utils.StringUtils; import org.dromara.common.core.utils.StringUtils;
@@ -16,15 +14,14 @@ import org.dromara.fishery.domain.Device;
import org.dromara.fishery.domain.Pond; import org.dromara.fishery.domain.Pond;
import org.dromara.fishery.domain.bo.DeviceBo; import org.dromara.fishery.domain.bo.DeviceBo;
import org.dromara.fishery.domain.vo.DeviceVo; import org.dromara.fishery.domain.vo.DeviceVo;
import org.dromara.fishery.mapper.AquUserMapper;
import org.dromara.fishery.mapper.DeviceMapper; import org.dromara.fishery.mapper.DeviceMapper;
import org.dromara.fishery.mapper.PondMapper;
import org.dromara.fishery.service.IDeviceService; import org.dromara.fishery.service.IDeviceService;
import org.springframework.stereotype.Service; import org.springframework.stereotype.Service;
import com.baomidou.mybatisplus.core.conditions.query.LambdaQueryWrapper; import com.baomidou.mybatisplus.core.conditions.query.LambdaQueryWrapper;
import com.baomidou.mybatisplus.core.toolkit.Wrappers; import com.baomidou.mybatisplus.core.toolkit.Wrappers;
import com.baomidou.mybatisplus.extension.plugins.pagination.Page; import com.baomidou.mybatisplus.extension.plugins.pagination.Page;
import com.github.yulichang.wrapper.MPJLambdaWrapper;
import lombok.RequiredArgsConstructor; import lombok.RequiredArgsConstructor;
import lombok.extern.slf4j.Slf4j; import lombok.extern.slf4j.Slf4j;
@@ -41,8 +38,6 @@ import lombok.extern.slf4j.Slf4j;
public class DeviceServiceImpl implements IDeviceService { public class DeviceServiceImpl implements IDeviceService {
private final DeviceMapper baseMapper; private final DeviceMapper baseMapper;
private final AquUserMapper aquUserMapper;
private final PondMapper pondMapper;
/** /**
* 查询设备管理 * 查询设备管理
@@ -56,7 +51,7 @@ public class DeviceServiceImpl implements IDeviceService {
} }
/** /**
* 分页查询设备管理列表 - 批量查询方案避免N+1问题 * 分页查询设备管理列表 - JOIN查询方案避免N+1问题
* *
* @param bo 查询条件 * @param bo 查询条件
* @param pageQuery 分页参数 * @param pageQuery 分页参数
@@ -64,12 +59,9 @@ public class DeviceServiceImpl implements IDeviceService {
*/ */
@Override @Override
public TableDataInfo<DeviceVo> queryPageList(DeviceBo bo, PageQuery pageQuery) { public TableDataInfo<DeviceVo> queryPageList(DeviceBo bo, PageQuery pageQuery) {
LambdaQueryWrapper<Device> lqw = buildQueryWrapper(bo); MPJLambdaWrapper<Device> wrapper = buildQueryWrapper(bo);
Page<DeviceVo> result = baseMapper.selectVoPage(pageQuery.build(), lqw); Page<DeviceVo> result = baseMapper.selectJoinPage(pageQuery.build(), DeviceVo.class, wrapper);
// 批量填充关联数据
enrichDeviceListWithUserInfo(result.getRecords());
enrichDeviceListWithPondInfo(result.getRecords());
// 计算过期信息 // 计算过期信息
calculateExpirationInfo(result.getRecords()); calculateExpirationInfo(result.getRecords());
@@ -77,84 +69,83 @@ public class DeviceServiceImpl implements IDeviceService {
} }
/** /**
* 查询符合条件的设备管理列表 - 批量查询方案避免N+1问题 * 查询符合条件的设备管理列表 - JOIN查询方案避免N+1问题
* *
* @param bo 查询条件 * @param bo 查询条件
* @return 设备管理列表 * @return 设备管理列表
*/ */
@Override @Override
public List<DeviceVo> queryList(DeviceBo bo) { public List<DeviceVo> queryList(DeviceBo bo) {
LambdaQueryWrapper<Device> lqw = buildQueryWrapper(bo); MPJLambdaWrapper<Device> wrapper = buildQueryWrapper(bo);
List<DeviceVo> deviceList = baseMapper.selectVoList(lqw); List<DeviceVo> deviceList = baseMapper.selectJoinList(DeviceVo.class, wrapper);
// 批量填充关联数据
enrichDeviceListWithUserInfo(deviceList);
enrichDeviceListWithPondInfo(deviceList);
// 计算过期信息 // 计算过期信息
calculateExpirationInfo(deviceList); calculateExpirationInfo(deviceList);
return deviceList; return deviceList;
} }
private LambdaQueryWrapper<Device> buildQueryWrapper(DeviceBo bo) { private MPJLambdaWrapper<Device> buildQueryWrapper(DeviceBo bo) {
Map<String, Object> params = bo.getParams(); Map<String, Object> params = bo.getParams();
LambdaQueryWrapper<Device> lqw = Wrappers.lambdaQuery(); MPJLambdaWrapper<Device> wrapper = new MPJLambdaWrapper<>();
// Select main table fields
wrapper.selectAll(Device.class);
// Select joined table fields with Lambda mapping
wrapper.selectAs(AquUser::getUserName, DeviceVo::getUserName)
.selectAs(AquUser::getMobilePhone, DeviceVo::getMobilePhone);
wrapper.selectAs(Pond::getPondName, DeviceVo::getPondName);
// Left join with user table
wrapper.leftJoin(AquUser.class, AquUser::getId, Device::getUserId);
// Left join with pond table
wrapper.leftJoin(Pond.class, Pond::getId, Device::getPondId);
// 处理排序逼辑如果expiredFlag为1按过期时间倒序否则按创建时间倒序 // 处理排序逼辑如果expiredFlag为1按过期时间倒序否则按创建时间倒序
String expiredFlag = params != null ? (String) params.get("expiredFlag") : null; String expiredFlag = params != null ? (String) params.get("expiredFlag") : null;
if ("1".equals(expiredFlag)) { if ("1".equals(expiredFlag)) {
// 按deadTime倒序过期时间越早的越靠后即过期越久的越靠前 // 按deadTime倒序过期时间越早的越靠后即过期越久的越靠前
lqw.orderByAsc(Device::getDeadTime); wrapper.orderByAsc(Device::getDeadTime);
} else { } else {
lqw.orderByDesc(Device::getCreateTime); wrapper.orderByDesc(Device::getCreateTime);
} }
lqw.eq(bo.getUserId() != null, Device::getUserId, bo.getUserId()); wrapper.eq(bo.getUserId() != null, Device::getUserId, bo.getUserId());
lqw.eq(StringUtils.isNotBlank(bo.getSerialNum()), Device::getSerialNum, bo.getSerialNum()); wrapper.eq(StringUtils.isNotBlank(bo.getSerialNum()), Device::getSerialNum, bo.getSerialNum());
lqw.like(StringUtils.isNotBlank(bo.getDeviceName()), Device::getDeviceName, bo.getDeviceName()); wrapper.like(StringUtils.isNotBlank(bo.getDeviceName()), Device::getDeviceName, bo.getDeviceName());
lqw.eq(bo.getDeviceType() != null, Device::getDeviceType, bo.getDeviceType()); wrapper.eq(bo.getDeviceType() != null, Device::getDeviceType, bo.getDeviceType());
lqw.eq(bo.getBindTime() != null, Device::getBindTime, bo.getBindTime()); wrapper.eq(bo.getBindTime() != null, Device::getBindTime, bo.getBindTime());
lqw.eq(bo.getPondId() != null, Device::getPondId, bo.getPondId()); wrapper.eq(bo.getPondId() != null, Device::getPondId, bo.getPondId());
lqw.like(StringUtils.isNotBlank(bo.getIccId()), Device::getIccId, bo.getIccId()); wrapper.like(StringUtils.isNotBlank(bo.getIccId()), Device::getIccId, bo.getIccId());
lqw.eq(StringUtils.isNotBlank(bo.getCategory()), Device::getCategory, bo.getCategory()); wrapper.eq(StringUtils.isNotBlank(bo.getCategory()), Device::getCategory, bo.getCategory());
// 处理额外的查询参数 // 处理额外的查询参数
if (params != null && !params.isEmpty()) { if (params != null && !params.isEmpty()) {
handleExtraParams(lqw, params); handleExtraParams(wrapper, params);
} }
return lqw; return wrapper;
} }
/** /**
* 处理额外的查询参数 * 处理额外的查询参数
* *
* @param lqw 查询包装器 * @param wrapper 查询包装器
* @param params 额外参数 * @param params 额外参数
*/ */
private void handleExtraParams(LambdaQueryWrapper<Device> lqw, Map<String, Object> params) { private void handleExtraParams(MPJLambdaWrapper<Device> wrapper, Map<String, Object> params) {
// 处理用户搜索关键词(用户名或手机号) // 处理用户搜索关键词(用户名或手机号)- 直接在SQL中进行JOIN查询
String userKeyword = (String) params.get("userKeyword"); String userKeyword = (String) params.get("userKeyword");
if (StringUtils.isNotBlank(userKeyword)) { if (StringUtils.isNotBlank(userKeyword)) {
List<Long> matchingUserIds = getMatchingUserIds(userKeyword); wrapper.and(w -> w.like(AquUser::getUserName, userKeyword)
if (matchingUserIds.isEmpty()) { .or()
// 如果没有符合条件的用户,直接返回空结果 .like(AquUser::getMobilePhone, userKeyword));
lqw.eq(Device::getId, -1); // 使用不存在的ID来返回空结果
return;
}
lqw.in(Device::getUserId, matchingUserIds);
} }
// 处理塘口搜索关键词(塘口名称) // 处理塘口搜索关键词(塘口名称)- 直接在SQL中进行JOIN查询
String pondKeyword = (String) params.get("pondKeyword"); String pondKeyword = (String) params.get("pondKeyword");
if (StringUtils.isNotBlank(pondKeyword)) { if (StringUtils.isNotBlank(pondKeyword)) {
List<Long> matchingPondIds = getMatchingPondIds(pondKeyword); wrapper.like(Pond::getPondName, pondKeyword);
if (matchingPondIds.isEmpty()) {
// 如果没有符合条件的塘口,直接返回空结果
lqw.eq(Device::getId, -1); // 使用不存在的ID来返回空结果
return;
}
lqw.in(Device::getPondId, matchingPondIds);
} }
// 处理是否过期查询条件 // 处理是否过期查询条件
@@ -165,10 +156,10 @@ public class DeviceServiceImpl implements IDeviceService {
Date currentDate = new Date(); Date currentDate = new Date();
if (isExpired == 1) { if (isExpired == 1) {
// 查询已过期的设备deadTime < 当前时间) // 查询已过期的设备deadTime < 当前时间)
lqw.lt(Device::getDeadTime, currentDate); wrapper.lt(Device::getDeadTime, currentDate);
} else if (isExpired == 0) { } else if (isExpired == 0) {
// 查询未过期的设备deadTime >= 当前时间 或 deadTime 为空) // 查询未过期的设备deadTime >= 当前时间 或 deadTime 为空)
lqw.and(wrapper -> wrapper.ge(Device::getDeadTime, currentDate) wrapper.and(w -> w.ge(Device::getDeadTime, currentDate)
.or() .or()
.isNull(Device::getDeadTime)); .isNull(Device::getDeadTime));
} }
@@ -230,122 +221,7 @@ public class DeviceServiceImpl implements IDeviceService {
return baseMapper.deleteByIds(ids) > 0; return baseMapper.deleteByIds(ids) > 0;
} }
/**
* 为设备列表批量填充用户信息
*
* @param deviceList 设备列表
*/
private void enrichDeviceListWithUserInfo(List<DeviceVo> deviceList) {
if (deviceList == null || deviceList.isEmpty()) {
return;
}
// 收集所有的用户ID
List<Long> userIds = deviceList.stream()
.map(DeviceVo::getUserId)
.filter(userId -> userId != null)
.distinct()
.collect(Collectors.toList());
if (userIds.isEmpty()) {
return;
}
// 批量查询用户信息
List<AquUser> userList = aquUserMapper.selectBatchIds(userIds);
Map<Long, AquUser> userMap = userList.stream()
.collect(Collectors.toMap(AquUser::getId, user -> user, (existing, replacement) -> existing));
// 为每个设备设置用户信息
for (DeviceVo device : deviceList) {
if (device.getUserId() != null) {
AquUser user = userMap.get(device.getUserId());
if (user != null) {
device.setUserName(user.getUserName());
device.setMobilePhone(user.getMobilePhone());
}
}
}
}
/**
* 为设备列表批量填充塘口信息
*
* @param deviceList 设备列表
*/
private void enrichDeviceListWithPondInfo(List<DeviceVo> deviceList) {
if (deviceList == null || deviceList.isEmpty()) {
return;
}
// 收集所有的塘口ID
List<Long> pondIds = deviceList.stream()
.map(DeviceVo::getPondId)
.filter(pondId -> pondId != null)
.distinct()
.collect(Collectors.toList());
if (pondIds.isEmpty()) {
return;
}
// 批量查询塘口信息
List<Pond> pondList = pondMapper.selectBatchIds(pondIds);
Map<Long, Pond> pondMap = pondList.stream()
.collect(Collectors.toMap(Pond::getId, pond -> pond, (existing, replacement) -> existing));
// 为每个设备设置塘口信息
for (DeviceVo device : deviceList) {
if (device.getPondId() != null) {
Pond pond = pondMap.get(device.getPondId());
if (pond != null) {
device.setPondName(pond.getPondName());
}
}
}
}
/**
* 根据用户关键词查询符合条件的用户ID列表
*
* @param userKeyword 用户搜索关键词(用户名或手机号)
* @return 符合条件的用户ID列表
*/
private List<Long> getMatchingUserIds(String userKeyword) {
if (StringUtils.isBlank(userKeyword)) {
return Arrays.asList();
}
LambdaQueryWrapper<AquUser> userQuery = Wrappers.lambdaQuery();
userQuery.like(AquUser::getUserName, userKeyword)
.or()
.like(AquUser::getMobilePhone, userKeyword);
List<AquUser> matchingUsers = aquUserMapper.selectList(userQuery);
return matchingUsers.stream()
.map(AquUser::getId)
.collect(Collectors.toList());
}
/**
* 根据塘口关键词查询符合条件的塘口ID列表
*
* @param pondKeyword 塘口搜索关键词(塘口名称)
* @return 符合条件的塘口ID列表
*/
private List<Long> getMatchingPondIds(String pondKeyword) {
if (StringUtils.isBlank(pondKeyword)) {
return Arrays.asList();
}
LambdaQueryWrapper<Pond> pondQuery = Wrappers.lambdaQuery();
pondQuery.like(Pond::getPondName, pondKeyword);
List<Pond> matchingPonds = pondMapper.selectList(pondQuery);
return matchingPonds.stream()
.map(Pond::getId)
.collect(Collectors.toList());
}
/** /**
* 计算设备列表的过期信息 * 计算设备列表的过期信息

View File

@@ -15,7 +15,6 @@ import org.dromara.fishery.domain.Fish;
import org.dromara.fishery.domain.Pond; import org.dromara.fishery.domain.Pond;
import org.dromara.fishery.domain.bo.PondBo; import org.dromara.fishery.domain.bo.PondBo;
import org.dromara.fishery.domain.vo.PondVo; import org.dromara.fishery.domain.vo.PondVo;
import org.dromara.fishery.mapper.AquUserMapper;
import org.dromara.fishery.mapper.FishMapper; import org.dromara.fishery.mapper.FishMapper;
import org.dromara.fishery.mapper.PondMapper; import org.dromara.fishery.mapper.PondMapper;
import org.dromara.fishery.service.IPondService; import org.dromara.fishery.service.IPondService;
@@ -24,6 +23,7 @@ import org.springframework.stereotype.Service;
import com.baomidou.mybatisplus.core.conditions.query.LambdaQueryWrapper; import com.baomidou.mybatisplus.core.conditions.query.LambdaQueryWrapper;
import com.baomidou.mybatisplus.core.toolkit.Wrappers; import com.baomidou.mybatisplus.core.toolkit.Wrappers;
import com.baomidou.mybatisplus.extension.plugins.pagination.Page; import com.baomidou.mybatisplus.extension.plugins.pagination.Page;
import com.github.yulichang.wrapper.MPJLambdaWrapper;
import lombok.RequiredArgsConstructor; import lombok.RequiredArgsConstructor;
import lombok.extern.slf4j.Slf4j; import lombok.extern.slf4j.Slf4j;
@@ -43,8 +43,6 @@ public class PondServiceImpl implements IPondService {
private final FishMapper fishMapper; private final FishMapper fishMapper;
private final AquUserMapper aquUserMapper;
/** /**
* 查询塘口管理 * 查询塘口管理
* *
@@ -57,7 +55,7 @@ public class PondServiceImpl implements IPondService {
} }
/** /**
* 分页查询塘口管理列表 * 分页查询塘口管理列表 - JOIN查询方案避免N+1问题
* *
* @param bo 查询条件 * @param bo 查询条件
* @param pageQuery 分页参数 * @param pageQuery 分页参数
@@ -65,44 +63,47 @@ public class PondServiceImpl implements IPondService {
*/ */
@Override @Override
public TableDataInfo<PondVo> queryPageList(PondBo bo, PageQuery pageQuery) { public TableDataInfo<PondVo> queryPageList(PondBo bo, PageQuery pageQuery) {
LambdaQueryWrapper<Pond> lqw = buildQueryWrapper(bo); MPJLambdaWrapper<Pond> wrapper = buildQueryWrapper(bo);
Page<PondVo> result = baseMapper.selectVoPage(pageQuery.build(), lqw); Page<PondVo> result = baseMapper.selectJoinPage(pageQuery.build(), PondVo.class, wrapper);
// 填充鱼类名称信息 // 填充鱼类名称信息鱼类ID是数组仍需后处理
enrichPondListWithFishNames(result.getRecords()); enrichPondListWithFishNames(result.getRecords());
// 填充用户信息
enrichPondListWithUserInfo(result.getRecords());
return TableDataInfo.build(result); return TableDataInfo.build(result);
} }
/** /**
* 查询符合条件的塘口管理列表 * 查询符合条件的塘口管理列表 - JOIN查询方案避免N+1问题
* *
* @param bo 查询条件 * @param bo 查询条件
* @return 塘口管理列表 * @return 塘口管理列表
*/ */
@Override @Override
public List<PondVo> queryList(PondBo bo) { public List<PondVo> queryList(PondBo bo) {
LambdaQueryWrapper<Pond> lqw = buildQueryWrapper(bo); MPJLambdaWrapper<Pond> wrapper = buildQueryWrapper(bo);
List<PondVo> list = baseMapper.selectVoList(lqw); List<PondVo> list = baseMapper.selectJoinList(PondVo.class, wrapper);
// 填充鱼类名称信息 // 填充鱼类名称信息鱼类ID是数组仍需后处理
enrichPondListWithFishNames(list); enrichPondListWithFishNames(list);
// 填充用户信息
enrichPondListWithUserInfo(list);
return list; return list;
} }
private LambdaQueryWrapper<Pond> buildQueryWrapper(PondBo bo) { private MPJLambdaWrapper<Pond> buildQueryWrapper(PondBo bo) {
Map<String, Object> params = bo.getParams(); Map<String, Object> params = bo.getParams();
LambdaQueryWrapper<Pond> lqw = Wrappers.lambdaQuery(); MPJLambdaWrapper<Pond> wrapper = new MPJLambdaWrapper<>();
// Select main table fields
wrapper.selectAll(Pond.class);
// Select joined table fields with Lambda mapping
wrapper.selectAs(AquUser::getUserName, PondVo::getUserName)
.selectAs(AquUser::getMobilePhone, PondVo::getMobilePhone);
// Left join with user table
wrapper.leftJoin(AquUser.class, AquUser::getId, Pond::getUserId);
// 基础条件查询 // 基础条件查询
lqw.eq(bo.getUserId() != null, Pond::getUserId, bo.getUserId()) wrapper.eq(bo.getUserId() != null, Pond::getUserId, bo.getUserId())
.like(StringUtils.isNotBlank(bo.getPondName()), Pond::getPondName, bo.getPondName()) .like(StringUtils.isNotBlank(bo.getPondName()), Pond::getPondName, bo.getPondName())
.like(StringUtils.isNotBlank(bo.getFishKindIds()), Pond::getFishKindIds, bo.getFishKindIds()) .like(StringUtils.isNotBlank(bo.getFishKindIds()), Pond::getFishKindIds, bo.getFishKindIds())
.eq(bo.getPlaceTime() != null, Pond::getPlaceTime, bo.getPlaceTime()) .eq(bo.getPlaceTime() != null, Pond::getPlaceTime, bo.getPlaceTime())
@@ -110,32 +111,28 @@ public class PondServiceImpl implements IPondService {
// 处理额外的查询参数 // 处理额外的查询参数
if (params != null && !params.isEmpty()) { if (params != null && !params.isEmpty()) {
handleExtraParams(lqw, params); handleExtraParams(wrapper, params);
} }
// 默认按创建时间升序排列 // 默认按创建时间升序排列
lqw.orderByDesc(Pond::getCreateTime); wrapper.orderByDesc(Pond::getCreateTime);
return lqw; return wrapper;
} }
/** /**
* 处理额外的查询参数 * 处理额外的查询参数
* *
* @param lqw 查询包装器 * @param wrapper 查询包装器
* @param params 额外参数 * @param params 额外参数
*/ */
private void handleExtraParams(LambdaQueryWrapper<Pond> lqw, Map<String, Object> params) { private void handleExtraParams(MPJLambdaWrapper<Pond> wrapper, Map<String, Object> params) {
// 处理用户搜索关键词 // 处理用户搜索关键词 - 直接在SQL中进行JOIN查询
String userKeyword = (String) params.get("userKeyword"); String userKeyword = (String) params.get("userKeyword");
if (StringUtils.isNotBlank(userKeyword)) { if (StringUtils.isNotBlank(userKeyword)) {
List<Long> matchingUserIds = getMatchingUserIds(userKeyword); wrapper.and(w -> w.like(AquUser::getUserName, userKeyword)
if (matchingUserIds.isEmpty()) { .or()
// 如果没有符合条件的用户,直接返回空结果 .like(AquUser::getMobilePhone, userKeyword));
lqw.eq(Pond::getId, -1); // 使用不存在的ID来返回空结果
return;
}
lqw.in(Pond::getUserId, matchingUserIds);
} }
// 处理时间范围查询 // 处理时间范围查询
@@ -152,7 +149,7 @@ public class PondServiceImpl implements IPondService {
if (startDateTime.length() == 10) { // yyyy-MM-dd 格式 if (startDateTime.length() == 10) { // yyyy-MM-dd 格式
startDateTime += " 00:00:00"; startDateTime += " 00:00:00";
} }
lqw.ge(Pond::getCreateTime, startDateTime); wrapper.ge(Pond::getCreateTime, startDateTime);
} catch (Exception e) { } catch (Exception e) {
log.warn("开始时间格式不正确: {}", startTime, e); log.warn("开始时间格式不正确: {}", startTime, e);
} }
@@ -165,7 +162,7 @@ public class PondServiceImpl implements IPondService {
if (endDateTime.length() == 10) { // yyyy-MM-dd 格式 if (endDateTime.length() == 10) { // yyyy-MM-dd 格式
endDateTime += " 23:59:59"; endDateTime += " 23:59:59";
} }
lqw.le(Pond::getCreateTime, endDateTime); wrapper.le(Pond::getCreateTime, endDateTime);
} catch (Exception e) { } catch (Exception e) {
log.warn("结束时间格式不正确: {}", endTime, e); log.warn("结束时间格式不正确: {}", endTime, e);
} }
@@ -294,63 +291,5 @@ public class PondServiceImpl implements IPondService {
} }
} }
/**
* 为塘口列表填充用户信息
*
* @param pondList 塘口列表
*/
private void enrichPondListWithUserInfo(List<PondVo> pondList) {
if (pondList == null || pondList.isEmpty()) {
return;
}
// 收集所有的用户ID
List<Long> userIds = pondList.stream()
.map(PondVo::getUserId)
.filter(userId -> userId != null)
.distinct()
.collect(Collectors.toList());
if (userIds.isEmpty()) {
return;
}
// 批量查询用户信息
List<AquUser> userList = aquUserMapper.selectBatchIds(userIds);
Map<Long, AquUser> userMap = userList.stream()
.collect(Collectors.toMap(AquUser::getId, user -> user, (existing, replacement) -> existing));
// 为每个塘口设置用户信息
for (PondVo pond : pondList) {
if (pond.getUserId() != null) {
AquUser user = userMap.get(pond.getUserId());
if (user != null) {
pond.setUserName(user.getUserName());
pond.setMobilePhone(user.getMobilePhone());
}
}
}
}
/**
* 根据用户关键词查询符合条件的用户ID列表
*
* @param userKeyword 用户搜索关键词(用户名或手机号)
* @return 符合条件的用户ID列表
*/
private List<Long> getMatchingUserIds(String userKeyword) {
if (StringUtils.isBlank(userKeyword)) {
return Arrays.asList();
}
LambdaQueryWrapper<AquUser> userQuery = Wrappers.lambdaQuery();
userQuery.like(AquUser::getUserName, userKeyword)
.or()
.like(AquUser::getMobilePhone, userKeyword);
List<AquUser> matchingUsers = aquUserMapper.selectList(userQuery);
return matchingUsers.stream()
.map(AquUser::getId)
.collect(Collectors.toList());
}
} }

View File

@@ -169,7 +169,7 @@ public class SysConfigServiceImpl implements ISysConfigService, ConfigService {
*/ */
@Override @Override
public void deleteConfigByIds(List<Long> configIds) { public void deleteConfigByIds(List<Long> configIds) {
List<SysConfig> list = baseMapper.selectByIds(configIds); List<SysConfig> list = baseMapper.selectBatchIds(configIds);
list.forEach(config -> { list.forEach(config -> {
if (StringUtils.equals(SystemConstants.YES, config.getConfigType())) { if (StringUtils.equals(SystemConstants.YES, config.getConfigType())) {
throw new ServiceException("内置参数【{}】不能删除", config.getConfigKey()); throw new ServiceException("内置参数【{}】不能删除", config.getConfigKey());

View File

@@ -102,7 +102,7 @@ public class SysDictDataServiceImpl implements ISysDictDataService {
*/ */
@Override @Override
public void deleteDictDataByIds(List<Long> dictCodes) { public void deleteDictDataByIds(List<Long> dictCodes) {
List<SysDictData> list = baseMapper.selectByIds(dictCodes); List<SysDictData> list = baseMapper.selectBatchIds(dictCodes);
baseMapper.deleteByIds(dictCodes); baseMapper.deleteByIds(dictCodes);
list.forEach(x -> CacheUtils.evict(CacheNames.SYS_DICT, x.getDictType())); list.forEach(x -> CacheUtils.evict(CacheNames.SYS_DICT, x.getDictType()));
} }

View File

@@ -138,7 +138,7 @@ public class SysDictTypeServiceImpl implements ISysDictTypeService, DictService
*/ */
@Override @Override
public void deleteDictTypeByIds(List<Long> dictIds) { public void deleteDictTypeByIds(List<Long> dictIds) {
List<SysDictType> list = baseMapper.selectByIds(dictIds); List<SysDictType> list = baseMapper.selectBatchIds(dictIds);
list.forEach(x -> { list.forEach(x -> {
boolean assigned = dictDataMapper.exists(new LambdaQueryWrapper<SysDictData>() boolean assigned = dictDataMapper.exists(new LambdaQueryWrapper<SysDictData>()
.eq(SysDictData::getDictType, x.getDictType())); .eq(SysDictData::getDictType, x.getDictType()));

View File

@@ -252,7 +252,7 @@ public class SysOssServiceImpl implements ISysOssService, OssService {
if (isValid) { if (isValid) {
// 做一些业务上的校验,判断是否需要校验 // 做一些业务上的校验,判断是否需要校验
} }
List<SysOss> list = baseMapper.selectByIds(ids); List<SysOss> list = baseMapper.selectBatchIds(ids);
for (SysOss sysOss : list) { for (SysOss sysOss : list) {
OssClient storage = OssFactory.instance(sysOss.getService()); OssClient storage = OssFactory.instance(sysOss.getService());
storage.delete(sysOss.getUrl()); storage.delete(sysOss.getUrl());

View File

@@ -222,7 +222,7 @@ public class SysPostServiceImpl implements ISysPostService, PostService {
*/ */
@Override @Override
public int deletePostByIds(List<Long> postIds) { public int deletePostByIds(List<Long> postIds) {
List<SysPost> list = baseMapper.selectByIds(postIds); List<SysPost> list = baseMapper.selectBatchIds(postIds);
for (SysPost post : list) { for (SysPost post : list) {
if (this.countUserPostById(post.getPostId()) > 0) { if (this.countUserPostById(post.getPostId()) > 0) {
throw new ServiceException("{}已分配,不能删除!", post.getPostName()); throw new ServiceException("{}已分配,不能删除!", post.getPostName());

View File

@@ -428,7 +428,7 @@ public class SysRoleServiceImpl implements ISysRoleService, RoleService {
@Transactional(rollbackFor = Exception.class) @Transactional(rollbackFor = Exception.class)
public int deleteRoleByIds(List<Long> roleIds) { public int deleteRoleByIds(List<Long> roleIds) {
this.checkRoleDataScope(roleIds); this.checkRoleDataScope(roleIds);
List<SysRole> roles = baseMapper.selectByIds(roleIds); List<SysRole> roles = baseMapper.selectBatchIds(roleIds);
for (SysRole role : roles) { for (SysRole role : roles) {
checkRoleAllowed(BeanUtil.toBean(role, SysRoleBo.class)); checkRoleAllowed(BeanUtil.toBean(role, SysRoleBo.class));
if (countUserRoleByRoleId(role.getRoleId()) > 0) { if (countUserRoleByRoleId(role.getRoleId()) > 0) {

View File

@@ -184,7 +184,7 @@ public class FlwDefinitionServiceImpl implements IFlwDefinitionService {
wrapper.in(FlowHisTask::getDefinitionId, ids); wrapper.in(FlowHisTask::getDefinitionId, ids);
List<FlowHisTask> flowHisTasks = flowHisTaskMapper.selectList(wrapper); List<FlowHisTask> flowHisTasks = flowHisTaskMapper.selectList(wrapper);
if (CollUtil.isNotEmpty(flowHisTasks)) { if (CollUtil.isNotEmpty(flowHisTasks)) {
List<FlowDefinition> flowDefinitions = flowDefinitionMapper.selectByIds(StreamUtils.toList(flowHisTasks, FlowHisTask::getDefinitionId)); List<FlowDefinition> flowDefinitions = flowDefinitionMapper.selectBatchIds(StreamUtils.toList(flowHisTasks, FlowHisTask::getDefinitionId));
if (CollUtil.isNotEmpty(flowDefinitions)) { if (CollUtil.isNotEmpty(flowDefinitions)) {
String join = StreamUtils.join(flowDefinitions, FlowDefinition::getFlowCode); String join = StreamUtils.join(flowDefinitions, FlowDefinition::getFlowCode);
log.info("流程定义【{}】已被使用不可被删除!", join); log.info("流程定义【{}】已被使用不可被删除!", join);

View File

@@ -171,7 +171,7 @@ public class FlwInstanceServiceImpl implements IFlwInstanceService {
*/ */
@Override @Override
public List<FlowInstance> selectInstListByIdList(List<Long> instanceIds) { public List<FlowInstance> selectInstListByIdList(List<Long> instanceIds) {
return flowInstanceMapper.selectByIds(instanceIds); return flowInstanceMapper.selectBatchIds(instanceIds);
} }
/** /**