【javaWeb 第八篇】后端-Mybatis(万字详细学习)

Mybatis框架

  • 前言
  • Mybatis
    • Mybatis入门
    • 配置SQL提示
    • JDBC
    • 数据库连接池
    • lombok
  • Mybatis基础操作
    • 日志输出
    • Mybatis的动态SQL

前言

这篇是作者学习数据持久层框架Mybatis的学习笔记,希望对大家有所帮助,希望大家能够与作者交流讨论

Mybatis

Mybatis是一款优秀的持久层框架,用于简化JDBC的开发
Mybatis自2013年11月迁移到GitHub中,源码由GitHub管理

官网:https://mybatis.org/mybatis-3/zh/index.html
本片讲解的是基于SpringBoot整合Mybatis来进行讲解

Mybatis入门

  • 使用Mybatis查询所有的用户数据

入门基础操作:

  1. 准备工作(创建springboot工程,数据库表user,实体类User)
  2. 引入Mybatis的相关依赖,配置Mybatis(数据库的连接信息)
  3. 编写SQL语句(注解/xml)

Mybatis通过java访问数据库服务器,将返回回来的数据封装到实体类当中
在引入Mybatis的相关依赖的时候需要连接数据库(知道自己要访问哪一个数据库),需要Mybatis数据库连接四要素:
标记数据库驱动的全类名
数据库连接的URL
用户名
密码


spring.datasource.driver-class-name=com.mysql.cj.jdbc.Driver
spring.datasource.url=jdbc:mysql://localhost:3306/mybatis
spring.datasource.username=root
spring.datasource.password=

在进行数据库的SQL语句编写的时候需要注意定义一个数据层接口

@Mapper
public interface UserMapper{
@select("select * from user")
public List<User> list();
}

使用注解 @Mapper来定义数据层接口,通过注解定义SQL语句, @select表示是查询操作,其中写入查询语句,当要执行次查询语句的时候调用下方的list方法就可以,这时框架会自动的执行SQL语句,并将结果封装到方法的返回值当中

如何配置mybatis,与数据库建立连接
在resources中打开application.properties文件(sping自带的默认配置文件)
在其中将连接四要素配置进去

# 驱动类名称
spring.datasource.driver-class-name=com.mysql.cj.jdbc.Driver
# 数据库连接的URL
spring.datasource.url=jdbc:mysql://localhost:3306/ mybatistest
# 连接数据库的用户名
spring.datasource.username=root
# 连接数据库的密码
spring.datasource.password=

进行测试:

@Mapper //在运行时,会自动生成该接口的实现类对象(代理对象),并且该对象交由IOC容器管理
public interface UserMapper {
    @Select("select * from user")
    public List<User> list();
}

测试通过test文件下的测试类进行测试
测试类中的注解 @SpringBootTest表示springboot整合单元测试的注解

@SpringBootTest
class MybatisTestApplicationTests {
    @Autowired
    private UserMapper userMapper;
    @Test
    void contextLoads() {
        List<User> list = userMapper.list();
        list.stream().forEach(user -> {
            System.out.println(user);
        });
    }

}

前面所说的UserMapper接口由IOC代理,所以使用依赖注入的方式在测试类中创建对象,通过stream流的方式将数据进行遍历

最后运行测试,由于此单元测试是spring整合的单元测试,在运行的时候会自动的加载整个spring环境,创建spring的IOC容器

配置SQL提示

默认在mybatis中编写SQL语句是不识别的,可以通过配置进行识别:
在这里插入图片描述
将sql语句全选右键选择
在这里插入图片描述
再次进行选择
在这里插入图片描述
选择Mysql(sql)
这个时候再次输入SQL语句就会有响应的提示,但是在输入表名的时候没有进行提示,这是因为:idea和数据库没有建立连接,不识别表信息

  • 解决方式
    在idea中配置MySQL数据库连接
    在这里插入图片描述
    配置idea连接MySQL数据库
    注意:配置完成后没有表的刷新一下就行了

JDBC

