【SpringBoot】mybatis基础操作

mybatis入门

1.mybatis准备操作

创建数据库:

CREATE DATABASE mybatis_test DEFAULT CHARACTER SET utf8mb4;
-- 使⽤数据数据
USE mybatis_test;
-- 创建表[用户表]
DROP TABLE IF EXISTS userinfo;
CREATE TABLE `userinfo` (
`id` INT ( 11 ) NOT NULL AUTO_INCREMENT,
`username` VARCHAR ( 127 ) NOT NULL,
`password` VARCHAR ( 127 ) NOT NULL,
`age` TINYINT ( 4 ) NOT NULL,
`gender` TINYINT ( 4 ) DEFAULT '0' COMMENT '1-男 2-⼥ 0-默认',
`phone` VARCHAR ( 15 ) DEFAULT NULL,
`delete_flag` TINYINT ( 4 ) DEFAULT 0 COMMENT '0-正常, 1-删除',
`create_time` DATETIME DEFAULT now(),
`update_time` DATETIME DEFAULT now(),
PRIMARY KEY ( `id` )
) ENGINE = INNODB DEFAULT CHARSET = utf8mb4;
-- 添加用户信息
INSERT INTO mybatis_test.userinfo ( username, `password`, age, gender, phone )
VALUES ( 'admin', 'admin', 18, 1, '18612340001' );
INSERT INTO mybatis_test.userinfo ( username, `password`, age, gender, phone )
VALUES ( 'zhangsan', 'zhangsan', 18, 1, '18612340002' );
INSERT INTO mybatis_test.userinfo ( username, `password`, age, gender, phone )
VALUES ( 'lisi', 'lisi', 18, 1, '18612340003' );
INSERT INTO mybatis_test.userinfo ( username, `password`, age, gender, phone )
VALUES ( 'wangwu', 'wangwu', 18, 1, '18612340004' );

创建项目添加依赖:

image-20240121181437837

创建实体类:

@Data
public class UserInfo {private Integer id;private String username;private String password;private Integer age;private Integer gender;private String phone;private Integer deleteFlag;private Date createTime;private Date updateTime;
}

配置文件,配置连接数据库的属性:

spring:datasource:url: jdbc:mysql://127.0.0.1:3306/mybatis_test?characterEncoding=utf8&useSSL=falseusername: rootpassword: abc123

2.写持久层代码

@Mapper
public interface UserMapper {@Select("select * from userinfo")//方法实现public List<UserInfo> getAll();//方法声明
}

3.编写测试类

在持久层代码中右击,点击Generate->点击Test

image-20240121180947187

然后勾选如下图,点击OK.就创建了测试类,就可以在测试类中编写测试代码

image-20240121181059770

@Slf4j 
@SpringBootTest //启动spring容器
class UserMapperTest {@BeforeEach //每个测试方法之前运行void setUp() {log.info("setUp");}@AfterEach //每个测试方法之后运行void tearDown() {log.info("tearDown");}@Autowiredprivate UserMapper userMapper;@Test //测试方法void getAll() {List<UserInfo> userInfoList = userMapper.getAll();log.info(userInfoList.toString());}
}

mybatis注解基础操作

1.配置打印日志

配置MyBatis日志打印,建议只出现在开发环境中,不要出现在线上环境

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

2.增加

我们增的时候需要进行传递参数。返回值为受影响的行数。

实体层代码:

@Insert("insert into userinfo (username,password,age,gender,phone)" +"values (#{username},#{password},#{age},#{gender},#{phone})")
Integer insert(UserInfo userInfo);

测试类中的测试代码:

@Test
void insert() {UserInfo userInfo = new UserInfo();userInfo.setUsername("张三");userInfo.setPassword("123456");userInfo.setAge(18);userInfo.setGender(1);userInfo.setPhone("1234444444");System.out.println(userMapper.insert(userInfo));
}

如果我们增加的时候使用@Param进行对象重命名就会报错:

@Insert("insert into userinfo (username,password,age,gender,phone)" +"values (#{username},#{password},#{age},#{gender},#{phone})")
Integer insert(@Param("userInfo") UserInfo userInfo);

要想解决这个问题#{…}需要使⽤参数.属性来获取

@Insert("insert into userinfo (username,password,age,gender,phone)" +"values (#{userInfo.username},#{userInfo.password},#{userInfo.age},#{userInfo.gender},#{userInfo.phone})")
Integer insert(@Param("userInfo") UserInfo userInfo);

上面返回值为受影响的行数。要想返回的是自增Id需要使用@Options注解。keyProperty表示自增ID赋值给哪个字段。

