[Spring] MyBatis操作数据库(进阶)

news2024/9/8 19:28:30

🌸个人主页:https://blog.csdn.net/2301_80050796?spm=1000.2115.3001.5343
🏵️热门专栏:
🧊 Java基本语法(97平均质量分)https://blog.csdn.net/2301_80050796/category_12615970.html?spm=1001.2014.3001.5482
🍕 Collection与数据结构 (92平均质量分)https://blog.csdn.net/2301_80050796/category_12621348.html?spm=1001.2014.3001.5482
🧀线程与网络(96平均质量分) https://blog.csdn.net/2301_80050796/category_12643370.html?spm=1001.2014.3001.5482
🍭MySql数据库(93平均质量分)https://blog.csdn.net/2301_80050796/category_12629890.html?spm=1001.2014.3001.5482
🍬算法(97平均质量分)https://blog.csdn.net/2301_80050796/category_12676091.html?spm=1001.2014.3001.5482
🍃 Spring(97平均质量分)https://blog.csdn.net/2301_80050796/category_12724152.html?spm=1001.2014.3001.5482
感谢点赞与关注~~~
在这里插入图片描述

目录

  • 1. 数据库连接池
    • 1.1 为什么会有连接池
    • 1.2 常用的数据库连接池
  • 2. 动态SQL
    • 2.1 ``<if>``标签
    • 2.2 ``<trim>``标签
    • 2.3 ``<where>``标签
    • 2.4 ``<set>``标签
    • 2.5 ``<for each>``标签
    • 2.6 ``<include>``标签
  • 3. 案例练习(表白墙)
    • 3.1 配置文件
    • 3.2 Mapper层代码
    • 3.3 Service层代码
    • 3.4 Controller层代码

1. 数据库连接池

1.1 为什么会有连接池

数据库连接池技术也是一种池化技术,他与线程池类似,在通过Spring连接数据库的时候,避免了频繁的创建和销毁与数据库的连接.保证了资源的合理利用.
没有使用数据库连接池的情况: 在每一次在Spring中执行SQL语句的时候,都会先创建一个与数据库的连接对象,在执行完SQL语句之后,再关闭SQL语句,这种重复创建和销毁对象的行为比较浪费系统资源.
使用数据连接池的情况: 在程序启动的时候,会在连接池中创建Collection对象,在执行SQL语句的时候,会在连接池中申请一个连接对象,在执行SQL语句之后,又会把线程归还给连接池.

1.2 常用的数据库连接池

• C3P0
• DBCP
• Druid
• Hikari
目前比较流行的是Hikari,Druid,其中,Druid是阿里巴巴开源项目中的连接池.
其中Hikari是一个Spring程序中默认使用的连接池.如果想把连接池切换为Druid,我们需要在pom文件中引入依赖:

<dependency>
	<groupId>com.alibaba</groupId>
	<artifactId>druid-spring-boot-3-starter</artifactId>
	<version>1.2.21</version>
</dependency>

2. 动态SQL

动态SQL可以完成不同场景下的SQL拼接.

2.1 <if>标签

在这里插入图片描述

在我们注册用户的时候,我们有一个这样的问题,有时候有的选项不是必填项.在添加用户的时候SQL字段个数是不确定的.这时候就需要我们使用<if>标签了.比如gender性别是非必填字段:

接口定义:

public Integer insertUser2(UserInfo userInfo);

sql:

<insert id="insertUser2">
    insert into userinfo (
                          id,
                          username,
                          password,
                          age,
                          <if test="gender != null">
                              gender,
                          </if>
                          phone
    ) values (
              #{id},
              #{username},
              #{password},
              #{age},
              <if test="gender != null">
                  #{gender},
              </if>
              #{phone}
    )
</insert>

或者我们也可以使用注解的方式,使用<script></script>标签把上面的xml语句括起来即可.但是我们不推荐用这种做法.

@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>")
public Integer insertUser2(UserInfo userInfo);

注意if中的gender是对象的属性,不是SQL的字段,因为对象是从前端传过来的,我们在判断的时候,是判断前端有没有传值过来,而前段的值就是传到后端的方法参数中的.

2.2 <trim>标签