就是使用java语言操作关系型数据库的一套API



驱动
什么是驱动:
java提供一个可供操作的接口JDBC各个数据库厂商通过实现实现接口将自己的数据库操作引入到java中,提供数据库驱动jar包,这个实现的过程就称为数据库的驱动,我们通过使用这套接口(JDBC)编程,真正执行的代码是驱动jar包中的实现类

  • 原始的JDBC操作数据库
    @Test
    public void jdbctest() throws Exception{
        //注册驱动
        Class.forName("com.mysql.cj.jdbc.Driver");
        //获取连接对象
        String url="jdbc:mysql://localhost:3306/mybatistest";
        String username="root";
        String password="";
        Connection connection= DriverManager.getConnection(url,username,password);
        //获取执行sql的对象statement,执行SQL,返回结果
        String sql="select * from user";
        //通过connection创建一个statement对象用来将sql语句与数据库联系执行
        Statement statement = connection.createStatement();
        //通过执行查询语句返回一个ResultSet结果集,用来封装得到的数据
        ResultSet resultSet=statement.executeQuery(sql);

        //解析并封装结果数据(需要一个一个字段的继续宁解析)
        List<User> list=new ArrayList<>();
        while (resultSet.next())
        {
            //在解析的时候需要一个字段一个字段进行解析
            int id=resultSet.getInt("id");
            String name=resultSet.getString("name");
            int age=resultSet.getInt("age");
            int gender=resultSet.getInt("gender");
            String phone=resultSet.getString("phone");
            User user=new User(id,name,age,gender,phone);
            list.add(user);
        }
        //释放资源
        statement.close();
        connection.close();
    }

在JDBC原始代码中进行数据库的连接与最后释放资源都是非常繁琐且损耗效率的。

数据库连接池

数据库连接池是一个容器,负责分配,管理数据库连接(Connetion)
它允许应用程序重复使用一个现有的数据库连接,而不是重新建立一个
释放空间时间超过最大空闲时间的连接,来避免因为没有释放连接而引起的数据库连接遗漏
数据库连接池的优势:
资源重用,提示系统响应速度,避免数据库连接遗漏

数据库连接池:
由官方提供一个标准接口:DataSource,所有的数据库连接池都必须实现DataSource
官方提供的数据库连接池接口,由第三方实现此接口
现在项目中使用的最多的连接池:Druid,Hikari(springboot默认)

  • Druid(德鲁伊)
    是阿里巴巴开源的数据库连接池项目,功能强大,性能优秀,是java语言最好的数据库连接池之一

  • 如何切换连接池

  1. 首先引进Druid的依赖
            <groupId>com.alibaba</groupId>
            <artifactId>druid-spring-boot-starter</artifactId>
            <version>1.2.8</version>
  1. 第二步
    同上述一样通过连接四要素进行数据库连接
# 驱动类名称
spring.datasource.driver-class-name=com.mysql.cj.jdbc.Driver
# 数据库连接的URL
spring.datasource.url=jdbc:mysql://localhost:3306/ mybatistest
# 连接数据库的用户名
spring.datasource.username=root
# 连接数据库的密码
spring.datasource.password=

或者

# 驱动类名称
spring.datasource.druid.driver-class-name=com.mysql.cj.jdbc.Driver
# 数据库连接的URL
spring.datasource.druid.url=jdbc:mysql://localhost:3306/ mybatistest
# 连接数据库的用户名
spring.datasource.druid.username=root
# 连接数据库的密码
spring.datasource.druid.password=

lombok

Lombok是一个实用的的java类库,能够通过注解的形式自动生成构造器,getter/setter,equals,hascode,toString等方法,并可以自动化生成日志变量,简化java开发,提高效率

注解
首先在pom.xml中提供lombok的依赖


        <dependency>
            <groupId>org.projectlombok</groupId>
            <artifactId>lombok</artifactId>
        </dependency>

注意:这里不需要版本号,因为在父工程依赖中已经集成lombok并对其版本进行管理


