Springboot+thymeleaf项目之@Query注解实现复杂查询

需求分析

现有一张表,字段有id,dept_name(部门名称),exit_or_enter(进或是出),staff_name(员工名称)

备注:exit_or_enter 进门为1,出门为2

需要统计每个部门 “进门”的人数(即 exit_or_enter=1)以及对应的人员名
在这里插入图片描述

为了方便理解,先来看一下最后的效果

字段dept_name(部门),num(“进”部门的人数),names(部门人员名集合)
在这里插入图片描述

自定义SQL语句

如果用JPA常用的查询语句,似乎不能完成这一复杂的查询。所以我们为了达到这一目标,需要自定义SQL语句

分析一下需求,要完成这一查询,或许要用到多表连接,这里我使用的是“LEFT JOIN ON”。

表a

首先查询所有的部门,作为表a

 (select distinct dept_name as dept_name from staff) a

看一下查询结果,一共4个部门,如图。

在这里插入图片描述

表b

再统计每个部门“进门”的人数和对应的人员名集合,作为表b

( select dept_name,count(*) as num,GROUP_CONCAT(staff_name) as names from staff where exit_or_enter=1 GROUP BY dept_name) b

COUNT(*) 很好理解,合计一下。问题在合并人员名,形成一个集合。

由于我这里使用的是MySQL数据库,所以我们要用到函数GROUP_CONCAT(), 功能是合并列值 将一列的多个值合并成一行

这样,整条SQL语句就很好理解了。看一下统计结果,一共3条记录,如图。

在这里插入图片描述

对比一下我们预期的效果,少了一个部门“运营”,虽然sum是0,names是空,但是我们为了之后的展示,仍要统计。

连接表a和表b

select a.dept_name,COALESCE(b.num,0) as num,COALESCE(b.names,'空') as names from
	 (select distinct dept_name from staff) a
	left join 
	( select dept_name,count(*) num,GROUP_CONCAT(staff_name) as names from staff where exit_or_enter=1 GROUP BY dept_name) b
	on a.dept_name=b.dept_name

使用LEFT JOIN ON 将两表连接,连接条件是a.dept_name=b.dept_name

LEFT JOIN学习链接: https://www.w3school.com.cn/sql/sql_join_left.asp.

根据列名“dept_name”将两个表连接成一个表之后,我们再对这个表进行查询,查询a表的dept_name,b表的num和names

然后这里用到一个函数COALESCE()
这个函数主要用来进行空值处理,其参数格式如下:
COALESCE ( expression,value1,value2……,valuen)
COALESCE()函数的第一个参数expression为待检测的表达式,而其后的参数个数不定。
COALESCE()函数将会返回包括expression在内的所有参数中的第一个非空表达式。
如果expression不为空值则返回expression;否则判断value1是否是空值,
如果value1不为空值则返回value1;否则判断value2是否是空值,
如果value2不为空值则返回value2;……以此类推,
如果所有的表达式都为空值,则返回NULL。

因此COALESCE(b.num,0)的意思就是处理表b字段num的空值
如果不是空值,就返回表达式的值;如果是空值即num=0,就返回value=0

所以,COALESCE(b.names,‘空’)的意思就是处理表b字段names的空值
如果不是空值,就返回表达式的值;如果是空值即names=null,就返回value=‘空’

查询结果如图,正确,达到预期目标!

在这里插入图片描述

至此,SQL语句就写好啦!

使用@Query注解

Entity层

package com.example.thymedemo.entity;

import javax.persistence.*;

@Entity(name = "staff")
public class StaffInfo {
    @GeneratedValue(strategy= GenerationType.IDENTITY)
    @Id
    private int id;
    private String deptName;
    private String staffName;
    private int exit_or_enter;

    public int getId(){
        return id;
    }
    public void setId(int id){
        this.id = id;
    }
    public String getDeptName() {
        return deptName;
    }
    public void setDeptName(String deptName) {
        this.deptName = deptName;
    }
    public String getStaffName() {
        return staffName;
    }
    public void setStaffName(String staffName) {
        this.staffName = staffName;
    }
    public int getExit_or_enter() { return exit_or_enter; }
    public void setExit_or_enter(int exit_or_enter) { this.exit_or_enter = exit_or_enter; }
}

Dao层

package com.example.thymedemo.dao;

import com.example.thymedemo.entity.StaffInfo;
import org.springframework.data.jpa.repository.JpaRepository;
import org.springframework.data.jpa.repository.Query;
import java.util.List;

public interface StaffInfoDao extends JpaRepository<StaffInfo, Integer> {

    @Query(value = "select a.dept_name,COALESCE(b.num,0) as num,COALESCE(b.names,'空') as names from\n" +
            "\t (select distinct dept_name from staff) a\n" +
            "\tleft join ( select dept_name,count(*) num,GROUP_CONCAT(staff_name) as names from staff where exit_or_enter=1 GROUP BY dept_name) b\n" +
            "\ton a.dept_name=b.dept_name",nativeQuery = true
    )
    List<Object[]> findContent();
}

@Query(value=" ",nativeQuery = true),value里面复制粘贴我们刚才整理好的SQL语句

nativeQuery = true,是可以执行原生sql语句,所谓原生sql,也就是说这段sql拷贝到数据库中,然后把参数值给一下就能运行了

然后定义一个方法 findContent(),准备调用

