前面我们学习了什么是 MyBatis,为什么要使用 MyBatis,如何创建 MyBatis 环境,并且了解了使用单元测试对代码功能进行测试。那么今天这篇文章将为大家分享关于 MyBatis 的基础操作。
书写 MyBatis 的方法有两种:注解和XML的方法,这里我们两种方法都是给大家写到。
前面文章中也提到了日志对于我们开发人员的重要性,在使用 MyBatis 框架的过程中,我们同样也可以看到执行 MyBatis 代码的过程中产生的日志。
那么如何打印出 MyBatis 日志呢?我们需要在配置文件中添加配置项来告知 Spring,我们需要知道 MyBatis 代码执行过程中产生的日志。
mybatis:
configuration: # 配置打印 MyBatis?志
log-impl: org.apache.ibatis.logging.stdout.StdOutImpl
添加这个配置了之后,我们在启动项目就会发现,MyBatis 执行过程中的日志就可以看到了。
假设我们需要查询 id 为 4 的用户的信息的时候,对应的 SQL 语句就是 select * from userinfo where id=4;
,对应到我们的 MyBatis 就是这样的。
@Select("select * from userinfo where id=4")
public UserInfo getById();
但是可以发现,这样写 SQL 语句的话,这个查询的条件就写死了,那么是否有一种方法可以根据我们用户传递的参数查询指定条件的数据呢?答案是可以的,这就需要用到 MyBatis 中参数传递的知识了。
MyBatis 中使用 #{}
来获取方法中的参数,当我们调用这个方法并且传入参数的时候通过这个 #{}
就能将传递过来的参数给 SQL 语句。
@Select("select * from userinfo where id=#{id}")
public UserInfo getById1(int id);
@Test
void getById1() {
UserInfo userInfo = userInfoMapper.getById1(3);
log.info(userInfo.toString());
}
如果 mapper 接口类型的方法只有一个普通类型的参数,#{…}里面的属性名可以随便写,但是还是建议和参数名保持一致。
@Select("select * from userinfo where id=#{userid}")
public UserInfo getById1(Integer id);
当然,如果我们觉得方法参数的名字不好的话,我们也可以使用 @Param
对方法的参数进行重命名,但是如果使用 @Param
对参数进行重命名的话,#{…}里面的属性名必须和别名是相同的。
@Select("select * from userinfo where id=#{userid}")
public UserInfo getById1(@Param("userid") Integer id);
@Select("select * from userinfo where id=#{id}")
public UserInfo getById1(@Param("userid") Integer id);
数据库的主要操作无非就是增删改查,那么我们使用 MyBatis 如何实现数据库的增删改查呢?
MyBatis 增加操作需要使用到 @Insert
注解。
@Insert("insert into userinfo (username, `password`, age, gender, phone)" +
"values (#{username}, #{password}, #{age}, #{gender}, #{phone})")
public Integer insert(UserInfo userInfo);
@Test
void insert() {
UserInfo userInfo = new UserInfo();
userInfo.setUsername("小美");
userInfo.setPassword("小美");
userInfo.setAge(18);
userInfo.setGender(2);
userInfo.setPhone("3139812381");
int ret = userInfoMapper.insert(userInfo);
log.info(ret + "行被更新");
}
这里可以选择将插入的数据封装成Java对象作为参数传入,也可以将这每个些信息当成一个参数进行传递。当以Java对象的形式作为参数进行传递的话,Java对象中的属性名需要保持和数据库中的列名相同,因为当传递的参数是对象的话,MyBatis 会进入到这个对象中,查看这个对象中的属性,然后与数据库中的列名做比较,如果相同就将Java对象中属性的值代入 SQL 语句中,不相同则不带入。
Insert 语句默认返回的是受影响的行数,如果我们需要只需要将方法的返回类型声明为 int/Integer 就可以了,如果不需要,就将方法的返回类型声明为 void。
但是有些时候,我们不仅仅需要 insert 语句影响的行数,可能还需要获取到其他的信息,比如自增主键的值,那么我们如何获取到 insert 之后自增主键的值呢?如果我们直接拿到Java对象中与自增主键列同名的属性,能拿到吗?
log.info(ret + "行被更新" + userInfo.getId());
按理来说这个自增主键应该是 5 了,但是拿到的确实 0,说明这个还是初始值,那么应该进行什么操作才能拿到这个 insert 的一行的自增主键的值呢?
如果想要拿到自增主键的值,需要在 Mapper 接口的方法上添加一个 @Options
注解。
@Options(useGeneratedKeys = true, keyProperty = "id")
@Insert("insert into userinfo (username, `password`, age, gender, phone)" +
"values (#{username}, #{password}, #{age}, #{gender}, #{phone})")
public Integer insert(UserInfo userInfo);
useGeneratedKeys 参数是是否需要使用到自增主键,然后 keyProperty 参数表示将获取到的自增主键的值赋值给 Java 的哪个属性,因为这里我们传递的参数是 Java 对象,所以当获取到 insert 插入的那行的自增主键赋值给 id 之后,MyBatis 就会去这个对象的属性去找有没有 id 这个属性,如果有就赋值给它。如果传递的参数是多个普通类型参数的话,那么 MyBatis 就会在这些传递的参数中找,看是否有相同名称的参数,然后赋值给它。
因为我这里为大家演示的时候,多次插入了,所以获取到的自增主键的值是8,这影响不大,我们关键看是否 insert 插入的这行的自增主键赋值给了 userInfo 对象中的 id 属性。
如果传递的是对象,并且在接收参数的时候对这个参数进行了重命名的话,SQL 中的 #{} 参数就需要指定是哪个引用的哪个属性。
@Options(useGeneratedKeys = true, keyProperty = "id")
@Insert("insert into userinfo (username, `password`, age, gender, phone)" +
"values (#{user.username}, #{user.password}, #{user.age}, #{user.gender}, #{user.phone})")
public Integer insert(@Param("user") UserInfo userInfo);
使用 MyBatis 进行删除数据的操作需要使用到 @Delete
注解。
@Delete("delete from userinfo where id=#{id}")
public void delete(Integer id);
@Test
void delete() {
userInfoMapper.delete(1);
}
MyBatis 实现修改操作需要使用到 @Update
注解。
@Update("update userinfo set gender=#{gender} where id=#{id}")
public void update(Integer gender, Integer id);
@Test
void update() {
userInfoMapper.update(0, 8);
}
查询操作不就是 select
吗,还有什么需要讲的吗?
通过前面的查询我们可以发现,我们 Java 的 userInfo 对象中的 deleteFlag createTime updateTime
都是初识值,也就是说,这些属性并没有被赋值,那么这是为什么呢?我的 select SQL 语句不是查询出结果了吗?
还记得我们前面说了什么吗?需要保证 Java 中的属性名和数据库表中的列名保持一致,因为当 select 查询出结果之后,MyBatis 会根据方法的返回值类型,如果是普通类型的话,并且 select 查询出来的结果的类型和方法返回值的类型相同的话,那么就会直接将这个查询出来的结果进行返回;但是如果方法返回值的类型是 Java 对象的话,MyBatis 会将查询出来的结果按照列名与 Java 对象中的属性名进行匹配,如果相同就赋值给这个属性,没有相同的属性名,那么 Java 对象的这个属性就是初始值。
虽然保证数据库表中的列名和 Java 对象中的属性名保持一致很重要,但是由于 SQL 和 Java 的变量名命名规则和习惯存在差异,比如:在 SQL 中,deletflg 就是这样表示——delete_flag,而在 Java 中是以小驼峰的习惯进行命名,也就是——deleteFlag,那么这样的话,就无法保证名称一致了,那么这样该怎么办呢?有三种解决方法:
前面我们学习 SQL 的时候肯定学过对查询结果进行重命名吧,我们可以将列名与Java属性名不相同的列进行重命名。
@Select("select id, username, password, age, gender, delete_flag as deleteFlag," +
"create_time as createTime, update_time as updateTime from userinfo")
public List<UserInfo> selectAll1();
@Test
void selectAll1() {
List<UserInfo> list = userInfoMapper.selectAll1();
log.info(list.toString());
}
@Documented
@Retention(RetentionPolicy.RUNTIME)
@Target({ElementType.METHOD})
public @interface Results {
String id() default "";
Result[] value() default {};
}
@Results 注解的参数有两个,一个是 String 类型,这个参数的用法稍后给大家讲,第二个参数就是 Result[] 一个数组,数组中的每个元素都是一个 Result 类型,而 Result 中也有很多参数,但是这里我们主要用到两个:
@Results(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, delete_flag," +
"create_time, update_time from userinfo")
public List<UserInfo> selectAll2();
@Test
void selectAll2() {
List<UserInfo> list = userInfoMapper.selectAll1();
log.info(list.toString());
}
如果我们又想根据给定的 id 进行查询的话,因为表的列名和Java的属性名不同,那么是否又需要写跟上面一样的 @Results
注解吗?不是的,这就需要用到 @Results 注解的第一个参数了,通过指定 @Results 的第一个参数,可以使得这个 @Results 注解可以重复使用。
@Results(id = "BaseMap", value = {
@Result(column = "delete_flag", property = "deleteFlag"),
@Result(column = "create_time", property = "createTme"),
@Result(column = "update_time", property = "updateTime")
})
如果后面的方法也想使用和这个 @Results 注解相同的配置的话,只需要在方法上加上 @ResultMap
注解就可以了。
@ResultMap("BaseMap")
@Select("select id, username, password, age, gender, delete_flag," +
"create_time, update_time from userinfo where id=#{id}")
public UserInfo getById2(Integer id);
这个 @ResultMap 中的参数需要保证和 @Results 注解中的第一个参数 id 的值保持一致才可以使用该 @Results 的相同配置。
通常数据库列使?蛇形命名法进?命名(下划线分割各个单词),? Java 属性?般遵循驼峰命名法约定。为了在这两种命名?式之间启??动映射,需要将 mapUnderscoreToCamelCase 设置为 true。
mybatis:
configuration:
map-underscore-to-camel-case: true #配置驼峰?动转换
当在配置文件添加这个配置之后,我们再试一次看看结果:
@Select("select * from userinfo")
public List<UserInfo> selectAll3();
@Test
void selectAll3() {
List<UserInfo> list = userInfoMapper.selectAll3();
log.info(list.toString());
}
可以发现通过在配置文件中添加配置从而解决 SQL 和 Java 命名规则不同而导致的问题非常的方便,所以也建议大家使用这个做法。
不管使用什么方法书写 MyBatis,首先就是需要配置数据库相关的信息,这里 XML 配置数据库信息的方式跟注解配置数据库的方式是一样的,我这里就不过多介绍了。
# 数据库连接配置
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 XML 的文件路径。
mybatis:
mapper-locations: classpath:mapper/**Mapper.xml
这个路径在我们的项目中这样显示:
classpath 就是我们项目的 resources 文件夹,mapper 是我们自己创建的用来操作 MyBatis 的文件夹,这个文件夹中存放的都是 xml 文件,而这个配置中的 **Mapper.xml
则表示所有以 Mapper.xml 结尾的文件,如果想表示以 .xml
结尾的文件,则用 **.xml
来表示。通常我们的操作 MyBatis 的 XML 文件都是以 Mapper.xml 作为后缀。这里的 mapper
文件夹和 **Mapper.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 <mapper namespace="com.example.mybatis20231226.Mapper.UserInfoXMLMapper">
</mapper>
这里 namespace 中的值是我们的要实现的接口的全限定类名。
这里的接口就正常写,主要的操作在我们的 xml 文件中。
package com.example.mybatis20231226.Mapper;
import com.example.mybatis20231226.Model.UserInfo;
import org.apache.ibatis.annotations.Mapper;
import java.util.List;
@Mapper
public interface UserInfoXMLMapper {
public List<UserInfo> selectAll();
}
后面的实现就是写在 <mapper>
标签中的,假设我们这里还是查询,就可以这样写:
<mapper namespace="com.example.mybatis20231226.Mapper.UserInfoXMLMapper">
<select id="selectAll" resultType="com.example.mybatis20231226.Model.UserInfo">
select * from userinfo
</select>
</mapper>
id 参数指定要实现的接口中的方法名称,resultType 指定该方法的返回的数据的类型的路径,这里不是 List,而是 List 中的元素的类型 UserInfo。
@Slf4j
@SpringBootTest
class UserInfoXMLMapperTest {
@Autowired
private UserInfoXMLMapper userInfoXMLMapper;
@Test
void selectAll() {
List<UserInfo> list = userInfoXMLMapper.selectAll();
log.info(list.toString());
}
}
public Integer insert(UserInfo userInfo);
<insert id="insert">
insert into userinfo (username, password, age, gender, phone)
values(#{username}, #{password}, #{age}, #{gender}, #{phone})
</insert>
@Test
void insert() {
UserInfo userInfo = new UserInfo();
userInfo.setUsername("小帅");
userInfo.setPassword("小帅");
userInfo.setAge(28);
userInfo.setGender(1);
userInfo.setPhone("31737137128");
int ret = userInfoXMLMapper.insert(userInfo);
log.info(ret + "被更新");
}
获取插入数据的自增主键
要想获取到插入数据的自增主键,需要在 <insert>
标签中配置 userGenerateKeys
参数和 keyProperty
参数。
<insert id="insert" useGeneratedKeys="true" keyProperty="id">
insert into userinfo (username, password, age, gender, phone)
values(#{username}, #{password}, #{age}, #{gender}, #{phone})
</insert>
别名
xml 别名的操作和使用注解的操作是类似的。
public Integer insert(@Param("user") UserInfo userInfo);
<insert id="insert" useGeneratedKeys="true" keyProperty="id">
insert into userinfo (username, password, age, gender, phone)
values(#{user.username}, #{user.password}, #{user.age}, #{user.gender}, #{user.phone})
</insert>
public void delete(Integer id);
<delete id="delete">
delete from userinfo where id=#{id}
</delete>
@Test
void delete() {
userInfoXMLMapper.delete(2);
}
public void update(Integer gender, Integer id);
<update id="update">
update userinfo set gender=#{gender} where id=#{id}
</update>
@Test
void update() {
userInfoXMLMapper.update(0, 3);
}
这里也是主要为了解决 SQL 命名规则和 Java 命名规则不同导致的问题,解决方法和通过注解是一样的。
public List<UserInfo> selectAll2();
<select id="selectAll2" resultType="com.example.mybatis20231226.Model.UserInfo">
select id, username, password, age, gender, delete_flag as deleteFlag,
create_time as createTime, update_time as updateTime from userinfo
</select>
@Test
void selectAll2() {
List<UserInfo> list = userInfoXMLMapper.selectAll1();
log.info(list.toString());
}
public List<UserInfo> selectAll1();
<resultMap id="XmlBaseMap" type="com.example.mybatis20231226.Model.UserInfo">
<id column="id" property="id"></id>
<result column="delete_flag" property="deleteFlag"></result>
<result column="create_time" property="createTime"></result>
<result column="update_time" property="updateTime"></result>
</resultMap>
<select id="selectAll1" resultMap="XmlBaseMap">
select * from userinfo
</select>
id标签用于标识主键列和Java对象的对应关系,<result>
标签则用于标识非主键列和Java对象的对应关系
@Test
void selectAll1() {
List<UserInfo> list = userInfoXMLMapper.selectAll1();
log.info(list.toString());
}
这个和前面注解的做法是一样的,都是在配置文件中讲驼峰转换的配置的值设置为 true 就可以了。
mybatis:
configuration:
map-underscore-to-camel-case: true #配置驼峰?动转换