/*@Getter
@Setter
@ToString
@EqualsAndHashCode*/ //这四个注解集成到一个注解当中
@Data
@NoArgsConstructor  //无参构造注解
@AllArgsConstructor  //全参构造注解
public class User {
    private Integer id;
    private String name;
    private Integer age;
    private Integer gender;
    private String phone;
}

Mybatis基础操作

  • 删除
    根据主键传回来的id进行删除

此处用到mybatis提供的参数占位符 #{}

@Mapper
public interface EmpMapper {
    @Delete("delete from emp where id=#{id}}")
    public void delete(Integer id);
}

在测试类中进行测试

@SpringBootTest
class Projecttest1ApplicationTests {
    @Autowired
    private EmpMapper empMapper;
    @Test
    void contextLoads() {
        empMapper.delete(2);
    }

}

注意:如果Mapper接口方法形参只有一个普通类型的参数,#{…}里面的属性名可以随便写,如#{id},#{value}

  • 新增

    @Insert("insert into user (username,name,gender,image,job,entrydate,dept_id,create_time,update_time) values(#{username},#{name},#{gender},#{image},#{job},#{entrydate},#{deptId},#{createTime},#{updateTime})")
    public void insert(Emp emp);

主键返回功能:
在数据添加成功后,需要获取插入数据库数据的关键
需要注解 @options(keyProperty=‘id’,useGeneratedKeys=true) (第一个参数表示获取的主键封装到哪个属性,第二个参数表示我们要返回主键值) //注解会自动生成主键值,赋值给emp对象的id属性值

  • 更新
    //更新数据
    @Update("update emp set username=#{username} ... where id=#{id}")
    public void update(Emp emp);
  • 查询(条件查询)

举例一个条件,根据输入的姓名,性别,入职时间,搜索满足条件的员工信息,其中姓名支持模糊匹配,对应的sql语句是

select * from emp where name like '%张%' and gender=1 and entrydate between '' and '' order by update_time desc;

对应的操作:


    //条件查询员工信息
    @Select("select * from emp where name like '%#{name}%' "+
            "and gender=#{gender} and entrydate between #{begin} and #{end} order by update_time desc")
    public List<Emp> list(String name, Short gender, LocalDate begin,LocalDate end);
}

注意: #{} 不能出现在引号当中,需要把此处的#换成$
但是这样就出现了性能低,不安全,存在SQL注入的问题,所以我们还可以使用mysql提供的字符串拼接函数concat
上述就可以修改为