上面只使用if标签有一定的问题.要是我们在拼接SQL语句的时候,if标签在最后一个字段,比如:

<insert id="insertUser2">
    insert into userinfo (
                          id,
                          username,
                          password,
                          age,
                          gender,
                          <if test="phone != null">
                              phone
                          </if>
    ) values (
              #{id},
              #{username},
              #{password},
              #{age},
              #{gender},
              <if test="phone != null">
                  #{phone}
              </if>
    )
</insert>

当我们没有phone属性的时候,我们在gender字段后面就会多上一个逗号,这显然不符合sql的语法,所以我们就需要使用<trim>标签.
标签中有如下属性:
• prefix:表示整个语句块,以prefix的值作为前缀
• suffix:表示整个语句块,以suffix的值作为后缀
• prefixOverrides:表示整个语句块要去除掉的前缀
• suffixOverrides:表示整个语句块要去除掉的后缀
调整上面的插入语句:

<insert id="insertUser3">
    insert into userinfo
          <trim prefix="(" suffix=")" suffixOverrides="," prefixOverrides=",">
              id,
              username,
              password,
              age,
              gender,
              <if test="phone != null">
                  phone
              </if>
          </trim>
              values
          <trim prefix="(" prefixOverrides="," suffix=")" suffixOverrides=",">
              #{id},
              #{username},
              #{password},
              #{age},
              #{gender},
              <if test="phone != null">
                  #{phone}
              </if>
          </trim>
</insert>

假如所有的字段都是可选字段:

<insert id="insertUser4">
    insert into userinfo
        <trim prefixOverrides="," suffixOverrides="," suffix=")" prefix="(">
            <if test="id != null">
                id,
            </if>
            <if test="username != null">
                username,
            </if>
            <if test="password != null">
                password,
            </if>
            <if test="age != null">
                age,
            </if>
            <if test="phone != null">
                phone
            </if>
        </trim>
            values
        <trim prefix="(" suffix=")" suffixOverrides="," prefixOverrides=",">
            <if test="id != null">
                #{id},
            </if>
            <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>

trim标签中的参数的用意就是:在前面和后面只要有逗号,就都会去掉,之后再在sql语句中的前后拼接上括号.
这个标签就是<trim>中的内容当做一个字符串,对这个字符串进行整体加减前后缀.
在注解中写的方法和上面的if标签是一样的,这里不再赘述.

2.3 <where>标签

下面这个场景,系统会根据我们的筛选条件,动态组装where条件.
在这里插入图片描述
这样的场景该如何实现呢?下面我们来看例子.
传入的用户对象,根据属性做where条件查询,用户对象中属性不为null的,都为查询条件.
接口定义:

public List<UserInfo> selectUser1(UserInfo userInfo);

根据传过来的对象限制查询条件.
sql:

<select id="selectUser1" resultType="com.jrj.mybatis.UserInfo">
    select * from userinfo
    <where>
        <if test="gender != null">
            #{gender},
        </if>
        <if test="age != null">
            #{age},
        </if>
        <if test="deleteFlag != null">
            #{deleteFlag}
        </if>
    </where>
</select>

<where>标签和<trim>标签类似,where可以去除掉前后的and和or,如果where后面没有字段的话,就没有where语句了,where关键字不予保留.效果和<trim prefix="where" prefixOverrides="and"> 相同,但是使用这个标签的时候,当没有限制条件的时候,where关键字就会保留下来.

2.4 <set>标签

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

public Integer updateUser1(UserInfo userInfo);

sql:

<update id="updateUser1">
    update userinfo
    <set>
        <if test="username != null">
            username = #{username},
        </if>
        <if test="age != null">
            age = #{age},
        </if>
        <if test="gender != null">
            gender = #{gender}
        </if>
    </set> where id = #{id}
</update>

<set>标签和上面的<where>标签一样,会自动去除逗号,也会根据是否有更新字段来判断是否要添加set关键字.与<trim prefix="set" suffixOverrides=",">效果一样,但是在没有传入跟新字段的时候,不会去掉set关键字.

2.5 <for each>标签

