Spring Boot Mybatis 使用教程
Mybatis 在当下互联网开发环境,十分重要。本章主要讲述 Mybatis 如何使用。
从本系列开始,都需要用到 mysql 数据库 和其他一些参考的数据库。请准备相关环节。本章需要以下环境支撑:
- mysql 5.6+
- jdk1.8+
- spring boot 2.1.6
- idea 2018.1
1 准备数据库
数据库教程系列都是使用相同的数据,如在 Spring Boot JDBC 使用教程使用的一样
| 字段 | 类型 | 主键 | 说明 |
|---|---|---|---|
| id | int | 是 | 自动编号 |
| user_name | varchar(100) | 否 | 用户名 |
| password | varchar(255) | 否 | 密码 |
| last_login_time | date | 否 | 最近登录时间 |
| sex | tinyint | 否 | 性别 0男 1女 2其他 |
SET NAMES utf8mb4;
SET FOREIGN_KEY_CHECKS = 0;
-- ----------------------------
-- Table structure for t_user
-- ----------------------------
DROP TABLE IF EXISTS `t_user`;
CREATE TABLE `t_user` (
`id` int(11) NOT NULL AUTO_INCREMENT,
`user_name` varchar(255) DEFAULT NULL,
`password` varchar(255) DEFAULT NULL,
`last_login_time` datetime DEFAULT NULL,
`sex` tinyint(4) DEFAULT NULL,
PRIMARY KEY (`id`)
) ENGINE=InnoDB AUTO_INCREMENT=6 DEFAULT CHARSET=armscii8;
-- ----------------------------
-- Records of t_user
-- ----------------------------
BEGIN;
INSERT INTO `t_user` VALUES (1, 'json', '123', '2019-07-27 16:01:21', 1);
INSERT INTO `t_user` VALUES (2, 'jack jo', '123', '2019-07-24 16:01:37', 1);
INSERT INTO `t_user` VALUES (3, 'manistal', '123', '2019-07-24 16:01:37', 1);
INSERT INTO `t_user` VALUES (4, 'landengdeng', '123', '2019-07-24 16:01:37', 1);
INSERT INTO `t_user` VALUES (5, 'max', '123', '2019-07-24 16:01:37', 1);
COMMIT;
SET FOREIGN_KEY_CHECKS = 1;
2 新建 Spring Boot 工程项目
- File > New > Project,如下图选择
Spring Initializr然后点击 【Next】下一步 - 填写
GroupId(包名)、Artifact(项目名) 即可。点击 下一步
groupId=com.fishpro
artifactId=mybatis - 选择依赖
Spring Web Starter前面打钩,勾选SQL选项的 mybatis , mysql。 - 项目名设置为
spring-boot-study-mybatis.
3 依赖引入 Pom.xml 配置
在新建工程的时候,如果选择 Mybatis 依赖,则不需要单独配置,以下是本项目的全部依赖。
<dependencies>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-web</artifactId>
</dependency>
<dependency>
<groupId>org.mybatis.spring.boot</groupId>
<artifactId>mybatis-spring-boot-starter</artifactId>
<version>2.1.0</version>
</dependency>
<dependency>
<groupId>mysql</groupId>
<artifactId>mysql-connector-java</artifactId>
<scope>runtime</scope>
</dependency>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-test</artifactId>
<scope>test</scope>
</dependency>
</dependencies>
3 Mybatis 的工程配置 application.yml
无论是 jdbc 还是 mybatis 都需要对 数据源进行配置,而且配置项是一样的。除了数据源基础的配置,我们还需要配置 mybatis 的映射文件 *.xml 的路径,mybatis 的独立 config.xml 文件路径等。
mapper-locations配置实体与数据库表的映射文件xml位置,我们配置为/src/main/resource/mybatis/**maper.xml下map-underscore-to-camel-case开启mybatis的驼峰命名转换type-aliases-package指定 实体domain的类,这是com.fishpro下所有以domain结尾的实体类。
server:
port: 8086
mybatis:
configuration:
map-underscore-to-camel-case: true
mapper-locations: mybatis/**/*Mapper.xml
type-aliases-package: com.fishpro.**.domain
针对 mybatis 的配置,我们也可以把所有 mybatis 配置配置到单独的 config 文件中。
5 编写示例代码
这里我们首次使用到 三层结构来演示本项目,即 controller-service-dao-mapper-mybatis

