JPA查询

jpa常用的查询方式

jpa官方文档

https://docs.spring.io/spring-data/jpa/docs/2.1.5.RELEASE/reference/html/#jpa.query-methods.query-creation

项目地址:
https://github.com/haipengliang/jpaDemo

1.通过方法名来创建查询

  1. 使用字段的值进行条件查询,如Is,In,Like, Between等等。

  2. 使用and ,or 来进行条件拼接。

  3. 利用Pageable接口来进行分页,Sort 对象排序。

  4. 利用Distinct去重

  5. 利用OrderBy进行排序

  6. 利用 Top 和 First来获取限制数据

    查询方法的结果可以通过使用first或top关键字来限制,可以互换使用。也可以附加一个可选的数值,top或者first指定要返回的最大结果大小。如果省略该数字,该数字默认为1。

  7. 流式查询结果。jpa可以使用Java 8 Stream<T>作为返回类型的方式处理查询结果。但是需要在只读事物环境下进行。

    Stream<Account> readAllByAgeIs(int i);
  1. 异步查询
    @Async
    Future<User> findByFirstname(String firstname);               
    
    @Async
    CompletableFuture<User> findOneByFirstname(String firstname); 
    
    @Async
    ListenableFuture<User> findOneByLastname(String lastname);    
  1. 可以通过@Lock(LockModeType.READ)来设置jpa锁

    乐观锁:
    1、OPTIMISTIC:它和READ锁模式相同,JPA 2.0仍然支持READ锁模式,但明确指出在新应用程序中推荐使用OPTIMISTIC。

    2、OPTIMISTIC_FORCE_INCREMENT:它和WRITE锁模式相同,JPA 2.0仍然支持WRITE锁模式,但明确指出在新应用程序中推荐使用OPTIMISTIC_FORCE_INCREMENT。

    悲观锁模式:
    1、PESSIMISTIC_READ:只要事务读实体,实体管理器就锁定实体,直到事务完成锁才会解开,当你想使用重复读语义查询数据时使用这种锁模式,换句话说就是,当你想确保数据在连续读期间不被修改,这种锁模式不会阻碍其它事务读取数据。

    2、PESSIMISTIC_WRITE:只要事务更新实体,实体管理器就会锁定实体,这种锁模式强制尝试修改实体数据的事务串行化,当多个并发更新事务出现更新失败几率较高时使用这种锁模式。

    3、PESSIMISTIC_FORCE_INCREMENT:当事务读实体时,实体管理器就锁定实体,当事务结束时会增加实体的版本属性,即使实体没有修改。

2.@Query

  1. @Query不仅仅可以查询,也可以进行增删改的操作

  2. 查询操作直接使用@Query就可以,增删改操作需要添加@Modifying以及在事务环境下运行,可以直接添加 @Transactional。

  3. @Query查询同样可以使用Sort,Pageable对象进行分页排序

  4. @Query可以使用 JPQL,原生sql,SpEL表达式

  5. JPQL

    JPQL就是一种查询语言,具有与 SQL 相类似的特征, JPQL 是完全面向对象的,具备继承、多态和关联等特性,和hibernate HQL很相似。

    JPQL 语句支持两种方式的参数定义方式 : 命名参数和位置参数 。在同一个查询语句中只允许使用一种参数定义方式。

    命令参数的格式为: : + 参数名 ,命令参数一般需要使用@Param来定义参数名

    位置参数的格式为: ?+ 位置编号

  1. 原生sql

    jpa也可以使用原生sql,但是需要在@Query 中添加nativeQuery = true的参数

    而且,使用原声sql无法分页,所以需要添加如下

  @Query(value = "SELECT * FROM USERS WHERE LASTNAME = ?1",
    countQuery = "SELECT count(*) FROM USERS WHERE LASTNAME = ?1",
    nativeQuery = true)
  Page<User> findByLastname(String lastname, Pageable pageable);
  1. SpEL表达式

    5.1 从Spring Data JPA 1.4版本开始,支持在@Query定义的查询中使用SpEL模板表达式。在执行查询的时候,这些表达式通过一个事先定义好的变量集合求值。Spring Data JPA支持成为entityName的变量。它的用法是select x from #{#entityName} x。它会将域类型的entityName与给定repository关联起来。entityName按照如下方式处理:如果域类型在@Entity注解上设置了name属性,则使用该属性。否则直接使用域类型的类名。

    5.2 一般使用是使用#{#entityName}代替实体名,这样你改动实体名,也不会对查询造成影响

    另一个#{#entityName}表达式的使用场景是如果你想定义一个通用的repository接口和一个用于具体域类型的指定repository接口。为了避免重复定义具体接口中的自定义查询方法,你可以在通用接口中使用实体名称表达式:

    我们系统来说,就可以使用这个定义一个关于sellerId 和 isdelete的通用查询接口。