Controller层

package com.example.thymedemo.controller;

import com.example.thymedemo.dao.StaffInfoDao;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Controller;
import org.springframework.ui.Model;
import org.springframework.web.bind.annotation.RequestMapping;
import java.util.List;

@Controller
public class StaffInfoController {
    @Autowired
    StaffInfoDao staffInfoDao;

    @RequestMapping("/count")
    public String count(Model model){

        List<Object[]> data = staffInfoDao.findContent();
        model.addAttribute("data",data);

        return "count";
    }
}

调用方法 findContent(),再用model.addAttribute()向thymeleaf页面传递数据,返回网页“count”即可。

网页效果演示

在这里插入图片描述
大功告成!

总结

主要是复杂SQL查询语句的编写,然后用@Query注解去完成查询。

源码地址

源码Github地址: https://github.com/Huge-Hammer/JPAlearn

帮我点个小星星哦

我是爱学习的诸葛铁锤,觉得有用的话点个赞哈,啾咪

  • 1
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
### 回答1: springboot+thymeleaf项目是一种基于Java语言开发的Web应用程序。它采用了Spring Boot框架和Thymeleaf模板引擎,可以快速地搭建一个高效、可靠、易于维护的Web应用程序。该项目具有以下特点: 1. 简单易用:Spring Boot框架提供了一系列的自动化配置,使得开发者可以快速地搭建一个Web应用程序,而不需要过多的配置。 2. 高效可靠:Spring Boot框架采用了一系列的优化措施,使得应用程序具有高效、可靠的性能。 3. 易于维护:Thymeleaf模板引擎提供了一种简单、易于维护的模板语言,使得开发者可以快速地开发出具有良好可读性的Web应用程序。 总之,springboot+thymeleaf项目是一种非常优秀的Web应用程序开发框架,可以帮助开发者快速地开发出高效、可靠、易于维护的Web应用程序。 ### 回答2: Spring Boot是一个基于Spring框架的快速开发框架,这个框架的优点在于其简单易用,能够快速搭建一个Java Web应用程序,无需进行复杂的配置和繁琐的XML文件编写。而Thymeleaf则是一种Web和HTML的模板引擎,可以方便地处理文本、链接和表单等元素,支持多重继承和页面片段的复用等特性。 Spring BootThymeleaf的结合,可以帮助开发人员更加简便地搭建Web应用程序。在使用Spring Boot进行项目开发时,可以使用Thymeleaf来完成Web开发的视图层,进行模版板的渲染和数据绑定。这样就可以很直接地将数据通过模板引擎展现出来,且更加方便。 在一个Spring Boot Thymeleaf项目的构建中,需要进行如下步骤: 1. 首先,引入Spring BootThymeleaf的依赖以及其他必要的依赖,例如web和mybatis等相关组件。 2. 创建一个Controller类,并使用@Controller注解将类标记为Controller,编写具体的Action方法,这些方法可以用@RequestMapping或@GetMapping等注解来定义处理请求的URL路径和请求类型等相关信息。 3. 创建一个Model类,用于封装需要传输到前端的数据和相关操作等。 4. 在Controller内部设置Model变量并将相关数据注入Model,然后将需要展现的数据作为参数传递给Thymeleaf进行渲染,最后将渲染完成后的结果返回给前端页面展现。 5. 编写HTML页面,使用Thymeleaf标签来渲染动态数据。 需要注意的是,在进行Thymeleaf模板的渲染时,需要遵守一定的规范,例如页面中的数据变量名称需与Model中的属性名称一致,引入Thymeleaf命名空间等等。 总之,Spring BootThymeleaf结合使用可以帮助开发人员快速地完成Web开发,整个过程简单而且高效。使用Thymeleaf能够降低模版制作的门槛,进一步提高开发效率,并且能够提供丰富的模版处理标签,使得页面制作更加灵活。 ### 回答3: 近年来,使用SpringBootThymeleaf进行Web开发已经成为越来越多的开发者选择的方案。SpringBoot是一个基于Spring框架的快速Web应用开发框架,而Thymeleaf是一种基于HTML的模板引擎,其中需要了解的内容包括以下几点: 首先,SpringBoot框架的优点是非常明显的。它提供了很多便于使用的方法,例如自动装配,以及基于配置的许多默认值。这使得开发者可以花更少的时间和精力来开发项目,将重点放在业务逻辑和功能实现上。 其次,Thymeleaf是一种非常强大和灵活的模板引擎,其语法简单易懂,而且支持HTML5标准。它还提供了一些样式和布局的工具,以及易于使用的表达式和标签,使得Web页面开发更加容易。 当然,SpringBoot集成Thymeleaf的过程也并不复杂。只需添加thymeleaf-starter包依赖,SpringBoot将自动将Thymeleaf注册为默认的模板引擎。然后,您只需要编写Thymeleaf模板文件即可。 最后,值得注意的是,使用SpringBootThymeleaf进行Web开发的好处在于它们之间的紧密集成。这种紧密集成可以更轻松地创建动态和交互性的Web应用程序,这是传统的HTML和JavaScript不能提供的。 总的来说,SpringBootThymeleaf是一对非常强大且易于使用的Web开发工具组合,它们的出现大大提高了Web开发的效率和质量,同时也为开发人员提供了更好的开发体验。我们相信,这对于Web开发者来说是非常有价值的组合。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值