本示例包括新增的页面
- src/main/java/com/fishpro/mybatis/controller/UserController.java 控制层 rest api
- src/main/java/com/fishpro/mybatis/domain/UserDO.java 实体对象
- src/main/java/com/fishpro/mybatis/dao/UserDao.java Dao数据库访问层
- src/main/java/com/fishpro/mybatis/service/UserService.java 服务层包括了接口+接口实现
- src/main/java/com/fishpro/mybatis/service/impl/UserServiceImpl.java 服务层包括了接口+接口实现
- src/main/resources/mybatis/UserMapper.xml 映射 xml 文件
5.1 映射文件 UserMapper.xml
UserMapper.xml 是一个以 mybatis 语法来编写的 xml 映射文件。
+ mapper 节点是根节点 具有属性 namespace 表示 mapper 对应的 java 的 Dao 接口类
|--select 查询节点
|--where
|--choose
|--when
|--otherwise
|--insert 插入节点
|--update 更新节点
|--delete 删除节点
详细的代码如下
<?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.fishpro.mybatis.dao.UserDao">
<select id="get" resultType="com.fishpro.mybatis.domain.UserDO">
select `id`,`user_name`,`password`,`last_login_time`,`sex` from t_user where id = #{value}
</select>
<select id="list" resultType="com.fishpro.mybatis.domain.UserDO">
select `id`,`user_name`,`password`,`last_login_time`,`sex` from t_user
<where>
<if test="id != null and id != '-1' " > and id = #{id} </if>
<if test="userName != null and userName != '' " > and user_name = #{userName} </if>
<if test="password != null and password != '' " > and password = #{password} </if>
<if test="lastLoginTime != null and lastLoginTime != '' " > and last_login_time = #{lastLoginTime} </if>
<if test="sex != null and sex != '-1' " > and sex = #{sex} </if>
</where>
<choose>
<when test="sort != null and sort.trim() != ''">
order by ${sort} ${order}
</when>
<otherwise>
order by id desc
</otherwise>
</choose>
<if test="offset != null and limit != null">
limit #{offset}, #{limit}
</if>
</select>
<select id="count" resultType="int">
select count(*) from t_user
<where>
<if test="id != null and id != '-1' " > and id = #{id} </if>
<if test="userName != null and userName != '' " > and user_name = #{userName} </if>
<if test="password != null and password != '' " > and password = #{password} </if>
<if test="lastLoginTime != null and lastLoginTime != '' " > and last_login_time = #{lastLoginTime} </if>
<if test="sex != null and sex != '-1' " > and sex = #{sex} </if>
</where>
</select>
<insert id="save" parameterType="com.fishpro.mybatis.domain.UserDO" useGeneratedKeys="true" keyProperty="id">
insert into t_user
(
`user_name`,
`password`,
`last_login_time`,
`sex`
)
values
(
#{userName},
#{password},
#{lastLoginTime},
#{sex}
)
</insert>
<update id="update" parameterType="com.fishpro.mybatis.domain.UserDO">
update t_user
<set>
<if test="userName != null">`user_name` = #{userName}, </if>
<if test="password != null">`password` = #{password}, </if>
<if test="lastLoginTime != null">`last_login_time` = #{lastLoginTime}, </if>
<if test="sex != null">`sex` = #{sex}</if>
</set>
where id = #{id}
</update>
<delete id="remove">
delete from t_user where id = #{value}
</delete>
<delete id="batchRemove">
delete from t_user where id in
<foreach item="id" collection="array" open="(" separator="," close=")">
#{id}
</foreach>
</delete>
</mapper>
5.2 实体对象 UserDO
与数据库表对应的(可自定义)对象实体类
public class UserDO {
private Integer id;
private String userName;
private String password;
private Integer sex;
private Date lastLoginTime;
public Integer getId() {
return id;
}
public void setId(Integer id) {
this.id = id;
}
public String getUserName() {
return userName;
}
public void setUserName(String userName) {
this.userName = userName;
}
public String getPassword() {
return password;
}
public void setPassword(String password) {
this.password = password;
}
public Integer getSex() {
return sex;
}
public void setSex(Integer sex) {
this.sex = sex;
}
public Date getLastLoginTime() {
return lastLoginTime;
}
public void setLastLoginTime(Date lastLoginTime) {
this.lastLoginTime = lastLoginTime;
}
}
5.3 数据库访问层 UserDao
Dao 是一个接口层,接口方法对应 mybatis/**Mapper.xml 中 Mapper 节点下的 id 值。例如上面的 <select id="get" resultType="com.fishpro.mybatis.domain.UserDO"> 那么对应的接口方法 UserDO get(Integer id)
Dao 必须使用 @Mappger注解,才能实现从Dao到Mapper.xml的映射
/**
* 数据库访问层 用户Dao t_user
* @author fishpro
*/
@Mapper
public interface UserDao {
/**
* 对应节点 select id="get" resultType="com.fishpro.mybatis.domain.UserDO"
* */
UserDO get(Integer id);
/**
* 对应节点 select id="list" resultType="com.fishpro.mybatis.domain.UserDO"
* */
List<UserDO> list(Map<String, Object> map);
/**
* 对应节点 select id="count" resultType="int"
* */
int count(Map<String, Object> map);
/**
* 对应节点 insert id="save" parameterType="com.fishpro.mybatis.domain.UserDO" useGeneratedKeys="true" keyProperty="id"
* */
int save(UserDO user);
/**
* 对应节点 update id="update" parameterType="com.fishpro.mybatis.domain.UserDO"
* */
int update(UserDO user);
/**
* 对应节点 delete id="remove"
* */
int remove(Integer id);
/**
* 对应节点 delete id="batchRemove"
* */
int batchRemove(Integer[] ids);
}
5.4 服务类 UseService
注意 UserServiceImpl 一定要使用 @Service 注解,才能实现 UserService类自动注入功能
在本示例中只是简单的增删改查功能
UserService 接口类
public interface UserService {
UserDO get(Integer id);
List<UserDO> list(Map<String, Object> map);
int count(Map<String, Object> map);
int save(UserDO user);
int update(UserDO user);
int remove(Integer id);
int batchRemove(Integer[] ids);
}
UserServiceImpl 实现类
@Service
public class UserServiceImpl implements UserService {
@Autowired
private UserDao userDao;
@Override
public UserDO get(Integer id){
return userDao.get(id);
}
@Override
public List<UserDO> list(Map<String, Object> map){
return userDao.list(map);
}
@Override
public int count(Map<String, Object> map){
return userDao.count(map);
}
@Override
public int save(UserDO user){
return userDao.save(user);
}
@Override
public int update(UserDO user){
return userDao.update(user);
}
@Override
public int remove(Integer id){
return userDao.remove(id);
}
@Override
public int batchRemove(Integer[] ids){
return userDao.batchRemove(ids);
}
}
5.5 控制层实现 Rest Api 增删改查接口 UserController
本示例代码使用标准的 Restful 风格编写 具体代码如下:
/**
* RESTful API + Mybatis 风格示例 对资源 user 进行操作
* 本示例没有使用数据库,也没有使用 service 类来辅助完成,所有操作在本类中完成
* 请注意几天
* 1.RESTful 风格使用 HttpStatus 状态返回 GET PUT PATCH DELETE 通常返回 201 Create ,DELETE 还有时候返回 204 No Content
* 2.使用 RESTful 一定是要求具有幂等性,GET PUT PATCH DELETE 本身具有幂等性,但 POST 不具备,无论规则如何定义幂等性,需要根据业务来设计幂等性
* 3.RESTful 不是神丹妙药,实际应根据实际情况来设计接口
* */
@RestController
@RequestMapping("/api/user")
public class UserController {
@Autowired
private UserService userService;
/**
* 测试用 参数为 name
* */
@RequestMapping("/hello")
public String hello(String name){
return "Hello "+name;
}
/**
* SELECT 查询操作,返回一个JSON数组
* 具有幂等性
* */
@GetMapping("/users")
@ResponseStatus(HttpStatus.OK)
public Object getUsers(){
return userService.list(new HashMap<>());
}
/**
* SELECT 查询操作,返回一个新建的JSON对象
* 具有幂等性
* */
@GetMapping("/users/{id}")
@ResponseStatus(HttpStatus.OK)
public Object getUser(@PathVariable("id") String id){
if(null==id){
return null;
}
return userService.get(Integer.valueOf(id));
}
/**
* 新增一个用户对象
* 非幂等
* 返回 201 HttpStatus.CREATED 对创建新资源的 POST 操作进行响应。应该带着指向新资源地址的 Location 头
* */
@PostMapping("/users")
@ResponseStatus(HttpStatus.CREATED)
public Object addUser(@RequestBody UserDO user){
if(user.getId()==0){
return null;
}
userService.save(user);
return user;
}
/**
* 编辑一个用户对象
* 幂等性
* */
@PutMapping("/users/{id}")
@ResponseStatus(HttpStatus.CREATED)
public Object editUser(@PathVariable("id") String id, @RequestBody UserDO user){
userService.update(user);
return user;
}
/**
* 删除一个用户对象
* 幂等性
* 返回 HttpStatus.NO_CONTENT 表示无返回内容
* */
@DeleteMapping("/users/{id}")
@ResponseStatus(HttpStatus.NO_CONTENT)
public void deleteUser(@PathVariable("id") String id){
userService.remove(Integer.valueOf(id));
}
}
6 运行示例
右键 MybatisApplication > Run MybatisApplication 在浏览器输入 http://localhost:8086/api/user/users
使用 PostMan 测试工具可测试 GET\POST\PUT\DELETE 动作的 Restful api