3.Example

  1. Example查询的组成

    Probe: 含有对应字段的实例对象。

    ExampleMatcher:ExampleMatcher携带有关如何匹配特定字段的详细信息,相当于匹配条件。设置包括 忽律,包含,为空,是否区分大小写,类型转换。

    Example:由Probe和ExampleMatcher组成,用于查询。

    主要限制:
    属性不支持嵌套或者分组约束,比如这样的查询 firstname = ?0 or (firstname = ?1 and lastname = ?2)
    灵活匹配只支持字符串类型,其他类型只支持精确匹配,而且不支持in查询。

  2. Example查询方式

     1.组成查询实体
     Account account = new Account();
     account.setAge(1);
     account.setFirstName("li");
     2.组成匹配方式
     ExampleMatcher exampleMatcher = ExampleMatcher.matching().withMatcher("firstName",e->e.contains());
     3.组合example
     Example<Account> example = Example.of(account,exampleMatcher);
     4.如需要分页和排序,则创建分页和排序
     Sort sort = new Sort(Sort.Direction.DESC,"age");
     PageRequest pageRequest = PageRequest.of(0,1,sort);
     5.查询
     Page<Account> all1 = repository.findAll(example, pageRequest);
    

4.Specification

  1. 简介

    由DDD之父 Eric Evans 和OO之父 Martin Fowler定义的Specification(规格模式)。Spring Data JPA已经帮助我们很大程度上简化了我们的查询操作,我们甚至只要写一个接口,然后单纯的写一些方法就可以完成各式各样的查询,但是对于我们程序设计人员而言,总希望所有的查询变得更加的简单方便,为了给程序人员进行再一次的封装,Spring Data JPA提供了Specification的方式进行查询。

5.querydsl

  1. 简介

    如果说Hibernate等ORM是JPA的实现,而SpringDataJPA是对JPA使用的封装,那么QueryDSL可以是与SpringDataJPA有着同阶层的级别,它也是基于各种ORM之上的一个通用查询框架,使用它的API类库可以写出“Java代码的sql”,不用去手动接触sql语句,表达含义却如sql般准确。更重要的一点,它能够构建类型安全的查询,这比起JPA使用原生查询时有很大的不同,我们可以不必再对恶心的“Object[]”进行操作了。当然,我们可以SpringDataJPA +QueryDSL.JPA联合使用,它们之间有着完美的相互支持,以达到更高效的编码。

  2. 就是生成了一个Q实体,使用这个Q实体,可以轻松的组合成各种个样的查询条件。

jpa使用实体做做参数

@Modifying
@Query("update CommunityEntity set status= :#{#entity.status} ,statusGroup= :#{#entity.statusGroup} where " +
"communityId= :#{#entity.communityId} and sellerId= :#{#entity.sellerId}  and isDeleted= :#{#entity.isDeleted}")
int updateCommunityStatus(@Param("entity") CommunityEntity entity);
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 225,337评论 6 524
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 96,560评论 3 406
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 172,632评论 0 370
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 61,219评论 1 303
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 70,219评论 6 401
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 53,670评论 1 316
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 42,018评论 3 431
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 41,000评论 0 280
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 47,552评论 1 326
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 39,565评论 3 347
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 41,692评论 1 355
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 37,280评论 5 351
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 43,009评论 3 341
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 33,435评论 0 25
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 34,587评论 1 277
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 50,276评论 3 383
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 46,752评论 2 367

推荐阅读更多精彩内容