@Options(useGeneratedKeys = true,keyProperty = "id")
@Insert("insert into userinfo (username,password,age,gender,phone)" +"values (#{userInfo.username},#{userInfo.password},#{userInfo.age},#{userInfo.gender},#{userInfo.phone})")
Integer insert(@Param("userInfo") UserInfo userInfo);

测试类中的代码:

@Test
void insert() {UserInfo userInfo = new UserInfo();userInfo.setUsername("张三");userInfo.setPassword("123456");userInfo.setAge(18);userInfo.setGender(1);userInfo.setPhone("1234444444");Integer result = userMapper.insert(userInfo);System.out.println("insert方法执行结果:" + result + ", 自增id:" + userInfo.getId());
}

3.删除

使用@Delete注解,实体层代码:

@Delete("delete from userinfo where id = #{id}")
Integer delete(Integer id);

测试类中的代码:

@Test
void delete() {userMapper.delete(6);
}

4.修改

使用@Update注解,实体层代码:

@Update("update userinfo set age = #{age} where id = #{id}")
Integer update(UserInfo userInfo);

测试类中的代码:

@Test
void update() {UserInfo userInfo = new UserInfo();userInfo.setId(5);userInfo.setAge(38);userMapper.update(userInfo);
}

5.查找

查询全部用户和根据id查询用户,实体层代码:

企业开发中,尽量不要使用*,需要查询哪些字段,就写哪些字段。如果需要全量的字段,就全部写完。

@Select("select id, username, password, age, gender, phone, delete_flag, create_time, update_time from userinfo")
public List<UserInfo> selectAll();
@Select("select id, username, password, age, gender, phone, delete_flag, create_time, update_time from userinfo where id = #{id}")
public List<UserInfo> selectOne(Integer id);

测试类中的代码:

@Test
void selectOne() {System.out.println(userMapper.selectOne(1));
}@Test
void selectAll() {System.out.println(userMapper.selectAll());
}

字段映射

查的时候数据库中的字段名称和java类中的属性名称相互对应,如果不同我们可以采用以下三种方式去解决:

1.起别名(不建议)

@Select("select id, username, password, age, gender, phone, delete_flag as deleteFlag, create_time as createTime, update_time as updateTime from userinfo")
public List<UserInfo> selectAll();

2.使用@Results注解

如果字段和Java属性名称一样,可以省略。

@Results(id = "BaseMap",value = {@Result(column = "delete_flag",property = "deleteFlag"),@Result(column = "create_time",property = "createTime"),@Result(column = "update_time",property = "updateTime")
})
@Select("select id, username, password, age, gender, phone, delete_flag, create_time, update_time from userinfo")
public List<UserInfo> selectAll();
@Results(id = "BaseMap")//复用Results定义
@Select("select id, username, password, age, gender, phone, delete_flag, create_time, update_time from userinfo where id = #{id}")
public List<UserInfo> selectOne(Integer id);

3.使用配置文件,自动转驼峰

这个必须java类中的属性,和数据库中的字段是相对应的。如果不是就建议用注解的方式

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

mybatis XML配置文件基础操作

1.配置连接字符串和MyBatis

spring:datasource:url: jdbc:mysql://127.0.0.1:3306/mybatis_test?characterEncoding=utf8&useSSL=falseusername: rootpassword: abc123driver-class-name: com.mysql.cj.jdbc.Driver
mybatis:configuration: # 配置打印 MyBatis日志log-impl: org.apache.ibatis.logging.stdout.StdOutImplmap-underscore-to-camel-case: true #配置驼峰自动转换
# 配置 mybatis xml 的文件路径,在 resources/mapper 创建所有表的 xml 文件mapper-locations: classpath:mapper/**Mapper.xml

classpath中的文件路径和文件夹名是在resources创建的文件夹是相对应的。

image-20240123174319684

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.mybatisdemo.mapper.UserXMLMapper"><select id="selectAll" resultType="com.example.mybatisdemo.entity.UserInfo">select * from userinfo</select>
</mapper>

实体层的代码:

import java.util.List;
@Mapper
public interface UserXMLMapper {List<UserInfo> selectAll();
}

测试类代码:

@SpringBootTest
class UserXMLMapperTest {@AutowiredUserXMLMapper userXMLMapper;@Testvoid selectAll() {System.out.println(userXMLMapper.selectAll());}
}

image-20240123180150423

如果出现没有找到对应的实现,可能原因:

  1. xml和接口定义的方法名称不一致
  2. mapper的路径配置和xml的路径不一样
  3. xml namespace写错了

2.增加

实体层java代码:

Integer insert(UserInfo userInfo);

