总结下SpringData JPA 的常用语法
liuian 2025-08-06 21:06 61 浏览
SpringData JPA常用有两种写法,一个是用Jpa自带方法进行CRUD,适合简单查询场景、例如查询全部数据、根据某个字段查询,根据某字段排序等等。另一种是使用注解方式,@Query、@Modifying。
1.方法方式
方法说明
接口方法如下,方法作用见注释:
public interface JpaRepository<T, ID> extends PagingAndSortingRepository<T, ID>, QueryByExampleExecutor<T> {
// 无条件,查询全部记录
List<T> findAll();
// 排序查询
List<T> findAll(Sort var1);
// 根据主键ID查询
List<T> findAllById(Iterable<ID> var1);
// 批量保存集合数据
<S extends T> List<S> saveAll(Iterable<S> var1);
void flush();
<S extends T> S saveAndFlush(S var1);
// 批量删除
void deleteInBatch(Iterable<T> var1);
// all in 全部删除
void deleteAllInBatch();
// 查询一条记录
T getOne(ID var1);
// 条件查询
<S extends T> List<S> findAll(Example<S> var1);
// 条件查询,带排序
<S extends T> List<S> findAll(Example<S> var1, Sort var2);
}
复制代码例子
一般dao实现JpaRepository接口,直接调用JpaRepository中的方法就可以实现了简单查询,例如查询User实例列表:
// 构建user的Example对象
Example<User> example =Example.of(User);
List<User> users = userRepository.findAll(example);
复制代码2.注解方式
jpa实现CRUD的主要注解是@Query
注解说明
@Query注解主要有以下参数,参数作用如下:
- value:SQL语句
- countQuery: 分页查询时统计总数
- nativeQuery: 使用执行这个方法的时候执行原生sql语句,直接写数据库中的实际表名和表的实际字段名
@Query的代码如下:
@Retention(RetentionPolicy.RUNTIME)
@Target({ElementType.METHOD, ElementType.ANNOTATION_TYPE})
@QueryAnnotation
@Documented
public @interface Query {
String value() default "";
String countQuery() default "";
String countProjection() default "";
boolean nativeQuery() default false;
String name() default "";
String countName() default "";
}
复制代码例子
- 使用注解方式分组查询
跟正常写sql语句相同,将sql写到value中,并且nativeQuery = true。下面例子是根据task_id进行分组查询task集合
@Query(value = "select task_id from task group by task_id", nativeQuery = true)
List<Task> queryByGroup();
复制代码- 使用注解方式排序
根据task_id进行排序查询task集合
@Query(value = "select task_id,task_date from task order by task_id", nativeQuery = true)
List<Task> queryOrder();
复制代码- 使用注解方式条件查询
条件查询时可以使用字段名 操作符 ?;例如:task_date >= ?,使用位置匹配?。也可以使用字段名 操作符 :属性名;例如:task_date >= :startDate,使用属性名匹配,推荐使用后者,如果字段顺序修改,不影响匹配结果。下面是根据任务时间(task_date)段内和未被删除(deleted)的任务
@Query(value = "select task_id,task_date from task where task_date >=? and task_date <=? and deleted=0 ", nativeQuery = true)
List<ApptTask> queryDate(@Param("startDate") String startDate, @Param("endDate") String endDate);
复制代码- 使用注解方式修改
修改一条数据需要加上@Modifying用于标识是修改操作,默认事务等级是只读,所以还需要加上@Transactional,这样覆盖了默认的@Transactional才可以执行修改操作。下面是根据task_id更新task表的备注信息
@Transactional(rollbackOn = Exception.class)
@Modifying
@Query(value = "update task set remark = ? where task_id=?", nativeQuery = true)
void updateRemark(@Param("remark") String remark, @Param("taskId") String taskId);
复制代码多表联查,且多条件、分页查询怎么写?
复杂的查询需要注意,以下使用一个Mysql的多表联查的例子来说明复杂的查询要怎么写。下面是user表task表关联查询出任务名称、任务ID、用户名称这些信息,并且根据task_name、task_date进行过滤;根据task_date倒序。
共有几点需要注意:
- 多表联查使用正常的JOIN就可以
- 多条件是常见的情况,需要区别传入的条件是否要去执行,这种情况需要使用where 1=1 and 这种方式来保证条件不传时仍然正常查询。
- 分页查询需要传入分页参数Pageable,并且写countQuery来统计总数。
- 多条件查询关键:if(:参数!='',k.字段名 =:参数,1=1),这里是使用了if进行判断,这个写法类似Mybatis xml中的<if>标签。if的含义是代表传入的参数如果不为""(Spring类型空是""而不是null)将参数传入,如果为空时显示1=1 代表参数为真,对查询结果不产生作用。
代码:
@Query(value =
" select a.task_name, a.task_id,u.user_name" +
" from task a " +
" LEFT JOIN usert u" +
" ON a.user_id = u.user_id" +
" where a.deleted=0 " +
" AND if(:taskName!='',a.task_name =:taskName,1=1)" +
" AND if(:startDate!='',a.task_date >=:startDate,1=1)" +
" AND if(:endDate!='',a.task_date <=:endDate,1=1)" +
" order by a.task_date desc"
,
nativeQuery = true
,
countQuery =
" select count(*)" +
" from task a " +
" LEFT JOIN usert u" +
" ON a.user_id = u.user_id" +
" where a.deleted=0 " +
" AND if(:taskName!='',a.task_name =:taskName,1=1)" +
" AND if(:startDate!='',a.task_date >=:startDate,1=1)" +
" AND if(:endDate!='',a.task_date <=:endDate,1=1)" +
" order by a.task_date desc")
Page<Map<String,Object>> queryUserTaskPage(@Param("taskName") String taskName, @Param("startDate") LocalDate startDate, @Param("endDate") LocalDate endDate, @Param("pageable") Pageable pageable);
复制代码小结
以上列举了两种JPA的crud方式,jpa方法与注解方式,平时写代码时更倾向于使用注解方式去写原生sql来实现业务。对于简单查询可以用JpaRepository里面这些方法就够用了,对于更复杂的场景推荐使用@Query写sql的方式来实现。
jpa方法可以屏蔽底层的sql,如果有不同数据库实现的服务,用jpa方法可以免于修改sql。但是jpa方法对于分组查询、limit支持、多条件、多表联查这些不太友好。
小伙伴们有兴趣想了解内容和更多相关学习资料的请点赞收藏+评论转发+关注我,后面会有很多干货。我有一些面试题、架构、设计类资料可以说是程序员面试必备!所有资料都整理到网盘了,需要的话欢迎下载!私信我回复【666】即可免费获取
.
作者:斗笠小熊猫
原文出处:
https://juejin.cn/post/7106487853126778910
相关推荐
- 系统门窗一线品牌有哪些(中国系统门窗一线品牌有哪些)
-
轩尼斯门窗轩尼斯门窗创立于2004年,总部植根于佛山,生产基地在广东江苏共有4个,分别在凤岗厂区、盐城厂区、官窑厂区和范湖厂区;轩尼斯门窗在创建伊始就自主研发出了首款断桥铝门窗,产品技术专利超100项...
- 笔记本显卡天梯图排行榜(笔记本显卡天梯图最新版)
-
1、华硕ASUSROG-STRIX-RTX3080-O10G-GAMING1440-1935MHz华硕RTX3080被称为猛禽显卡,它将ROGSTRIX的轴流风扇进一步升级,扇叶数量也进行了...
- 惠普打印机售后维修官网(惠普打印机官方维修点查询)
-
是位于唐河县城区的惠普授权服务中心。这家维修点提供惠普打印机的售后服务和维修,包括故障排查、维修、更换零部件等。他们的技术人员都经过惠普的专业培训,能够提供高质量的服务。如果您的惠普打印机出现了问题,...
- 万能网卡驱动下载win11(万能网卡驱动windows7版2018最新版)
-
在windows11系统中点击桌面下方的开始图标,打开设置页面鼠标点击选择设备管理器选项。找到其中的网络适配器功能。右键选择显卡,点击卸载设备按钮等待卸载完成后重新安装驱动并重启计算机设备即可。想要修...
- cpu总是100使用率怎么回事呢
-
CPU占用率100%可能有多种原因。以下是一些可能的原因:驱动没有经过认证,这可能导致CPU资源占用100%。杀毒软件可能会占用大量的CPU资源,因为它们需要实时监控网页、邮件、个人隐私等功能。病毒或...
- 苹果6怎么升级系统版本(苹果6怎么升级系统版本最高能到多少)
-
要是喜欢自己动手可以自己去官网上下载,新系统在更新,他只是不支持自动更新的。如果不想自己动手,可以去拼多多或者是淘宝里面找一找,多的是那种帮你刷新系统的。也就是一点点钱的事情。现在选择很多活人不会被尿...
- windows2003镜像32位下载(win2003系统镜像)
-
虚拟光驱装系统,(win7,xp通用)具体步骤一、将从网上下载的win7旗舰版ISO系统文件存放到D盘。二、从网上下载虚拟光驱,打开安装后在任务栏右通知区显示“虚拟DAEMON管理器”图标,在我的电脑...
- win10电脑自动更新怎么关闭(win10电脑怎么关闭自动更新系统)
-
win10老推送win11打开的方法步骤如下,1,首先,打开设置,点击更新和安全2,打开后,点击windows预览体验计划3,打开后,点击开始4,然后按流程进行注册5,注册完成后,点击选择帐户6,然后...
- window7下载steam(window7下载一键重装如何恢复网络)
-
回答如下:要在Windows7上下载Steam,您可以按照以下步骤操作:1.打开您的浏览器,访问Steam官网(https://store.steampowered.com)。2.点击页面右上角...
- 系统还原没有还原点怎么办(系统还原点不动怎么办)
-
如果电脑没有创建还原点,就不能使用系统还原来回到之前的状态。但是,可以尝试使用其他备份工具或软件来恢复数据或重建系统。比如,可以使用第三方备份软件来备份重要文件和数据。如果是系统出现问题,可以尝试重新...
- 正在准备windows(正在准备windows请勿关机怎么办)
-
这个情况在使用华为电脑时可能会遇到。一般来说,这是因为电脑正在进行系统更新或者安装软件程序等操作,导致启动时间较长。如果电脑显示“正在准备Windows,请勿关闭电源”,则说明电脑正在进行系统更新。...
- 一周热门
-
-
飞牛OS入门安装遇到问题,如何解决?
-
如何在 iPhone 和 Android 上恢复已删除的抖音消息
-
Boost高性能并发无锁队列指南:boost::lockfree::queue
-
大模型手册: 保姆级用CherryStudio知识库
-
用什么工具在Win中查看8G大的log文件?
-
如何在 Windows 10 或 11 上通过命令行安装 Node.js 和 NPM
-
威联通NAS安装阿里云盘WebDAV服务并添加到Infuse
-
Trae IDE 如何与 GitHub 无缝对接?
-
idea插件之maven search(工欲善其事,必先利其器)
-
如何修改图片拍摄日期?快速修改图片拍摄日期的6种方法
-
- 最近发表
- 标签列表
-
- python判断字典是否为空 (50)
- crontab每周一执行 (48)
- aes和des区别 (43)
- bash脚本和shell脚本的区别 (35)
- canvas库 (33)
- dataframe筛选满足条件的行 (35)
- gitlab日志 (33)
- lua xpcall (36)
- blob转json (33)
- python判断是否在列表中 (34)
- python html转pdf (36)
- 安装指定版本npm (37)
- idea搜索jar包内容 (33)
- css鼠标悬停出现隐藏的文字 (34)
- linux nacos启动命令 (33)
- gitlab 日志 (36)
- adb pull (37)
- python判断元素在不在列表里 (34)
- python 字典删除元素 (34)
- vscode切换git分支 (35)
- python bytes转16进制 (35)
- grep前后几行 (34)
- hashmap转list (35)
- c++ 字符串查找 (35)
- mysql刷新权限 (34)
