JPA查询部分字段的最佳实践

最近在接手一个项目,用的是JPA,以前没使用过。在开发新的需求过程中,发现有查询部分字段的情况,网上逛了半天,发现都是抄来抄去的"古文"。于是用英文搜索了下,总结了以下几点。

版本信息:Spring Boot 2.3.7

  1. 不需要新建Entity,但是需要一个所有部分字段的构造函数。当然如果你觉得返回的Entity中包含大量null值字段,以及多一个所有部分字段的构造函数不妥,也可以新建一个子集Entity。

  2. 不使用findAll,而是使用EntityManager来执行CriteriaQuery.multiselect的查询。

  3. 没有用@Query,因为我的查询是动态的。

  4. 也可以支持分页。

示例代码,用模糊查询Merchant的部分字段,并要求分页。

(不过,总觉得这些代码有些冗余,不知道有没有其它更优的写法)

复制代码
public Page<Merchant> listMerchant(String name, Pageable pageable) {
	CriteriaBuilder totalRowsCb = this.entityManager.getCriteriaBuilder();
	CriteriaQuery<Long> totalRowsQuery = totalRowsCb.createQuery(Long.class);
	Root<Merchant> totalRowsRoot = totalRowsQuery.from(Merchant.class);
	totalRowsQuery.where(totalRowsCb.like(totalRowsRoot.get("name"), "%" + name + "%"));
	totalRowsQuery.select(totalRowsCb.count(totalRowsRoot));
	Long totalRows = entityManager.createQuery(totalRowsQuery).getSingleResult();

	CriteriaBuilder listCb = this.entityManager.getCriteriaBuilder();
	CriteriaQuery<Merchant> listQuery = listCb.createQuery(Merchant.class);
	Root<Merchant> listRoot = listQuery.from(Merchant.class);
	List<Order> orderList = QueryUtils.toOrders(pageable.getSort(), listRoot, listCb);

	listQuery .multiselect(
			listRoot.get("mid").alias("mid"),
			listRoot.get("name").alias("name"),
			listRoot.get("businessName").alias("businessName"),
			listRoot.get("contactEmail").alias("contactEmail"),
			listRoot.get("status").alias("status"),
			listRoot.get("accountStatus").alias("accountStatus")
	).where(listCb.like(listRoot.get("name"), "%" + name + "%")).orderBy(orderList);

	List<Merchant> merchantList = entityManager
			.createQuery(listQuery)
			.setMaxResults(pageable.getPageSize())
			.setFirstResult((int)pageable.getOffset())

			.getResultList();

	return new PageImpl<>(merchantList, pageable, totalRows);
}
相关推荐
偷光29 分钟前
浏览器中的隐藏IDE: Elements (元素) 面板
开发语言·前端·ide·php
DKPT32 分钟前
JVM栈溢出和堆溢出哪个先满?
java·开发语言·jvm·笔记·学习
m0_4750645032 分钟前
jvm双亲委派的含义
java·jvm
gopyer4 小时前
180课时吃透Go语言游戏后端开发6:Go语言的循环语句
开发语言·游戏·golang·循环语句
毕设源码-朱学姐5 小时前
【开题答辩全过程】以 爱心捐赠网站为例,包含答辩的问题和答案
java·eclipse
楼田莉子7 小时前
Qt开发学习——QtCreator深度介绍/程序运行/开发规范/对象树
开发语言·前端·c++·qt·学习
尘觉7 小时前
中秋节与 Spring Boot 的思考:一场开箱即用的团圆盛宴
java·spring boot·后端
Le1Yu8 小时前
2025-10-7学习笔记
java·笔记·学习
韩立学长8 小时前
【开题答辩实录分享】以《基于python的奶茶店分布数据分析与可视化》为例进行答辩实录分享
开发语言·python·数据分析
popoxf8 小时前
spring容器启动流程(反射视角)
java·后端·spring