xml代码:

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

我们用XML文件进行书写sql代码,返回的数据时影响的行数,要想获取插入数据的自增id.我们做如下改变:

<insert id="insert" useGeneratedKeys="true" keyProperty="id">insert into userinfo(username,password,age,gender,phone)values (#{username},#{password},#{age},#{gender},#{phone})
</insert>

我们对参数重命名也会产生和注解的方式一样报错,解决方式也是#{}内部使用对象.属性的方式进行解决。

实体层代码:

Integer insert(@Param("userInfo") UserInfo userInfo);

xml文件代码:

<insert id="insert" useGeneratedKeys="true" keyProperty="id">insert into userinfo(username,password,age,gender,phone)values (#{userInfo.username},#{userInfo.password},#{userInfo.age},#{userInfo.gender},#{userInfo.phone})
</insert>

3. 删除

实体层代码:

Integer delete(Integer id);

xml文件代码

<delete id="delete">delete from userinfo where id = #{id}
</delete>

4. 修改

实体层代码:

Integer update(UserInfo userInfo);

xml文件代码

<update id="update">update userinfo set gender = #{gender} where id = #{id}
</update>

测试类代码:

@Test
void update() {UserInfo userInfo = new UserInfo();userInfo.setId(9);userInfo.setGender(2);System.out.println(userXMLMapper.update(userInfo));
}

5.查找

字段映射

使用xml文件查找,数据库字段和java类的属性之间和只用注解有相同的问题,字段名和属性名不同就不能进行映射

我们有三种解决方法:

  1. sql字段起别名
  2. 定义Result
  3. 配置自动转驼峰

第一条和第三条。和上面使用注解的解决方式是一样的。

主要介绍第二种定义Result

<resultMap id="XMLBaseMap" type="com.example.mybatisdemo.entity.UserInfo"><id column="id" property="id"></id><result column="deleteFlag" property="deleteFlag"></result><result column="create_time" property="createTime"></result><result column="update_time" property="updateTime"></result>
</resultMap><select id="selectAll" resultMap="XMLBaseMap">select * from userinfo
</select>

image-20240123203315466

其他查询操作

1.多表查询

多表查询操作,在工作中尽量避免使用多表查询,尤其是对性能要求非常高的项目。多表查询必然是慢的。

多表查询准备工作:

创建数据库表:

-- 创建文章表
DROP TABLE IF EXISTS articleinfo;
CREATE TABLE articleinfo (id INT PRIMARY KEY AUTO_INCREMENT,title VARCHAR ( 100 ) NOT NULL,content TEXT NOT NULL,uid INT NOT NULL,delete_flag TINYINT ( 4 ) DEFAULT 0 COMMENT '0-正常, 1-删除',create_time DATETIME DEFAULT NOW(),update_time DATETIME DEFAULT NOW()
) DEFAULT CHARSET 'utf8mb4';
-- 插入测试数据
INSERT INTO articleinfo ( title, content, uid ) VALUES ( 'Java', 'Java正文', 1
);

创建实体类:

@Data
public class ArticleInfo {Integer id;String title;String content;Integer uid;String deleteFlag;Date createTime;Date updateTime;
}

创建多表查询结果的实体类:

@Data
@ToString(callSuper = true)
public class ArticleInfoVO extends ArticleInfo{String username;Integer age;
}

多表查询的实体层:

@Mapper
public interface ArticleMapper {//根据文章Id获取作者相关信息@Select("select * from articleinfo ta " +" left join userinfo tu on ta.uid = tu.id where ta.id = #{id}")ArticleInfoVO selectArticleAndUserById(Integer id);
}

多表查询的测试代码:

@SpringBootTest
class ArticleMapperTest {@AutowiredArticleMapper articleMapper;@Testvoid selectArticleAndUserById() {System.out.println(articleMapper.selectArticleAndUserById(1));}
}

2.#{}和${}

使用#和使用KaTeX parse error: Expected 'EOF', got '#' at position 13: 都可以进行获取变量的值。#̲{}是预编译SQL,{}是即时SQL.

使用#{}通过名字查询用户

@Select("select * from userinfo where username = #{username}")
List<UserInfo> selectByName(String username);

使用KaTeX parse error: Expected 'EOF', got '#' at position 14: {}通过名字查询用户,使用#̲时,如果参数为String,会…不会,$符号是直接拼接,如果是字符串类型,需要加上‘’

@Select("select * from userinfo where username = '${username}'")
List<UserInfo> selectByName(String username);

image-20240124134628709

排序时,不能使用#{},如果使用#会给参数加上",使用排序的时候必须使用#{},但是#{}不安全会造成sql注入。使用之前需要进行参数检验。使用排序的时候一共就两种可能,也可以设置两个方法,或者使用参数校验是否是两个参数中的一个

@Select("select * from userinfo order by id ${sort}")
List<UserInfo> selectUserBySort(String sort);

使用模糊查询不能直接使用#{}因为会自动加上’',使用${}会有sql注入的问题。

模糊查询使用CONCAT来连接字符串。可以避免字符串拼接造成的sql注入。

@Select("select * from userinfo where username like CONCAT('%',#{username},'%')")
List<UserInfo> selectUserByLike(String username);

#{}和${}的区别。

其中之一就是预编译SQL和即时SQL的区别

  1. 预编译SQL性能更高
  2. 预编译SQL不存在SQL注入的问题

排序时不能使用#{}。表名,字段名等作为参数时,也不能使用#。模糊查询时,如果使用#,需要搭配mysql的内置函数,concat,而不能直接使用。

实际开发中,能使用#的,都使用#。使用$时,一定要考虑到SQL注入的问题。

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

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

相关文章

LeetCode刷题---分隔链表

解题思路: 根据题意对链表进行分割 创建链表l用来表示小于给定值x的链表,创建辅助链表left指向l的初始节点 创建链表r用来表示大于等于给定值x的链表&#xff0c;创建辅助链表right指向r的初始节点 创建辅助链表cur&#xff0c;指向head节点 对原链表进行遍历 如果当前节点的va…

Flink多流转换(2)—— 双流连结

双流连结&#xff08;Join&#xff09;&#xff1a;根据某个字段的值将数据联结起来&#xff0c;“配对”去做处理 窗口联结&#xff08;Window Join&#xff09; 可以定义时间窗口&#xff0c;并将两条流中共享一个公共键&#xff08;key&#xff09;的数据放在窗口中进行配…

五、垃圾回收

1. 垃圾回收基础 1.1 什么是垃圾 简单说就是&#xff1a;内存中已经不再被使用到的内存空间就是垃圾。 1.2 如何判定是垃圾 1.2.1 引用计数法 引用计数法&#xff1a;给对象添加一个引用计数器&#xff0c;有访问就 1&#xff0c;引用失效就 -1 引用计数法的优缺点&#…

抖音信息流广告引流,这种方法你要知道-数灵通

随着抖音的普及&#xff0c;我们经常会在刷视频的过程中遇到各种广告。这些广告不仅种类繁多&#xff0c;而且形式各异。除了常见的开屏广告和达人合作广告&#xff0c;信息流广告也是抖音广告的一种重要形式。那么&#xff0c;什么是信息流广告呢&#xff1f; 信息流广告是一种…

03. 静态路由

文章目录 一. 静态路由概述1.1. 概述1.2. 路由信息获取方式1.3. 路由表的参数1.4. 路由协议的优先级1.5. 最优路由条目优先1.6. 最长前缀匹配原则 二. 实验实操2.1. 实验1&#xff1a;静态路由2.1.1. 实验目的2.1.2. 实验拓扑图2.1.3. 实验步骤&#xff08;1&#xff09;配置网…

数据结构——双链表

双链表中节点类型的描述&#xff1a; 双链表的初始化&#xff08;带头结点&#xff09; 、 双链表的插入操作 后插操作 InsertNextDNode(p, s): 在p结点后插入s结点 按位序插入操作&#xff1a; 思路&#xff1a;从头结点开始&#xff0c;找到某个位序的前驱结点&#xff…

周鸿祎回应坚定支持华为:因为 360 也被制裁了

在昨天的华为鸿蒙生态千帆启航仪式上&#xff0c;360集团创始人兼CEO周鸿祎发表演讲表示&#xff0c;360坚定地支持华为的决定源于双方都曾遭到制裁。周鸿祎在演讲中提到&#xff1a;“在华为最早被制裁的时候&#xff0c;我们是少数几个公开站出来坚定支持华为的公司。其实也很…

如何进行H.265视频播放器EasyPlayer.js的中性化设置?

H5无插件流媒体播放器EasyPlayer属于一款高效、精炼、稳定且免费的流媒体播放器&#xff0c;可支持多种流媒体协议播放&#xff0c;可支持H.264与H.265编码格式&#xff0c;性能稳定、播放流畅&#xff0c;能支持WebSocket-FLV、HTTP-FLV&#xff0c;HLS&#xff08;m3u8&#…

【域名解析】如何将域名指向对应服务器IP

目录 &#x1f337;一、域名解析基本概念 &#x1f33c;1. 定义 &#x1f33c;2. 域名解析类型 &#x1f337;二、域名解析服务器IP地址 &#x1f33c;1. 操作步骤 &#x1f33c;2. 验证 &#x1f337;一、域名解析基础知识 &#x1f33c;1. 基本概念 定义&#xff1a; …

luceda ipkiss教程 58:输出器件的版图和三维模型

在ipkiss中&#xff0c;通过visualize_3d_povray可以输出包含器件的三维模型参数的.pov文件&#xff0c;再通过POV-Ray&#xff08;免费软件&#xff0c;下载地址&#xff1a;https://www.povray.org/download/&#xff09;就可以查看器件的三维模型。 如&#xff1a; 代码如…

如何安装MeterSphere并实现无公网ip远程访问服务管理界面

文章目录 前言1. 安装MeterSphere2. 本地访问MeterSphere3. 安装 cpolar内网穿透软件4. 配置MeterSphere公网访问地址5. 公网远程访问MeterSphere6. 固定MeterSphere公网地址 正文开始前给大家推荐个网站&#xff0c;前些天发现了一个巨牛的 人工智能学习网站&#xff0c; 通…

vivado 接口、端口映射

接口 重要&#xff01;接口只能在“fpga”类型的&#xff1c;component&#xff1e;中定义。接口部分提供了<component>上所有可用物理接口的列表。<interfaces>部分包含嵌套在其中的一个或多个<interface>标记。一个接口是通过使用<port_map>标记由多…

Spring 声明式事务讲解,和 @Transactional注解的用法

目录 一、Spring框架介绍 二、什么是声明式事务 三、如何解决并发性事务问题 四、Transactional注解的用法 一、Spring框架介绍 Spring框架是一个开源的Java应用程序开发框架&#xff0c;旨在简化企业级Java应用程序的开发。它提供了一种轻量级的、全面的编程和配置模型&a…

进阶C语言-自定义类型

为了便于描述复杂的对象,C语言就支持了自定义类型&#xff0c;其中包括了结构体、枚举和联合体&#xff0c;下面将为大家一一介绍。 自定义类型 &#x1f388;1.结构体&#x1f50e;1.1结构的基础知识&#x1f50e;1.2结构的声明&#x1f50e;1.3特殊的声明&#x1f50e;1.4结构…

基于springboot在线学习平台源码和论文

在Internet高速发展的今天&#xff0c;我们生活的各个领域都涉及到计算机的应用&#xff0c;其中包括学习平台的网络应用&#xff0c;在外国学习平台已经是很普遍的方式&#xff0c;不过国内的管理平台可能还处于起步阶段。学习平台具有学习信息管理功能的选择。学习平台采用ja…

每天掌握一个软测高级技巧:接口自动化神器apin进阶操作

之前写了一篇关于接口自动化框架 apin 入门使用是文章&#xff0c;主要介绍了 apin 的安装以及用例编写的方法。 今天这篇文章来给大家聊聊&#xff0c;apin 中的一些高级使用技巧。比如依赖接口的变量提取和引用&#xff0c;用例断言&#xff0c;以及函数工具的使用。 01 变…

web安全学习笔记【09】——算法2

基础[1] 入门-算法逆向&散列对称非对称&JS源码逆向&AES&DES&RSA&SHA #知识点&#xff1a; 1、Web常规-系统&中间件&数据库&源码等 2、Web其他-前后端&软件&Docker&分配站等 3、Web拓展-CDN&WAF&OSS&反向&负载…

Web09--jQuery基础

1、jQuery概述 1.1 什么是jQuery jQuery是一款优秀的JavaScript的轻量级框架之一&#xff0c;封装了DOM操作、事件绑定、ajax等功能。特别值得一提的是基于jQuery平台的插件非常丰富&#xff0c;大多数前端业务场景都有其封装好的工具可直接使用。 jQuery下载和版本介绍 官…

Make.com的发送邮件功能已经登峰造极

make.com的发送邮件功能已经做到了登峰造极。 我给你个任务&#xff0c;让你发送个新邮件给谁谁&#xff0c;你一定想到SMTP服务器不就行了。 我给你第二个任务&#xff0c;我让你自动回复一个邮件&#xff0c;注意是回复。 做不到了吧&#xff5e;&#xff5e;&#xff01;…

TS基础知识点快速回顾(上)

基础介绍 什么是 TypeScript&#xff1f; TypeScript&#xff0c;简称 ts&#xff0c;是微软开发的一种静态的编程语言&#xff0c;它是 JavaScript 的超集。 那么它有什么特别之处呢? js 有的 ts 都有&#xff0c;所有js 代码都可以在 ts 里面运行。ts 支持类型支持&#…