concat('%',#{name},'%')
  • 数据封装
    实体类属性名和数据库表查询返回的字段名一致,mybatis会自动进行封装
    如果实体类属性名和数据库表查询返回的字段名不一致,不能自动封装
    如何解决?

方法一:给字段起别名,让别名与实体类属性一致
方法二:通过 @Results@Result 注解手动映射封装

@Results({@Result(column="",property="")}) //Results就是一个一个的Result注解,注解中定义映射,前一个参数时表中的字段名,后一个参数表示需要映射到的属性名

(推荐使用)方法三:开启mybatis的驼峰命名自动映射开关(下划线会自动封装到对应的驼峰命名的属性)
需要在spring配置文件中进行配置
mybatis.configuration.map-underscore-to-camel-case=true

日志输出

虽然我们使用mybatis进行数据操作,但是我们并不知道mybatis底层是如何执行操作的,我们可以在spring配置文件中打开mybatis的日志,并指定输出到控制台

配置信息:
mybatis.configuration.log-impl=org.apache.lbatis.logging.stdout.StdOutImpl
意思是要往控制台传输日志
输出结果如下:
预编译
此处是SQL预编译,在运行的时候系统将两条消息一起发送给数据库
为什么这样做:预编译SQL性能更高,更安全(防止SQL注入)
为什么性能更高:


java中sql语句执行的流程:

在这里插入图片描述java将sql语句传入数据库中,首先通过前三个内容,处理完后语句停留在缓存当中,当再次遇上相同语句的时候,将不需要前面的步骤,可以直接执行,但是如果没有预编译直接传入带有id的语句,可能每次都碰不到相同的语句,但是通过预编译采用参数占位符,这样同种语句会更方便识别,效率更高
为什么更加安全:

  • SQL注入

SQL注入是通过操作输入的数据来修改事先定义好的SQL语句,以达到执行代码对服务器进行攻击的方法

进行SQL注入的演示:
在一个用户登录的界面,登录进入后台需要填写用户名和密码
当我们在密码栏中写入:

‘ or '1' = '1'

时发现登录成功,实际上用户的登录就是将输入的用户名和密码在数据库中进行查找,当查询到有相同数据便是登陆成功,但是输入上述数据后,查询操作就会变为:

select count(*) from emp where username='' and password='' or '1'='1';

这样查询操作永远为真,就会修改查询操作,这就是SQL注入

在mybatis开发中使用#{}下来就是预编译语句,#{}最终会被?替代掉
还存在另外一种${},这是直接将参数拼接在sql语句后,存在sql注入问题


## XML映射文件
  • 规范
  1. XML映射文件的名称与Mapper接口名称一致,并且将XML映射文件和Mapper接口放置在相同包下(同包同名)
  2. XML映射文件的namespace属性为Mapper接口全限定名一致
  3. XML映射文件中sql语句的id与Mapper接口中的方法名一致,并保持返回类型一致
  • 配置XML映射文件

首先根据规范配置xml文件,在xml中配置sql语句:
在这里插入图片描述
在mybatis中文网中找到xml的配置信息复制到xml文件当中

<?xml version="1.0" encoding="UTF-8" ?>
<!DOCTYPE mapper
        PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN"
        "http://mybatis.org/dtd/mybatis-3-mapper.dtd">
<mapper namespace="com.example.projecttest1.mapper.EmpMapper">
    <!--resultType :单条记录所封装的类型-->
    <select id="list" resultType="com.example.projecttest1.pojo.Emp">
        select * from emp where name like '%#{name}%'and gender=#{gender} and entrydate between #{begin} and #{end} order by update_time desc
    </select>

</mapper>

使用注解来映射简单语句会使代码显得更加简洁,但对于稍微复杂一点的语句,Java 注解不仅力不从心,还会让你本就复杂的 SQL 语句更加混乱不堪。 因此,如果你需要做一些很复杂的操作,最好用 XML 来映射语句。

  • MybatisX

MybatisX是一款基于idea的快速开发Mybatis的插件,为效率而生
可以通过点击前面的标识快速的实现他们之间的互相定位

Mybatis的动态SQL

随着用户的输入或外部条件的变化而变化的SQL语句,我们称为动态SQL

通俗的来说,就好比我需要三个条件进行查询,我只前将三个条件固定写入sql,如果后期我只通过两个条件进行查询另一个就是null,这样就查不到数据

我们可以通过if标签进行有条件的拼接:
在这里插入图片描述

  • <if>
    进行有条件的sql语句拼接
  • <where>
    在这里插入图片描述
    会自动去除拼接后多余的and
  • <set>

使用set标签将更新字段进行包裹,动态的在首行创建set标签,可以去除多余的特殊字符
在这里插入图片描述

  • <foreach>

属性:
collection:要遍历的集合
item:遍历出来的元素
separator:分隔符
open:遍历开始前拼接的sql片段
close:遍历结束之后拼接的sql片段

  • <sql><include>标签

当程序中sql语句出现大量重复的时候,为了提高程序的可复用性,我们可以采取sql和include标签将重复的sql语句进行抽取和复用

通过<sql>进行抽取,在sql属性id给定一个唯一的id值
例如:

<sql id="commonSelect">
select ....
</sql>

然后通过<include>标签进行复用,在需要sql语句的地方:

<include refid="commonSelect"/>

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

海豚【货运系统源码】货运小程序【用户端+司机端app】源码物流系统搬家系统源码师傅接单

技术栈&#xff1a;前端uniapp后端vuethinkphp 主要功能&#xff1a; 不通车型配置不通价格参数 多城市定位服务 支持发货地 途径地 目的地智能费用计算 支持日期时间 预约下单 支持添加跟单人数选择 支持下单优惠券抵扣 支持司机收藏订单评价 支持订单状态消息通知 支…

uniapp数组合并函数使用几录

let that { listAll: [1, 2, 3] }; let data [4, 5, 6]; let mergedArray that.listAll.concat(data); console.log(mergedArray); // 输出: [1, 2, 3, 4, 5, 6] console.log(that.listAll); // 输出: [1, 2, 3]&#xff0c;原始数组没有改变 唯有美景&#xff0c;可以抚…

1.ByteBuffer介绍

一、Buffer基本使用 /*** buffer正确使用姿势* 1.向buffer写入数据&#xff0c;调用channel.read(buffer)* 2.调用flip方法切换到读模式* 3.从buffer读数据&#xff0c;通过get方法&#xff0c;每次读取一个字节或字符* 4.调用clear方法或者compact方法切换到写模式*/ Slf4j p…

静态住宅IP优缺点,究竟要怎么选?

在进行海外 IP 代理时&#xff0c;了解动态住宅 IP 和静态住宅 IP 的区别以及如何选择合适的类型非常重要。本文将介绍精态住宅 IP 特点和&#xff0c;并提供选择建议&#xff0c;帮助您根据需求做出明智的决策。 静态住宅 IP 的特点 静态住宅 IP 是指 IP 地址在一段时间内保…

2-linux磁盘常用的命令lsblk

目录 1.lsblk常用命令 2.1-lsblk -a&#xff1a;显示所有设备&#xff08;包括空设备&#xff09;的信息。 2.2 lsblk -f&#xff1a;显示文件系统类型。 2.3 lsblk -m&#xff1a;显示磁盘和分区的归属帐号权限 1.lsblk常用命令 lsblk的英文&#xff1a;List block 列出磁…

windows下QT如何集成OpenCV

说明 我在windows下使用QT Creator12创建的CMake项目&#xff0c;需要OpenCV的一些功能。由于安装的时候我选择的QT组件都是MInGW的&#xff0c;所以无法使用VS studio版本的dll库。 为什么vs的版本不能用 我安装QT选择的是MinGW版本&#xff0c;本地编译QT工程只能选择MinG…

安全SCDN的威胁情报库对DDOS防护有什么好处

目前网络攻击事件频频发生&#xff0c;DDoS&#xff08;分布式拒绝服务&#xff09;攻击已成为各种企业&#xff08;小到区域性小公司大到各种跨国公司&#xff09;的主要威胁&#xff0c;DDoS 攻击可能会对企业造成重大损害和破坏&#xff0c;比如对目标公司的业务造成产生不利…

OpenHarmony无人机MAVSDK开源库适配方案分享

MAVSDK 是 PX4 开源团队贡献的基于 MavLink 通信协议的用于无人机应用开发的 SDK&#xff0c;支持多种语言如 C/C、python、Java 等。通常用于无人机间、地面站与通信设备的消息传输。 MAVLink 是一种非常轻量级的消息传递协议&#xff0c;用于与无人机&#xff08;以及机载无…

Halcon3D表面平面度检测-平面差值法

//倾斜平面矫正 https://blog.csdn.net/m0_51559565/article/details/137146179前言 通常我们对表面平面度进行检测时&#xff0c;通常使用2种方式。1&#xff1a;通过大卷积核的高斯滤波进行拟合平面&#xff0c;然后求取拟合平面与3D模型间的点间的距离。2&#xff1a;通过平…

向机器人传授人类社会同步的艺术

在数字时代&#xff0c;社交机器人正迅速成为主流社会的一部分&#xff0c;从培训医生和教育儿童到提供谈话疗法和客户服务。现在&#xff0c;这是一个价值数十亿美元的全球产业&#xff0c;对具有类似人类社交智能的机器人的需求不断增长&#xff0c;这标志着我们技术史上的一…

CavalierContours 二维线操作

CavalierContours 二维线操作 2D polyline library for offsetting, combining, etc. 用于偏移、交并补等组合等操作的 2D 多折段线库。 Polyline Structure 多段线结构 Polylines are defined by a sequence of vertexes and a bool indicating whether the polyline is cl…

深度学习入门简单实现一个神经网络

实现一个三层神经网络 引言测试数据 代码 引言 今天我们实现一个简单的神经网络 俩个输入神经元 隐藏层两个神经元 一个输出神经元 激活函数我们使用sigmoid 优化方法使用梯度下降 我们前期准备是需要把这些神经元的关系理清楚 x1&#xff1a;第一个输入 x2&#xff1a;第二个…

Linux系统----------探索mysql数据库MHA高可用

目录 一、MHA概述 1.1 什么是 MHA 1.2MHA 的组成 1.2.1MHA Node&#xff08;数据节点&#xff09; 1.2.2MHA Manager&#xff08;管理节点&#xff09; 1.3MHA 的特点 1.4MHA工作原理 1.5数据同步的方式 1.5.1同步复制 1.5.2异步复制 1.5.3半同步复制 二、搭建 MySQ…

8.6 循环神经网络的简洁实现

每个步长共用参数 加载数据 虽然 8.5节 对了解循环神经网络的实现方式具有指导意义&#xff0c;但并不方便。 本节将展示如何使用深度学习框架的高级API提供的函数更有效地实现相同的语言模型。 我们仍然从读取时光机器数据集开始。 import torch from torch import nn from…

华为流量整形配置

组网需求 如图1所示&#xff0c;企业网内部LAN侧的语音、视频和数据业务通过Switch连接到RouterA的Eth2/0/0上&#xff0c;并通过RouterA的GE3/0/0连接到WAN侧网络。 不同业务的报文在LAN侧使用802.1p优先级进行标识&#xff0c;在RouterA上根据报文的802.1p优先级入队列&…

Flutter环境搭建及版本管理

前言 Flutter已成为移动应用开发的主流选择&#xff0c;其跨平台优势日益凸显&#xff0c;使学习Flutter成为现代开发者的必备技能。越来越多的公司和团队选择Flutter构建应用&#xff0c;这为开发者提供了更广阔的就业机会和职业发展空间。Flutter的热度源自其强大的特性和生…

【动手学深度学习-pytorch】-9.3深度循环神经网络

到目前为止&#xff0c;我们只讨论了具有一个单向隐藏层的循环神经网络。 其中&#xff0c;隐变量和观测值与具体的函数形式的交互方式是相当随意的。 只要交互类型建模具有足够的灵活性&#xff0c;这就不是一个大问题。 然而&#xff0c;对一个单层来说&#xff0c;这可能具有…

WebCopilot:一款功能强大的子域名枚举和安全漏洞扫描工具

关于WebCopilot WebCopilot是一款功能强大的子域名枚举和安全漏洞扫描工具&#xff0c;该工具能够枚举目标域名下的子域名&#xff0c;并使用不同的开源工具检测目标存在的安全漏洞。 工具运行机制 WebCopilot首先会使用assetsfinder、submaster、subfinder、accumt、finddom…

layui laydate日期初始化的一些坑

layui laydate日期初始化的一些坑 背景坑一&#xff1a;利用class属性初始化时间控件失败坑二&#xff1a;后加载页面时间控件初始化失败坑三&#xff1a;结束时间需要默认追加23:59:59 背景 在日常开发中&#xff0c;总是会不可避免的用到日期插件&#xff0c;不同的日期插件…

IoT网关在智能制造工厂生产线监控与管理中的应用-天拓四方

随着工业4.0时代的到来&#xff0c;智能制造已成为工业发展的重要方向。IoT网关在智能制造工厂中扮演着关键角色&#xff0c;它能够实现设备间的互联互通、数据的实时采集与处理&#xff0c;以及生产线的智能监控与管理。本案例将详细介绍IoT网关在智能制造工厂生产线监控与管理…