对集合进行遍历的时候,可以用这个标签,一般与sql语句中的in配合使用.
对集合进行遍历时可以使用该标签。标签有如下属性:
• collection:绑定方法参数中的集合,如List,Set,Map或数组对象
• item:遍历时的每⼀个对象
• open:语句块开头的字符串
• close:语句块结束的字符串
• separator:每次遍历之间间隔的字符串
比如根据多个用户id删除用户数据.
接口:

public Integer updateDeleteFlag1(List<UserInfo> userList);

sql:

<update id="updateDeleteFlag1">
    update userinfo set delet_flag
    where id in
        <foreach collection="userList" item="id" open="(" close=")">
            #{id}
        </foreach>
</update>

这里我们在删除用户数据的时候,使用的是逻辑删除的方式:

删除方式一共有两种,一种是逻辑删除,一种是物理删除,物理删除就是使用Delete语句,在计算机的存储空间中彻底删除,想要回复只能通过日志回滚的方式进行恢复,非常麻烦.而逻辑删除就是把表中每个数据的必备字段,delete_flag标记为无效的数值,并没有从硬盘上真正删除掉,恢复的时候只需要把这个数值改成有效数值即可.我们在企业开发的时候,一般用的都是逻辑删除.

[补充] 其中的collection参数也可以用list1,list2来表示,和前面的方法形参与sql语句传参类似.

2.6 <include>标签

这个标签通常与<sql>标签配合使用,在xml文件中,有一些sql语句是赘余的,即sql语句中的赘余片段.
在这里插入图片描述
我们可以对重复的代码片段段进行抽取,将其通过 <sql> 标签封装到⼀个SQL片段.之后在每一个sql语句中使用<include>标签进行引用即可.

  • <sql> : 对可以重复引用的sql语句进行定义,在参数id中定义该sql片段的名字.
  • <include> : 通过属性refid,指定引用的sql片段.
<sql id="selectAllColumn">
    id,username,password,age,gender,phone,delete_flag,create_time,update_time
</sql>

<select id="selectAllUser2" resultType="com.jrj.mybatis.UserInfo">
    select
        <include refid="selectAllColumn"></include>
    from userinfo
</select>

3. 案例练习(表白墙)

前面我们写了表白墙,但是在服务器重启之后,就会清空之前的数据,数据无法永久保留,这时候就需要引入MyBatis了,我们需要把数据存储在数据库中.

3.1 配置文件

spring:
  application:
    name: vindicatewall
  # 数据库连接配置
  datasource:
    url: jdbc:mysql://127.0.0.1:3306/mybatis_test?characterEncoding=utf8&useSSL=false
    username: root
    password: qwe123524
    driver-class-name: com.mysql.cj.jdbc.Driver

mybatis:
  configuration: # 配置打印 MyBatis⽇志
    log-impl: org.apache.ibatis.logging.stdout.StdOutImpl
    map-underscore-to-camel-case: true #配置驼峰⾃动转换

需要配置的选项有这几个,第一个是数据库相关配置,包括数据库url,用户名,密码,MySQl驱动类.其次,需要配置MyBatis中的日志打印和驼峰自动转换.

3.2 Mapper层代码

@Mapper
public interface MessageMapper {
    @Insert("insert into message_info (`from`,`to`,`message`) values " +
            "(#{from},#{to},#{message})")
    public Integer updateMessage(Message message);
    @Select("select * from message_info")
    public List<Message> selectAllMessage();
}

3.3 Service层代码



@Service
public class MessageService {
    @Autowired
    public MessageMapper messageMapper;
    public Integer updateMessage(Message message){
        return messageMapper.updateMessage(message);
    }
    public List<Message> selectAllMessage(){
        return messageMapper.selectAllMessage();
    }
}

3.4 Controller层代码

@RestController
@RequestMapping("/messagewall")
public class MessageController {
    @Autowired
    public MessageService messageService;

    @RequestMapping("/publish")
    public Boolean updateMessage(Message message){
        if (StringUtils.hasLength(message.getFrom())&&
        StringUtils.hasLength(message.getTo())&&
        StringUtils.hasLength(message.getMessage())){
            messageService.updateMessage(message);
            return true;
        }
        return false;
    }

