热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

Spring Data Exists查询方法如何编写

本篇内容介绍了“Spring Data Exists查询方法如何编写”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小

本篇内容介绍了“Spring Data Exists查询方法如何编写”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!

    Spring Data Exists查询方法如何编写

    简介

    在这篇文章中,我将向你展示编写Spring Data Exists查询的最佳方法,从SQL的角度来看,它是高效的。

    在做咨询的时候,我遇到了几个常用的选项,而开发者却不知道其实还有更好的选择。

    领域模型

    让我们假设我们有以下Post 实体。

    Spring Data Exists查询方法如何编写

    slug 属性是一个业务键,意味着它有一个唯一的约束,为此,我们可以用下面的注解来注解它 @NaturalIdHibernate注解。

    @Entity
    @Entity
    @Table(
        name = "post",
        uniqueConstraints = @UniqueConstraint(
            name = "UK_POST_SLUG",
            columnNames = "slug"
        )
    )
    public class Post {
        @Id
        private Long id;
        private String title;
        @NaturalId
        private String slug;
        public Long getId() {
            return id;
        }
        public Post setId(Long id) {
            this.id = id;
            return this;
        }
        public String getTitle() {
            return title;
        }
        public Post setTitle(String title) {
            this.title = title;
            return this;
        }
        public Post setSlug(String slug) {
            this.slug = slug;
            return this;
        }
    }

    如何不使用Spring Data来写Exists查询?

    首先,让我们从各种方法开始,这些方法虽然很流行,但你最好避免使用。

    用findBy查询模拟存在

    Spring Data提供了一种从方法名派生查询的方法,所以你可以写一个findBy 查询来模拟存在,就像这样。

    @Repository
    public interface PostRepository 
            extends JpaRepository {
        Optional findBySlug(String slug);   
    }

    由于findBySlug 方法是用来获取Post 实体的,我见过这样的情况:这个方法被用来进行平等检查,就像下面的例子。

    assertTrue(
        postRepository.findBySlug(slug).isPresent()
    );

    这种方法的问题在于,实体的获取实际上只是为了检查是否有一个与所提供的过滤条件相关的记录。

    SELECT 
        p.id AS id1_0_,
        p.slug AS slug2_0_,
        p.title AS title3_0_
    FROM 
        post p
    WHERE 
        p.slug = 'high-performance-java-persistence'

    使用fidnBy 查询来获取实体以检查其存在性是一种资源浪费,因为如果你在slug 属性上有一个索引的话,你不仅不能使用覆盖查询,而且你必须通过网络将实体结果集发送到JDBC驱动程序,只是默默地将其丢弃。

    使用实例查询来检查存在性

    另一个非常流行的,但效率低下的检查存在性的方法是使用Query By Example功能。

    assertTrue(
        postRepository.exists(
            Example.of(
                new Post().setSlug(slug),
                ExampleMatcher.matching()
                    .withIgnorePaths(Post_.ID)
                    .withMatcher(Post_.SLUG, exact())
            )
        )
    );

    Query By Example功能建立了一个Post 实体,在匹配所提供的ExampleMatcher 规范给出的属性时,该实体将被用作参考。

    当执行上述Query By Example方法时,Spring Data会生成与之前findBy 方法所生成的相同的SQL查询。

    SELECT 
        p.id AS id1_0_,
        p.slug AS slug2_0_,
        p.title AS title3_0_
    FROM 
        post p
    WHERE 
        p.slug = 'high-performance-java-persistence'

    虽然Query By Example功能对于获取实体可能很有用,但是将其与Spring Data JPA的exists 通用方法Repository ,效率并不高。

    如何使用Spring Data编写Exists查询

    有更好的方法来编写Spring Data Exists查询。

    用existsBy查询方法检查存在性

    Spring Data提供了一个existsBy 查询方法,我们可以在PostRepository ,定义如下。

    @Repository
    public interface PostRepository 
            extends JpaRepository {
        boolean existsBySlug(String slug);
    }

    当在PostgreSQL或MySQL上调用existsBySlug 方法时。

    assertTrue(
        postRepository.existsBySlug(slug)
    );

    Spring Data会生成以下SQL查询。

    SELECT 
        p.id AS col_0_0_
    FROM 
        post p
    WHERE 
        p.slug = 'high-performance-java-persistence'
    LIMIT 1

    这个查询的PostgreSQL执行计划看起来如下。

    Limit  
        (cost=0.28..8.29 rows=1 width=8) 
        (actual time=0.021..0.021 rows=1 loops=1)
      ->  Index Scan using uk_post_slug on post p  
          (cost=0.28..8.29 rows=1 width=8) 
          (actual time=0.020..0.020 rows=1 loops=1)
            Index Cond: ((slug)::text = 'high-performance-java-persistence'::text)
    Planning Time: 0.088 ms
    Execution Time: 0.033 ms

    还有,MySQL的,像这样。

    -> Limit: 1 row(s)  
       (cost=0.00 rows=1) 
       (actual time=0.001..0.001 rows=1 loops=1)
        -> Rows fetched before execution  
           (cost=0.00 rows=1) 
           (actual time=0.000..0.000 rows=1 loops=1)

    所以,这个查询非常快,而且额外的LIMIT 操作并不影响性能,因为它反正是在一个记录的结果集上完成。

    用COUNT SQL查询来检查存在性

    模拟存在性的另一个选择是使用COUNT查询。

    @Repository
    public interface PostRepository 
            extends JpaRepository {
        @Query(value = """
            select count(p.id) = 1 
            from Post p
            where p.slug = :slug
            """
        )
        boolean existsBySlugWithCount(@Param("slug") String slug);
    }

    COUNT 查询在这种特殊情况下可以正常工作,因为我们正在匹配一个UNIQUE列值。

    然而,一般来说,对于返回有多条记录的结果集的查询,你应该倾向于使用EXISTS ,而不是COUNT ,正如Lukas Eder在这篇文章中所解释的那样。

    在PostgreSQL和MySQL上调用existsBySlugWithCount 方法时。

    assertTrue(
        postRepository.existsBySlugWithCount(slug)
    );

    Spring Data会执行以下SQL查询。

    SELECT 
        count(p.id) > 0 AS col_0_0_
    FROM 
        post p
    WHERE 
        p.slug = 'high-performance-java-persistence'

    而且,这个查询的PostgreSQL执行计划看起来如下。

    Aggregate  
      (cost=8.29..8.31 rows=1 width=1) 
      (actual time=0.023..0.024 rows=1 loops=1)
      ->  Index Scan using uk_post_slug on post p  
          (cost=0.28..8.29 rows=1 width=8) 
          (actual time=0.019..0.020 rows=1 loops=1)
            Index Cond: ((slug)::text = 'high-performance-java-persistence'::text)
    Planning Time: 0.091 ms
    Execution Time: 0.044 ms

    而在MySQL上。

    -> Aggregate: count('1')  
       (actual time=0.002..0.002 rows=1 loops=1)
        -> Rows fetched before execution  
           (cost=0.00 rows=1) 
           (actual time=0.000..0.000 rows=1 loops=1)

    尽管COUNT操作有一个额外的Aggregate步骤,但由于只有一条记录需要计算,所以这个步骤非常快。

    用CASE WHEN EXISTS SQL查询来检查存在性

    最后一个模拟存在的选项是使用CASE WHEN EXISTS本地SQL查询。

    @Repository
    public interface PostRepository 
            extends JpaRepository {
        @Query(value = """
            SELECT 
                CASE WHEN EXISTS (
                    SELECT 1 
                    FROM post 
                    WHERE slug = :slug
                ) 
                THEN 'true' 
                ELSE 'false'
                END
            """,
            nativeQuery = true
        )
        boolean existsBySlugWithCase(@Param("slug") String slug);
    }

    而且,我们可以像这样调用existsBySlugWithCase 方法。

    assertTrue(
        postRepository.existsBySlugWithCase(slug)
    );

    这个查询的PostgreSQL执行计划看起来如下。

    Result  
      (cost=8.29..8.29 rows=1 width=1) 
      (actual time=0.021..0.022 rows=1 loops=1)
      InitPlan 1 (returns $0)
        ->  Index Only Scan using uk_post_slug on post  
              (cost=0.27..8.29 rows=1 width=0) 
              (actual time=0.020..0.020 rows=1 loops=1)
              Index Cond: (slug = 'high-performance-java-persistence'::text)
              Heap Fetches: 1
    Planning Time: 0.097 ms
    Execution Time: 0.037 ms

    而在MySQL上。

    -> Rows fetched before execution  
       (cost=0.00 rows=1) 
       (actual time=0.000..0.000 rows=1 loops=1)
    -> Select #2 (subquery in projection; run only once)
        -> Limit: 1 row(s)  
            (cost=0.00 rows=1) 
            (actual time=0.000..0.001 rows=1 loops=1)
            -> Rows fetched before execution  
               (cost=0.00 rows=1) 
               (actual time=0.000..0.000 rows=1 loops=1)

    所以,这和之前的LIMITCOUNT 查询一样快。在其他数据库上,你可能要检查一下,看看是否有什么不同。

    “Spring Data Exists查询方法如何编写”的内容就介绍到这里了,感谢大家的阅读。如果想了解更多行业相关的知识可以关注编程笔记网站,小编将为大家输出更多高质量的实用文章!


    推荐阅读
    • Spring特性实现接口多类的动态调用详解
      本文详细介绍了如何使用Spring特性实现接口多类的动态调用。通过对Spring IoC容器的基础类BeanFactory和ApplicationContext的介绍,以及getBeansOfType方法的应用,解决了在实际工作中遇到的接口及多个实现类的问题。同时,文章还提到了SPI使用的不便之处,并介绍了借助ApplicationContext实现需求的方法。阅读本文,你将了解到Spring特性的实现原理和实际应用方式。 ... [详细]
    • 本文讨论了一个关于cuowu类的问题,作者在使用cuowu类时遇到了错误提示和使用AdjustmentListener的问题。文章提供了16个解决方案,并给出了两个可能导致错误的原因。 ... [详细]
    • 本文介绍了一个在线急等问题解决方法,即如何统计数据库中某个字段下的所有数据,并将结果显示在文本框里。作者提到了自己是一个菜鸟,希望能够得到帮助。作者使用的是ACCESS数据库,并且给出了一个例子,希望得到的结果是560。作者还提到自己已经尝试了使用"select sum(字段2) from 表名"的语句,得到的结果是650,但不知道如何得到560。希望能够得到解决方案。 ... [详细]
    • springmvc学习笔记(十):控制器业务方法中通过注解实现封装Javabean接收表单提交的数据
      本文介绍了在springmvc学习笔记系列的第十篇中,控制器的业务方法中如何通过注解实现封装Javabean来接收表单提交的数据。同时还讨论了当有多个注册表单且字段完全相同时,如何将其交给同一个控制器处理。 ... [详细]
    • 标题: ... [详细]
    • SpringMVC接收请求参数的方式总结
      本文总结了在SpringMVC开发中处理控制器参数的各种方式,包括处理使用@RequestParam注解的参数、MultipartFile类型参数和Simple类型参数的RequestParamMethodArgumentResolver,处理@RequestBody注解的参数的RequestResponseBodyMethodProcessor,以及PathVariableMapMethodArgumentResol等子类。 ... [详细]
    • 本文详细介绍了使用C#实现Word模版打印的方案。包括添加COM引用、新建Word操作类、开启Word进程、加载模版文件等步骤。通过该方案可以实现C#对Word文档的打印功能。 ... [详细]
    • 解决java.lang.IllegalStateException: ApplicationEventMulticaster not initialized错误的方法和原因
      本文介绍了解决java.lang.IllegalStateException: ApplicationEventMulticaster not initialized错误的方法和原因。其中包括修改包名、解决service name重复、处理jar包冲突和添加maven依赖等解决方案。同时推荐了一个人工智能学习网站,该网站内容通俗易懂,风趣幽默,值得一看。 ... [详细]
    • 如何使用Java获取服务器硬件信息和磁盘负载率
      本文介绍了使用Java编程语言获取服务器硬件信息和磁盘负载率的方法。首先在远程服务器上搭建一个支持服务端语言的HTTP服务,并获取服务器的磁盘信息,并将结果输出。然后在本地使用JS编写一个AJAX脚本,远程请求服务端的程序,得到结果并展示给用户。其中还介绍了如何提取硬盘序列号的方法。 ... [详细]
    • 《数据结构》学习笔记3——串匹配算法性能评估
      本文主要讨论串匹配算法的性能评估,包括模式匹配、字符种类数量、算法复杂度等内容。通过借助C++中的头文件和库,可以实现对串的匹配操作。其中蛮力算法的复杂度为O(m*n),通过随机取出长度为m的子串作为模式P,在文本T中进行匹配,统计平均复杂度。对于成功和失败的匹配分别进行测试,分析其平均复杂度。详情请参考相关学习资源。 ... [详细]
    • 本文介绍了Android 7的学习笔记总结,包括最新的移动架构视频、大厂安卓面试真题和项目实战源码讲义。同时还分享了开源的完整内容,并提醒读者在使用FileProvider适配时要注意不同模块的AndroidManfiest.xml中配置的xml文件名必须不同,否则会出现问题。 ... [详细]
    • 本文介绍了在处理不规则数据时如何使用Python自动提取文本中的时间日期,包括使用dateutil.parser模块统一日期字符串格式和使用datefinder模块提取日期。同时,还介绍了一段使用正则表达式的代码,可以支持中文日期和一些特殊的时间识别,例如'2012年12月12日'、'3小时前'、'在2012/12/13哈哈'等。 ... [详细]
    • Python爬虫中使用正则表达式的方法和注意事项
      本文介绍了在Python爬虫中使用正则表达式的方法和注意事项。首先解释了爬虫的四个主要步骤,并强调了正则表达式在数据处理中的重要性。然后详细介绍了正则表达式的概念和用法,包括检索、替换和过滤文本的功能。同时提到了re模块是Python内置的用于处理正则表达式的模块,并给出了使用正则表达式时需要注意的特殊字符转义和原始字符串的用法。通过本文的学习,读者可以掌握在Python爬虫中使用正则表达式的技巧和方法。 ... [详细]
    • 本文介绍了Oracle存储过程的基本语法和写法示例,同时还介绍了已命名的系统异常的产生原因。 ... [详细]
    • 使用圣杯布局模式实现网站首页的内容布局
      本文介绍了使用圣杯布局模式实现网站首页的内容布局的方法,包括HTML部分代码和实例。同时还提供了公司新闻、最新产品、关于我们、联系我们等页面的布局示例。商品展示区包括了车里子和农家生态土鸡蛋等产品的价格信息。 ... [详细]
    author-avatar
    可爱鬼猫_380
    这个家伙很懒,什么也没留下!
    PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
    Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有