Spring - 10 ( 9000 字 Spring 入门级教程 )

news2025/2/24 21:00:54

一:MyBatis 进阶

动态 SQL 是 Mybatis 的强大特性之⼀,能够完成不同条件下不同的 sql 拼接。

1.1 if 标签

在注册用户的时候,可能会有这样⼀个问题,如下图所示:

在这里插入图片描述

注册分为两种字段:必填字段和非必填字段,那如果在添加用户的时候有不确定的字段传⼊,程序应该如何实现呢

这个时候就需要使用动态标签来判断了,比如添加的时候性别 gender 为非必填字段,具体实现如下:

  1. 接口定义:
Integer insertUserByCondition(UserInfo userInfo);
  1. Mapper.xml 实现:
<insert id="insertUserByCondition">
	INSERT INTO userinfo (
	username,
	`password`,
	age,
	<if test="gender != null">
		gender,
	</if>
	phone)
	VALUES (
	#{username},
	#{age},
	<if test="gender != null">
		#{gender},
	</if>
	#{phone})
</insert>

或者使用注解方式(不推荐)

@Insert("<script>" +
	"INSERT INTO userinfo (username,`password`,age," +
	"<if test='gender!=null'>gender,</if>" +
	"phone)" +
	"VALUES(#{username},#{age}," +
	"<if test='gender!=null'>#{gender},</if>" +
	"#{phone})"+
	"</script>")
Integer insertUserByCondition(UserInfo userInfo);

把上面 SQL (包括标签), 使用 < script >< /script > 标签括起来就可以

1.2 trim 标签

之前的插入用户功能,只是有⼀个 gender 字段可能是选填项,如果有多个字段,⼀般考虑使用标签结合标签,对多个字段都采取动态生成的方式。

标签中有如下属性:

  • prefix:表示整个语句块,以prefix的值作为前缀
  • suffix:表示整个语句块,以suffix的值作为后缀
  • prefixOverrides:表示整个语句块要去除掉的前缀
  • suffixOverrides:表示整个语句块要去除掉的后缀

以下是对这四个属性的详细解释和代码示例:

  1. - prefix:表示整个语句块,以prefix的值作为前缀。

这个属性通常用于拼接查询结果中的字段值,添加一个固定的前缀。例如,如果数据库中有一个名为name的字段,值为John Doe,你可以使用prefix属性来添加一个前缀,比如Mr. ,使其变为Mr. John Doe

   <resultMap id="userMap" type="User">
       <result property="name" column="name" prefix="Mr. " />
   </resultMap>
  1. - suffix:表示整个语句块,以suffix的值作为后缀。

prefix相反,suffix属性可以用于在查询结果的字段值后面添加一个固定的后缀。例如,如果你希望将查询结果的age字段值后面添加一个单位years,你可以这样使用:

   <resultMap id="userMap" type="User">
       <result property="age" column="age" suffix=" years" />
   </resultMap>
  1. - prefixOverrides:表示整个语句块要去除掉的前缀。

在某些情况下,数据库中的字段值可能已经包含了一个固定的前缀,而你希望在映射结果时去除这个前缀。这时就可以使用prefixOverrides属性。例如,假设name字段中的值是Mr. John Doe,而你只想得到John Doe,可以这样配置:

    <resultMap id="userMap" type="User">
        <result property="name" column="name" prefixOverrides="Mr. " />
    </resultMap>
  1. - suffixOverrides:表示整个语句块要去除掉的后缀。

类似于prefixOverridessuffixOverrides属性可以用于去除字段值的后缀部分。比如,如果字段值是30 years,但你只需要得到30,可以这样设置:

  <resultMap id="userMap" type="User">
      <result property="age" column="age" suffixOverrides=" years" />
  </resultMap>

所以调整 Mapper.xml 的插入语句:

<insert id="insertUserByCondition">
    INSERT INTO userinfo
    <trim prefix="(" suffix=")" suffixOverrides=",">
        <if test="username !=null">
            username,
        </if>
        <if test="password !=null">
            `password`,
        </if>
        <if test="age != null">
            age,
        </if>
        <if test="gender != null">
            gender,
        </if>
        <if test="phone != null">
            phone,
        </>
    </trim>
    VALUES
    <trim prefix="(" suffix=")" suffixOverrides=",">
        <if test="username !=null">
            #{username},
        </if>
        <if test="password !=null">
            #{password},
        </if>
        <if test="age != null">
            #{age},
        </if>
        <if test="gender != null">
            #{gender},
        </if>
 <if test="phone != null">
            #{phone}
        </if>
    </trim>
</insert>

在以上 sql 动态解析时,会将第⼀个 部分做如下处理:

  • 基于 prefix 配置,开始部分加上 (
  • 基于 suffix 配置,结束部分加上 )
  • 多个组织的语句都以 , 结尾,在最后拼接好的字符串还会以 , 结尾,会基于suffixOverrides 配置去掉最后⼀个 ,
  • 注意 中的 username 是传入对象的属性

1.3 where 标签

看下面这个场景, 系统会根据我们的筛选条件, 动态组装 where 条件

在这里插入图片描述

这种如何实现呢?

需求: 传入的用户对象,根据属性做 where 条件查询,用户对象中属性不为 null 的,都为查询条件. 如 username 为 “a”,则查询条件为 where username=“a”

  1. 原有 SQL:
SELECT
	*
FROM
	userinfo
WHERE
	age = 18
	AND gender = 1
	AND delete_flag =0
  1. 接口定义:
List<UserInfo> queryByCondition();
  1. Mapper.xml 实现:
<select id="queryByCondition" resultType="com.example.demo.model.UserInfo">
    SELECT 
        id, username, age, gender, phone, delete_flag, create_time, update_time
    FROM 
        userinfo
    <where>
        <if test="age != null">
            AND age = #{age}
        </if>
        <if test="gender != null">
            AND gender = #{gender}
        </if>
        <if test="deleteFlag != null">
            AND delete_flag = #{deleteFlag}
		 </if>
    </where>
</select>

where 只会在子元素有内容的情况下才插入 where 子句,而且会自动去除子句的开头的 AND 或 OR,以上标签也可以使用 < trim prefix=“where” prefixOverrides=“and” > 替换, 但是此种情况下, 当子元素都没有内容时, where 关键字也会保留

  1. 或者使用注解方式
@Select("<script>select id, username, age, gender, phone, delete_flag,
create_time, update_time" +
	"    from userinfo" +
	"    <where>" +
	"        <if test='age != null'> and age = #{age} </if>" +
	"        <if test='gender != null'> and gender = #{gender} </if>" +
	"        <if test='deleteFlag != null'> and delete_flag = #{deleteFlag} </if>" +
"    	</where>" +
	"</script>")
List<UserInfo> queryByCondition(UserInfo userInfo);

1.4 set 标签

需求: 根据传入的用户对象属性来更新用户数据,可以使用标签来指定动态内容.

  1. 接口定义: 根据传入的用户 id 属性,修改其他不为 null 的属性
Integer updateUserByCondition(UserInfo userInfo);
  1. Mapper.xml
<update id="updateUserByCondition">
	update userinfo
	<set>
		<if test="username != null">
			username = #{username},
		</if>
		<if test="age != null">
			age = #{age},
		</if>
		<if test="deleteFlag != null">
			delete_flag = #{deleteFlag},
		</if>
	</set>
	where id = #{id}
</update>

set :动态的在 SQL 语句中插入 set 关键字,并会删掉额外的逗号. (用于update语句中)
以上标签也可以使用 < trim prefix=“set” suffixOverrides=“,” > 替换。

  1. 或者使用注解方式
@Update("<script>" +
	"update userinfo " +
	"<set>" +
		"<if test='username!=null'>username=#{username},</if>" +
		"<if test='age!=null'>age=#{age},</if>" +
		"<if test='deleteFlag!=null'>delete_flag=#{deleteFlag},</if>" +
	"</set>" +
	"where id=#{id}" +
	"</script>")
Integer updateUserByCondition(UserInfo userInfo);

1.5 foreach 标签

对集合进行遍历时可以使用该标签。标签有如下属性:

  • collection:绑定方法参数中的集合,如 List,Set,Map或数组对象
  • item:遍历时的每⼀个对象
  • open:语句块开头的字符串
  • close:语句块结束的字符串
  • separator:每次遍历之间间隔的字符串

需求: 根据多个 userid, 删除用户数据

  1. 接口方法:
void deleteByIds(List<Integer> ids);
  1. ArticleMapper.xml 中新增删除 sql:
<delete id="deleteByIds">
	delete from userinfo
	where id in
	<foreach collection="ids" item="id" separator="," open="(" close=")">
		#{id}
	</foreach>
</delete>

比如,如果 ids 集合中包含元素 1, 2, 3,那么生成的 SQL 语句将会是 delete from userinfo where id in (1, 2, 3)。这样就实现了根据传入的 ids 集合中的元素批量删除对应 ID 的记录。

  1. 或者使用注解方式:
@Delete("<script>" +
	"delete from userinfo where id in" +
	"<foreach collection='ids' item='id' separator=',' open='('close=')'>" +
		"#{id}" +
		"</foreach>" +
	"</script>")
Integer deleteUser(Integer id);

1.6 include 标签

在 xm l映射文件中配置的 SQL,有时可能会存在很多重复的片段,此时就会存在很多冗余的代码

在这里插入图片描述

我们可以对重复的代码片段进行抽取,将其通过 sql 标签封装到⼀个 SQL 片段,然后再通过 include 标签进行引用。

  • sql :定义可重用的 SQL片段
  • include :通过属性 refid,指定包含的 SQL 片段
<sql id="allColumn">
	id, username, age, gender, phone, delete_flag, create_time, update_time
</sql>

通过 include 标签在原来抽取的地方进行引用。操作如下:

<select id="queryAllUser" resultMap="BaseMap">
	select
	<include refid="allColumn"></include>
	from userinfo
</select>
<select id="queryById" resultType="com.example.demo.model.UserInfo">
	select
	<include refid="allColumn"></include>
	from userinfo where id= #{id}
</select>

这两段代码就相当于:

<select id="queryAllUser" resultMap="BaseMap">
	select
	id, username, age, gender, phone, delete_flag, create_time, update_time
	from userinfo
</select>
<select id="queryById" resultType="com.example.demo.model.UserInfo">
	select
	id, username, age, gender, phone, delete_flag, create_time, update_time
	from userinfo where id= #{id}
</select>

二: 案例练习

基于以上知识的学习, 我们就可以做⼀些简单的项目了

2.1 表白墙

前面的案例中, 我们写了表白墙, 但是⼀旦服务器重启, 数据仍然会丢失,要想数据不丢失, 需要把数据存储在数据库中. 接下来咱们借助 MyBatis 来实现数据的操作

在这里插入图片描述

2.1.1 数据准备

DROP TABLE IF EXISTS message_info;
CREATE TABLE `message_info` (
	`id` INT ( 11 ) NOT NULL AUTO_INCREMENT,
	`from` VARCHAR ( 127 ) NOT NULL,
	`to` VARCHAR ( 127 ) NOT NULL,
	`message` VARCHAR ( 256 ) NOT NULL,
	`delete_flag` TINYINT ( 4 ) DEFAULT 0 COMMENT '0-正常, 1-删除',
	`create_time` DATETIME DEFAULT now(),
	`update_time` DATETIME DEFAULT now() ON UPDATE now(),
PRIMARY KEY ( `id` )
) ENGINE = INNODB DEFAULT CHARSET = utf8mb4;

ON UPDATE now(): 当数据发生更新操作时, 自动把该列的值设置为 now(),

2.1.2 引入 MyBatis 和 MySQL 驱动依赖

修改 pom 文件

<dependency>
	<groupId>org.mybatis.spring.boot</groupId>
	<artifactId>mybatis-spring-boot-starter</artifactId>
	<version>2.3.1</version>
</dependency>
<dependency>
	<groupId>com.mysql</groupId>
	<artifactId>mysql-connector-j</artifactId>
	<scope>runtime</scope>
</dependency>

或者使用插件 EditStarters 来引入依赖

在这里插入图片描述

2.1.3 配置 MySQL 账号密码

spring:
	datasource:
	url: jdbc:mysql://127.0.0.1:3306/mybatis_test?characterEncoding=utf8&useSSL=false
	username: root
	password: root
	driver-class-name: com.mysql.cj.jdbc.Driver
mybatis:
	configuration: # 配置打印 MyBatis⽇志
		map-underscore-to-camel-case: true #配置驼峰⾃动转换

2.1.4 编写后端代码

  1. Model
import lombok.Data;
@Data
public class MessageInfo {
	private Integer id;
	private String from;
	private String to;
	private String message;
	private Integer deleteFlag;
	private Date createTime;
	private Date updateTime;
}
  1. MessageInfoMapper
import com.example.demo.model.MessageInfo;
import org.apache.ibatis.annotations.Insert;
import org.apache.ibatis.annotations.Mapper;
import org.apache.ibatis.annotations.Select;

import java.util.List;

@Mapper
public interface MessageInfoMapper {
	@Select("select `id`, `from`, `to`, `message` from message_info where delete_flag=0")
	List<MessageInfo> queryAll();

	@Insert("insert into message_info (`from`,`to`, `message`) values(#{from},#{to},#{message})")
	Integer addMessage(MessageInfo messageInfo);
}
  1. MessageInfoService
import com.example.demo.mapper.MessageInfoMapper;
import com.example.demo.model.MessageInfo;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Service;

import java.util.List;

@Service
public class MessageInfoService {
    @Autowired
    private MessageInfoMapper messageInfoMapper;
    
    public List<MessageInfo> queryAll() {
        return messageInfoMapper.queryAll();
    }
    
    public Integer addMessage(MessageInfo messageInfo) {
        return messageInfoMapper.addMessage(messageInfo);
    }
}
  1. MessageController
import com.example.demo.model.MessageInfo;
import com.example.demo.service.MessageInfoService;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.util.StringUtils;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;

import java.util.List;

@RequestMapping("/message")
@RestController
public class MessageController {
    @Autowired
    private MessageInfoService messageInfoService;
    /**
     * 获取留⾔列表
     * @return
     */
    @RequestMapping("/getList")
    public List<MessageInfo> getList() {
        return messageInfoService.queryAll();
    }
    
    /**
     * 发表留⾔
     * @param messageInfo
     * @return
     */
    @RequestMapping("/publish")
    public boolean publish(MessageInfo messageInfo) {
        System.out.println(messageInfo);
        if (StringUtils.hasLength(messageInfo.getFrom())
                && StringUtils.hasLength(messageInfo.getTo())
                && StringUtils.hasLength(messageInfo.getMessage())) {
            messageInfoService.addMessage(messageInfo);
            return true;
        }
        return false;
    }
}

2.1.5 测试

部署程序, 验证服务器是否能正确响应: http://127.0.0.1:8080/messagewall.html

在这里插入图片描述

输⼊留言信息, 点击提交, 发现页面列表显示新的数据, 并且数据库中也添加了⼀条记录.

在这里插入图片描述

在这里插入图片描述

重启服务, ⻚⾯显⽰不变.

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1638569.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

【二】电力系统规约IEC 104详解

电力系统规约IEC 104详解 概述 很早就准备梳理出一下电力系统规约系列的文章&#xff0c;因为自己在实践过程中发现这方面太难找了&#xff0c;网上的资料也都比较陈旧。我接触和使用IEC系列规约也有一段时间了&#xff0c;本着总结和分享的想法&#xff0c;我想推出这系列的文…

Linux的基础IO:文件描述符 重定向本质

目录 前言 文件操作的系统调用接口 open函数 close函数 write函数 read函数 注意事项 文件描述符-fd 小补充 重定向 文件描述符的分配原则 系统调用接口-dup2 缓冲区 缓冲区的刷新策略 对于“2”的理解 小补充 前言 在Linux中一切皆文件&#xff0c;打开文件…

Android 音视频基础知识

本系列文章会介绍两个 Android NDK Demo&#xff0c;拉流端会实现一个基于 FFmpeg 的视频播放器 Demo&#xff0c;推流端会实现一个视频直播 Demo&#xff0c;当然在做 Demo 之前会介绍音视频的基础知识。以下是本系列文章的目录&#xff1a; Android 音视频基础知识 Android 音…

SpringBoot集成Kafka开发

4.SpringBoot集成Kafka开发 4.1 创建项目 4.2 配置文件 application.yml spring:application:name: spring-boot-01-kafka-basekafka:bootstrap-servers: 192.168.2.118:90924.3 创建生产者 package com.zzc.producer;import jakarta.annotation.Resource; import org.spri…

HTML5本地存储账号密码

<!DOCTYPE html> <html lang"zh-CN"> <head><meta charset"UTF-8"><meta name"viewport" content"widthdevice-width, initial-scale1.0"><title>HTML5本地存储账号密码</title> </head…

C语言/数据结构——每日一题(合并两个有序链表)

一.前言 嗨嗨嗨&#xff0c;大家好久不见&#xff01;今天我在LeetCode看到了一道单链表题&#xff1a;https://leetcode.cn/problems/merge-two-sorted-lists想着和大家分享一下&#xff0c;废话不多说&#xff0c;让我们开始今天的题目分享吧。 二.正文 1.1题目描述 1.2题…

2.2 Java全栈开发前端+后端(全栈工程师进阶之路)-前端框架VUE3-基础-Vue基本语法

文本渲染指令 文本渲染指令-v-html与v-text Vue使用了基于HTML的模板语法&#xff0c;允许开发者声明式地将DOM绑定至底层Vue实例的数据。所有Vue的模板都是 合法的HTML&#xff0c;所以能被遵循规范的浏览器和HTML解析器解析。 在前面&#xff0c;我们一直使用的是字符串插…

利用STM32实现语音识别功能

引言 随着物联网和智能设备的普及&#xff0c;语音识别技术正逐渐成为用户交互的主流方式之一。 STM32微控制器具备处理高效率语音识别算法的能力&#xff0c;使其成为实现低成本、低功耗语音交互系统的理想选择。 本教程将介绍如何在STM32平台上开发和部署一个基础的语音识…

创建codereview

创建codereview流程 一、开始创建二、选择分支三、添加细节 一、开始创建 点击codereivew按钮 为新的codereview选择一个工程后点击create review 二、选择分支 选择目标分支和要比对的分支&#xff0c;比如develop 三、添加细节 Add branch后&#xff0c;可以继续Edit …

Linux深入学习内核 - 中断与异常(下)

软中断&#xff0c;Tasklet和Work Queue 由内核执行的几个任务之间有一些不是紧急的&#xff0c;他们可以被延缓一段时间&#xff01;把可延迟的中断从中断处理程序中抽出来&#xff0c;有利于使得内核保持较短的响应时间&#xff0c;所以我们现在使用以下面的这些结构&#x…

JVM的垃圾回收机制(GC机制)

在Java代码运行的过程中&#xff0c;JVM发现 某些资源不需要再使用的时候&#xff0c;就会自动把资源所占的内存给回收掉&#xff0c;就不需要程序员自行操作了。“自动回收资源”就是JVM的“垃圾回收机制”&#xff0c;“垃圾回收机制”也称"GC机制"。 对于Java代码…

文心一言 VS 讯飞星火 VS chatgpt (250)-- 算法导论18.2 3题

三、请说明如何在一棵 B 树中找出最小关键字&#xff0c;以及如何找出某一给定关键字的前驱。如果要写代码&#xff0c;请用go语言。 文心一言&#xff1a; 在B树&#xff08;也称为B-tree&#xff09;中&#xff0c;最小的关键字总是位于最左边的叶子节点的最左边的元素。要…

C++ | 类和对象(中) (构造函数 | 析构函数 | 拷贝构造函数 | 赋值运算符重载 | 取地址 | const取地址)

目录 默认成员函数 构造函数 构造函数是什么 构造函数特征 什么是默认构造函数 注意事项 编译器自动生成的默认构造 缺省值 对象如何传值给构造函数 初始化列表 析构函数 析构函数的特征 编译器默认生成的析构函数 总结 拷贝构造函数 拷贝构造函数的使用场景 拷…

MySQL数据库练习(13)

schooldb库——utf8字符集——utf8_general_ci排序规则 61. DDL CREATE TABLE settlements (settlementId int(11) NOT NULL AUTO_INCREMENT COMMENT 自增ID,settlementNo varchar(20) NOT NULL COMMENT 结算单号,settlementType tinyint(4) NOT NULL DEFAULT 0 COMMENT 结算…

金融案例:统一查询方案助力数据治理与分析应用更高效、更安全

随着企业数据规模的增长和业务多元化发展&#xff0c;海量数据实时、多维地灵活查询变成业务常见诉求。同时多套数据库系统成为常态&#xff0c;这既带来了数据管理的复杂性&#xff0c;又加大了数据使用的难度&#xff0c;面对日益复杂的数据环境和严格的数据安全要求&#xf…

Centos7 安装Git、使用

Centos7 安装Git 一、安装步骤1.1 查看版本1.2 卸载1.3 安装 二、创建仓库2.1 新增仓库2.2 新增配置项 三、管理文件3.1 文件创建3.2 文件修改、add、commit3.3 tree结构探索 四、分支4.1 创建分支&#xff1a;4.2 查看分支4.3 切换分支4.4 删除分支4.5 合并冲突 一、安装步骤 …

【蓝桥杯嵌入式】第七届省赛 - 模拟液位检测告警系统

代码开源&#xff0c;Gitee自取 代码开源&#xff0c;Gitee自取 代码开源&#xff0c;Gitee自取 目录 0 前言 1 展示 1.1 源码 1.2 演示视频 1.3 题目展示 2 工程配置 3 资源配置&代码实现 3.1 定时器 3.2 液位检测 3.3 液位阈值设定 3.4 液位阈值设定 3.5 串…

使用unreal engine5.3.2创建c++第一人称游戏

UE5系列文章目录 文章目录 UE5系列文章目录前言一、NuGet 简介二、解决方法&#xff1a; 前言 为了使用unreal engine5.3.2创建c第一人称游戏&#xff0c;今天安装了Visual Studio 2022专业版。在ue5中创建c工程&#xff0c;结果编译器报错&#xff1a; 严重性 代码 说明 项目…

UDP和TCP(传输层)

这里写目录标题 UDPUDP的基本特点UDP协议报文格式 TCPTCP协议报文格式TCP特点可靠传输实现机制确认应答超时重传数据丢了应答报文丢了 小结 UDP UDP的基本特点 无连接不可靠传输面向数据报全双工 UDP协议报文格式 2个字节有效范围(无符号): 0 ~ 65535(2^16 - 1). 2个字节有效范…

Web开发:<div>作用

< div >元素作用 解释用途1. 布局&#xff1a;2. 样式化&#xff1a;3. 结构化&#xff1a;4. JavaScript操作&#xff1a;5. 响应式设计&#xff1a;6. 模块化和重用&#xff1a;7. 包裹其他元素&#xff1a;8. 清除浮动&#xff1a; 总结举例示例结果分析 解释 在Web开…