    @RequestMapping("/getlist")
    public List<Message> selectAllMessage(){
        return messageService.selectAllMessage();
    }
}

Postman测试后端代码:
在这里插入图片描述
在这里插入图片描述
数据库已更新:
在这里插入图片描述

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

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

相关文章

功能性的安全保障:输入校验

前言 在软件开发过程中&#xff0c;确保系统的安全性是至关重要的一环。它不仅关乎保护用户数据的完整性和隐私性&#xff0c;也是维护系统稳定运行的基石。我认为&#xff0c;从宏观角度审视&#xff0c;软件开发的安全性保障主要可分为两大类&#xff1a;功能性的安全性保障…

GitEval — 预测你的 GitHub 个人资料的质量

使用机器学习来预测你是否擅长编码 可直接在橱窗里购买&#xff0c;或者到文末领取优惠后购买&#xff1a; 如果你曾经申请过技术职位&#xff0c;你可能已经向公司发送了你的 GitHub 个人资料链接。此个人资料中的信息可以很好地表明你的编码能力以及是否适合团队。所有这些信…

【全国大学生电子设计竞赛】2024年C题

&#x1f970;&#x1f970;全国大学生电子设计大赛学习资料专栏已开启&#xff0c;限时免费&#xff0c;速速收藏~

Opencv画出红底白字标准中文显示框

链接&#xff1a;https://pan.baidu.com/s/1iEJKpqt-z_5yBJdenUABbA 提取码&#xff1a;uoox 先把这个文件拿了&#xff0c;这个文件是一个ttf的字体&#xff0c;用于显示中文。 核心代码&#x1f451; def cv2AddChineseText(self, img_ori, text, p1, box_color, textColo…

【JavaScript】函数声明和函数表达式的区别

文章目录 一、函数声明1. 定义方式2. 作用域提升&#xff08;Hoisting&#xff09;3. 块级作用域 二、函数表达式1. 定义方式2. 作用域提升&#xff08;Hoisting&#xff09;3. 自引用 三、其他区别1. 函数名2. 可读性和代码组织3. 使用场景 四、总结函数声明函数表达式 在Java…

昇思MindSpore学习入门-自动混合精度

混合精度&#xff08;Mix Precision&#xff09;训练是指在训练时&#xff0c;对神经网络不同的运算采用不同的数值精度的运算策略。在神经网络运算中&#xff0c;部分运算对数值精度不敏感&#xff0c;此时使用较低精度可以达到明显的加速效果&#xff08;如conv、matmul等&am…

Android 10.0 Launcher3仿ios的folder文件夹widget功能实现一

1.前言 在10.0的系统ROM开发中,在进行一些系统Launcher3定制功能开发中,需要实现folder文件夹widget的功能,由于launcher3 默认不支持folder跨行显示,所以就需要借助自定义的widget小部件功能来实现相关功能,接下来分析实现相关功能 2.Launcher3仿ios的folder文件夹widge…

【保姆级教程】免费域名注册 Cloudflare 域名解析 Ngnix端口转发

前段时间&#xff0c;带大家搞了两台云服务器&#xff1a; 玩转云服务&#xff1a;Oracle Cloud甲骨文永久免费云服务器注册及配置指南玩转云服务&#xff1a;手把手带你薅一台腾讯云服务器&#xff0c;公网 IP 基于这两台云服务器&#xff0c;我们玩转了很多有趣的开源项目&…

ElasticSearch(八)— 聚集查询1

一、总概 聚集查询(Aggregation)提供了针对多条文档的统计运算功能&#xff0c;它不是针对文档本身内容的检索&#xff0c;而是要将它们聚合到一起运算某些方面的特征值。 聚集查询与 SQL 语言中的聚集函数非常像&#xff0c;聚集函数在 Elasticsearch 中相当于是聚集查询的一…

掌控数据流的智能仪表板:Redpanda Console

Redpanda Console&#xff1a; 一站式管理&#xff0c;让数据流尽在掌控之中。- 精选真开源&#xff0c;释放新价值。 概览 Redpanda Console&#xff0c;是一款创新的Web界面工具&#xff0c;专为简化Kafka和Redpanda数据流的监控与管理而设计。它以用户友好的交互方式&#…