7 问题思考
- 如何自动生成代码
- 如何动态配置数据源
- 如何分表分库
- xml的详细语法由哪些坑
- xml 好还是 注解好
- 如何解决事物处理
- 扩库事物如何处理
下一章节我们演示下使用 注解来实现
Spring Boot Mybatis 使用教程的更多相关文章
- spring boot mybatis 整合教程
本项目使用的环境: 开发工具:Intellij IDEA 2017.1.3 springboot: 1.5.6 jdk:1.8.0_161 maven:3.3.9 额外功能 PageHelper 分页 ...
- Spring Boot JPA 使用教程
JPA 是 Spring Boot 官方推荐的数据库访问组件,其充分体现了面向对象编程思想,有点像 asp.net 的 EFCore.JPA 也是众多 ORM 的抽象. 从本系列开始,都需要用到 my ...
- Spring Boot入门教程2-1、使用Spring Boot+MyBatis访问数据库(CURD)注解版
一.前言 什么是MyBatis?MyBatis是目前Java平台最为流行的ORM框架https://baike.baidu.com/item/MyBatis/2824918 本篇开发环境1.操作系统: ...
- Spring Boot + Mybatis 实现动态数据源
动态数据源 在很多具体应用场景的时候,我们需要用到动态数据源的情况,比如多租户的场景,系统登录时需要根据用户信息切换到用户对应的数据库.又比如业务A要访问A数据库,业务B要访问B数据库等,都可以使用动 ...
- spring boot + mybatis + druid
因为在用到spring boot + mybatis的项目时候,经常发生访问接口卡,服务器项目用了几天就很卡的甚至不能访问的情况,而我们的项目和数据库都是好了,考虑到可能时数据库连接的问题,所以我打算 ...
- spring boot + mybatis + druid配置实践
最近开始搭建spring boot工程,将自身实践分享出来,本文将讲述spring boot + mybatis + druid的配置方案. pom.xml需要引入mybatis 启动依赖: < ...
- spring boot+mybatis+quartz项目的搭建完整版
1. 利用spring boot提供的工具(http://start.spring.io/)自动生成一个标准的spring boot项目架构 2. 因为这里我们是搭建spring boot+mybat ...
- 快速搭建一个Spring Boot + MyBatis的开发框架
前言:Spring Boot的自动化配置确实非常强大,为了方便大家把项目迁移到Spring Boot,特意总结了一下如何快速搭建一个Spring Boot + MyBatis的简易文档,下面是简单的步 ...
- spring boot mybatis 打成可执行jar包后启动UnsatisfiedDependencyException异常
我的spring boot + mybatis项目在idea里面执行正常,但发布测试环境打成可执行jar包后就启动失败,提示错误如下: [ ERROR] [2018-08-30 17:23:48] o ...
随机推荐
- APP 安全测试
http://www.cnblogs.com/wetest/p/6694529.html
- 【Python】 数字求和
# 用户输入数字 num1 = input('输入第一个数字:') num2 = input('输入第二个数字:') # 求和 sum = float(num1) + float(num2) # 显示 ...
- IntelliJ IDEA 2017.3尚硅谷-----设置项目文件编码
这也可以 R暂时显示 C转换
- mysql修改密码的4种方式
转:https://www.cnblogs.com/jdxn/p/6847089.html 方法1: 用SET PASSWORD命令 首先登录MySQL. 格式:mysql> set passw ...
- opencv:全局阈值
图像的二值化分割,最重要的就是计算阈值 阈值的计算方法很多,基本分为两类,全局阈值与自适应阈值 OTSU.Triangle #include <opencv2/opencv.hpp> #i ...
- 题解【洛谷P5019】[NOIP2018]铺设道路
题目描述 春春是一名道路工程师,负责铺设一条长度为 \(n\) 的道路. 铺设道路的主要工作是填平下陷的地表.整段道路可以看作是 \(n\) 块首尾相连的区域,一开始,第 \(i\) 块区域下陷的深度 ...
- 在Linux系统下安装nginx教程
最近学习了nginx,就打算nginx安装在Linux系统下,于是我就把安装步骤记录下来了,分享给大家,希望能对大家有帮助! 我的博客地址:https://www.cnblogs.com/themys ...
- zol验证码抓包
http://www.zol.com/detail/lcd/samsung/25254662.html?zp_from=pro_price_pricenode 网站’ http://city.zol. ...
- 一篇文章带你了解JavaScript中的变量,作用域和内存问题
1 在JavaScript中的变量分别区分为两种: 一种为基本类型值,一种为应用类型值. 基本类型值指的是简单的数据段 引用类型值为可能由多个值组成的对象 引用类型的值是保存在内存中的对象,JavaS ...
- opencv:Mat对象
Mat对象:图像文件的内存数据对象 读取为 Mat 对象 读取图像位Mat对象,获取图像的相关信息 #include <opencv2/opencv.hpp> #include <i ...