MyBatis 框架学习(I)
文章目录
- MyBatis 框架学习(I)
- 1. 介绍
- 2. 准备&测试
- 3. MyBatis 注解基础操作
- 3.1 日志输出
- 3.2 Insert 操作
- 3.3 Delete 操作
- 3.4 Update 操作
- 3.5 Select 操作
- 总结
1. 介绍
之前我们学习过利用JDBC操作数据库进行项目开发,但我们发现它操作起来会比较复杂,而MyBatis就是一款简化JDBC操作的优秀持久层框架
持久层:可以理解为进行持久化操作的层,通常指数据访问层(dao),是用来操作数据库的
简单来说MyBatis就是更简单完成程序和数据库交互的框架,也是一个更好操作数据库的工具!
2. 准备&测试
在我们开始MyBatis
具体操作之前,我们需要完成环境的配置:
- 创建SpringBoot工程、引入相关依赖;
- 数据库准备、实体类准备;
- 配置MyBatis连接信息;
- 创建Mapper接口
先创建SpringBoot项目:
在创建项目时引入相关驱动依赖,pom.xml
就会自动引入这些相关依赖:
项目创建好后,我们来创建测试用的数据库:
-- 创建数据库
DROP DATABASE IF EXISTS mybatis_test;
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' );
同时创建实体类,即该实体类中的属性要与数据库中的字段对应:
import lombok.Data;
import java.util.Date;@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;}
注:数据库中的字段有时候是多个单词拼接,数据库中使用_
分割,在Java中我们使用小驼峰的方式来进行命名
之后配置MyBatis连接信息, MyBaits连接数据库需要配置以下信息:
- MySQL驱动类
- 数据库连接的URL
- 用户名
- 密码
如果是配置文件是application.properties
,代码如下:
#驱动类名称
spring.datasource.driver-class-name=com.mysql.cj.jdbc.Driver
#数据库连接的url
spring.datasource.url=jdbc:mysql://localhost:3306/mybatis_test // 刚刚创建的表
#连接数据库的用户名
spring.datasource.username=root // 填写自己的用户名
#连接数据库的密码
spring.datasource.password=123456 // 填写自己的密码
如果配置文件是application.yml
,代码如下:
# 数据库连接配置 每个空格都不能省略,yml严格要求格式正确
spring:datasource:url: jdbc:mysql://127.0.0.1:3306/mybatis_testusername: rootpassword: 123456driver-class-name: com.mysql.cj.jdbc.Driver
最后创建接口,并通过测试类进行测试(测试类Spring在创建时已经给我们配置好,在test目录下,我们只需要根据需求编写相关代码即可,或者右击mapper接口->Generate->Test, 勾选相应mapper方法,然后编写需求代码即可):
package org.example.blog_mybatis.mapper;import org.apache.ibatis.annotations.Mapper;
import org.apache.ibatis.annotations.Select;
import org.example.blog_mybatis.pojo.UserInfo;import java.util.List;@Mapper
public interface UserInfoMapper {@Select("select id, username, password, age, gender, phone, delete_flag, create_time, update_time from userinfo")public List<UserInfo> queryAllUser();}
这样我们就能通过Mybatis
来获取数据库中的信息了!
小技巧:当我们在进行sql语句编写时,有时候字段多起来后我们自己手写可能会出现问题,对此我们可以将数据库引入IDEA,这样当我们编写sql语句时它就会自动为我们提示字段信息:
按以下步骤配置:
这样就配置好了,同时它也能自动为我们获取字段信息了:
3. MyBatis 注解基础操作
对于MyBatis操作数据库的方式有两种,一种的通过注解的方式进行调用,一种是通过xml文件进行调用,对于使用那个没有固定的标准,根据情况选择使用(比较简单的数据库操作可以使用注解,较复杂的操作可以使用xml文件进行动态SQL操作)。xml
后面会再开一篇文章进行讲解,本文使用注解的方式进行操作!
3.1 日志输出
上面我们通过println函数打印输出信息来观察数据库操作状态,这样的方式观察起来效果不是很好,对此,我们可以使用MyBatis提供的日志输出来进行观察:
在application.properties
配置如下:
#配置mybatis的日志, 指定输出到控制台
mybatis.configuration.log-impl=org.apache.ibatis.logging.stdout.StdOutImpl
3.2 Insert 操作
当我们要给数据库中的userinfot表添加一条数据时,它的SQL语句是:
insert into mybatis_test(username, password, age, gender, phone) values ('zhangsan', '111', 18, 1, '123');
将SQL中的常量替换为动态的参数,在Mapper接口中它的注解为:
@Insert("insert into userinfo(username, password, age, gender, phone) values (#{username}, #{password}, #{age}, #{gender}, #{phone})")
void InsertUser(UserInfo userInfo);
在注解中,为了使我们的sql语句的操作条件不为固定值,我们通过方法形参获取需求参数,在注解中我们接收参数的方式是 #{}
,在括号里传入对应的方法参数
注:
- 括号中填入的参数要与方法参数的名称相同;
- 如果传入的方法参数是对象的话,填入括号的参数名称与对象的属性相同
编写测试类代码,这里使用自己构建的测试类:
package org.example.blog_mybatis.mapper;import org.example.blog_mybatis.pojo.UserInfo;
import org.junit.jupiter.api.Test;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.test.context.SpringBootTest;@SpringBootTest
class UserInfoMapperTest {@Autowiredprivate UserInfoMapper userInfoMapper;@Testvoid insertUser() {UserInfo userInfo = new UserInfo();userInfo.setUsername("zhangsan");userInfo.setPassword("111");userInfo.setAge(18);userInfo.setGender(1);userInfo.setPhone("123");userInfoMapper.InsertUser(userInfo);}
}
同时我们也可以通过@Param
注解给方法参数设置别名,则注解在接收参数时#{}中的内容需要与@Param中的参数一样,如果参数是对象则需要为参数.属性
@Insert("insert into userinfo(username, password, age, gender, phone) values (#{userInfo.username}, #{userInfo.password}, #{userInfo.age}, #{userInfo.gender}, #{userInfo.phone})")
void InsertUser(@Param("userInfo") UserInfo userInfo);
拓展:
Insert语句默认返回的是受影响的行数,但在有些情况下我们需要获取新插入数据的ID,这时如果我们想要拿到这个自增ID,需要在Mapper接口的方法上添加一个@Options
注解:
@Options(useGeneratedKeys = true, keyProperty = "id")
@Insert("insert into userinfo(username, password, age, gender, phone) values (#{username}, #{password}, #{age}, #{gender}, #{phone})")
void InsertUser(UserInfo userInfo);
useGeneratedKeys
:这会令MyBatis使用JDBC的getGeneratedKeys方法来取出数据库内部生成的主键,默认为false;keyProperty
:指定能够唯⼀识别对象的属性,MyBatis 使用 getGeneratedKeys 的返回值或 insert 语句的 selectKey子元素设置它的值,默认值:未设置(unset)
编写测试代码:
@Test
void insertUser() {UserInfo userInfo = new UserInfo();userInfo.setUsername("XiaoMa");userInfo.setPassword("111");userInfo.setAge(18);userInfo.setGender(1);userInfo.setPhone("10086");Integer count = userInfoMapper.InsertUser(userInfo);System.out.println("添加数据条数:" + count + " 数据ID:" + userInfo.getId());
}
3.3 Delete 操作
当我们要删除数据库中的userinfo表中的一条数据时,它的SQL语句是:
delete from maybatis_test where id = 3;
将SQL中的常量替换为动态的参数,在Mapper接口中它的注解为:
@Delete("delete from userinfo where id = #{id}")
void DeleteUser(Integer id);
编写测试类代码:
@Test
void deleteUser() {userInfoMapper.DeleteUser(3);
}
3.4 Update 操作
当我们要修改数据库中的userinfo表中的数据时,它的SQL语句是:
update userinfo set username = "zhaoliu", age = "36" where id = 5;
将SQL中的常量替换为动态的参数,在Mapper接口中它的注解为:
@Update("update userinfo set username = #{username}, age = #{age} where id = #{id}")
Integer UpdateUser(UserInfo userInfo);
编写测试类代码:
@Test
void updateUser() {UserInfo userInfo = new UserInfo();userInfo.setId(5);userInfo.setUsername("zhaoliu");userInfo.setAge(36);userInfoMapper.UpdateUser(userInfo);
}
3.5 Select 操作
当我们要查询数据库中的userinfo表中的数据时,它的SQL语句是:
select id, username, password, age, gender, phone, delete_flag, create_time, update_time from userinfo where id = 5;
在Mapper接口中它的注解为:
@Select("select id, username, password, age, gender, phone, delete_flag, create_time, update_time from userinfo where id = #{id}")
UserInfo queryUser(Integer id);
编写测试类代码:
@Test
void queryUser() {UserInfo userInfo = userInfoMapper.queryUser(5);System.out.println(userInfo);
}
MyBatis会根据方法返回的结果进行赋值:
若方法用对象UserInfo接收返回结果,MySQL查询出来的数据为一条就会自动赋值给对象;
若方法用集合List接收返回结果,MySQL查询出来的数据若为多条就会自动赋值集合List;
但若MySQL返回的结果为多条数据,却用UserInfo去接收多条数据,则MyBatis就会报错
通过上述查询语句,数据能够被显示出来,但却存在着一些问题:后面三个数的值为null?
原因就在于在返回结果时,MyBaits会自动获取结果中返回的列名并在Java类中查找相同名字的属性,若名称相同则会将结果数据赋值给该属性,但是我们的UserInfo中的属性名称为deleteFlag
,而数据库中对应的字段名称为delete_flag
,两者从名称上看并不相同,则UserInfo中的deleteFlag
属性无法被赋到值:
那怎么解决上面这个问题呢?这里我们开启驼峰命名解决(还有其它方法如起别名,结果映射等,这里使用驼峰命名,只要配置好就能直接使用,方便高效):
通常数据库中使用蛇形命名法进行命名(下划线分割各个单词),而Java属性一般遵循驼峰命名法约定。为了在这两种命名方式之间启用自动映射,需要将
map-underscore-to-camel-case
设置为true
在配置文件application.properties
中加入:
#开启mybatis的驼峰命名自动映射开关 a_column ------> aCloumn
mybatis.configuration.map-underscore-to-camel-case=true
这样缺失的数据也能查询到了!
总结
以上就是本文对MyBatis框架学习注解方式的介绍与使用了,接下来也会更新MyBatis框架学习XML的方式的文章,敬请期待!!