浏览器用户文件夹详解 - Favicons(二)

1. 引言 上一篇文章我们深入探讨了Chromium用户文件夹中的Bookmarks文件,了解了它的JSON结构以及如何解析和修改书签数据。 在本文中,我们将继续探索Chromium用户文件夹中的另一个重要文件:Favicons。Favicons,也就是我们常说的网站图标,是浏览器中不可或缺的一部分。它们不仅…

RK3568笔记四十七:PWM 子系统

若该文为原创文章&#xff0c;转载请注明原文出处。 pwm 子系统功能单一&#xff0c;很少单独使用&#xff0c;一般用于控制显示屏的背光、控制无源蜂鸣器、伺服电机、电压调节等等。 一、PWM介绍 PWM(Pulse width modulation)&#xff0c;脉冲宽度调制。在内核中 PWM 驱动较简…

并发编程工具集——并发容器-下(二十五)

List List 里面只有一个实现类就是 CopyOnWriteArrayList。CopyOnWrite&#xff0c;写的时候会将共享变量新复制一份出来&#xff0c;读操作完全无锁&#xff1b;适合读多写少的场景&#xff0c;写操作会复制数组&#xff0c;在新的数组中操作实现原理&#xff1a;CopyOnWriteA…

备考软考高级系统架构设计师,需要买哪些资料?

距离2024下半年软考高级系统架构设计师考试仅剩三个月&#xff01;时间紧迫&#xff0c;单单啃书已经不够了&#xff0c;毕竟是软考高级科目&#xff0c;难度不是那些初、中级可以比拟的。要想短时间速通架构考试&#xff0c;学会抓重点真的很重要&#xff0c;45分说多不多说少…

【音视频】RTSP、RTMP与流式传输

文章目录 前言RTSP与RTMPRTSP&#xff08;Real-Time Streaming Protocol&#xff09;RTMP&#xff08;Real-Time Messaging Protocol&#xff09;主要差异 什么是流式传输&#xff1f;流式传输的特点流式传输与传统下载的区别 使用VLC播放RTSP监控 总结 前言 在现代网络环境中…

C++ //练习 16.2 编写并测试你自己版本的compare函数。

C Primer&#xff08;第5版&#xff09; 练习 16.2 练习 16.2 编写并测试你自己版本的compare函数。 环境&#xff1a;Linux Ubuntu&#xff08;云服务器&#xff09; 工具&#xff1a;vim 代码块 /*********************************************************************…

PyTorch深度学习快速入门(下)

PyTorch深度学习快速入门&#xff08;下&#xff09; 一、现有网络模型的使用及修改&#xff08;一&#xff09;背景知识&#xff08;二&#xff09;修改网络模型的三种方法 二、网络模型的保存与加载&#xff08;一&#xff09;保存网络模型的两种方法&#xff08;二&#xff…

阿里云 服务器安装rabbit

现在我们去服务器安装一个rabbit 进入home 创建一个rabbit文件夹 /home/rabbit vim deployRabbit.sh 脚本内容 #!/bin/bash docker run -d \ --name dev.rabbit \ --network dev-net \ -p 15672:15672 \ -v ./data:/var/lib/rabbitmq \ --hostname dev.rabbit \ rabbitmq:…

css3 红色阴影边框紧急提醒呼吸灯特效

效果截图 代码 <!DOCTYPE html> <html><head><title>红色呼吸灯紧急特效</title><style>keyframes alarm {0% {box-shadow: 0 0 30px #ff0000;}50% {box-shadow: 0 0 60px #ff0000, inset 0 0 60px #ff0000;}100% {box-shadow: 0 0 30px …

八、【Python】基础 - 【Python while 循环全解析】:掌握无限循环的艺术

&#x1f4a1;&#x1f4da;【Python while 循环全解析】&#xff1a;掌握无限循环的艺术&#x1f4da;&#x1f4a1; 目录 1.基本语法 2.示例 3.注意事项 4.嵌套循环与循环控制语句 5.示例&#xff1a;使用 break 和 continue 6.示例&#xff1a;计数器 7.示例&#xf…