Mybatis Plus 入门

MyBatis-Plus(简称 MP)是一个 MyBatis 的增强工具,在 MyBatis 的基础上只做增强不做改变,为简化开发、提高效率而生。

在继续看这篇文章之前,请确保你会 SpringBoot 以及 Mybatis,以便有更好的观看体验。

首先,丢出 SQL 语句:

/*
 Navicat Premium Data Transfer

 Source Server         : MySQL
 Source Server Type    : MySQL
 Source Server Version : 50623
 Source Host           : localhost:3306
 Source Schema         : mp

 Target Server Type    : MySQL
 Target Server Version : 50623
 File Encoding         : 65001

 Date: 09/12/2019 14:19:03
*/

SET NAMES utf8mb4;
SET FOREIGN_KEY_CHECKS = 0;

-- ----------------------------
-- Table structure for user
-- ----------------------------
DROP TABLE IF EXISTS `user`;
CREATE TABLE `user`  (
  `id` bigint(20) NOT NULL COMMENT '主键',
  `name` varchar(30) CHARACTER SET utf8 COLLATE utf8_general_ci NULL DEFAULT NULL COMMENT '姓名',
  `age` int(11) NULL DEFAULT NULL COMMENT '年龄',
  `email` varchar(50) CHARACTER SET utf8 COLLATE utf8_general_ci NULL DEFAULT NULL COMMENT '邮箱',
  `manager_id` bigint(20) NULL DEFAULT NULL COMMENT '直属上级id',
  `create_time` datetime(0) NULL DEFAULT NULL COMMENT '创建时间',
  PRIMARY KEY (`id`) USING BTREE,
  INDEX `manager_fk`(`manager_id`) USING BTREE,
  CONSTRAINT `manager_fk` FOREIGN KEY (`manager_id`) REFERENCES `user` (`id`) ON DELETE RESTRICT ON UPDATE RESTRICT
) ENGINE = InnoDB CHARACTER SET = utf8 COLLATE = utf8_general_ci ROW_FORMAT = Compact;

-- ----------------------------
-- Records of user
-- ----------------------------
INSERT INTO `user` VALUES (1087982257332887525, '刘明强', 31, 'lmq@baomidou.com', 1088248166370832385, '2019-12-06 02:29:20');
INSERT INTO `user` VALUES (1087982257332887553, '大boss', 40, 'boss@baomidou.com', NULL, '2019-01-11 14:20:20');
INSERT INTO `user` VALUES (1088248166370832385, '王天风', 25, 'wtf@baomidou.com', 1087982257332887553, '2019-02-05 11:12:22');
INSERT INTO `user` VALUES (1088250446457389058, '李艺伟', 28, 'lyw@baomidou.com', 1088248166370832385, '2019-02-14 08:31:16');
INSERT INTO `user` VALUES (1094590409767661570, '张雨琪', 31, 'zjq@baomidou.com', 1088248166370832385, '2019-01-14 09:15:15');
INSERT INTO `user` VALUES (1094592041087729666, '刘红雨', 32, 'lhm@baomidou.com', 1088248166370832385, '2019-01-14 09:48:16');

SET FOREIGN_KEY_CHECKS = 1;

Mybatis Plus 快速入门

导入依赖

<dependency>
    <groupId>mysql</groupId>
    <artifactId>mysql-connector-java</artifactId>
</dependency>
<dependency>
    <groupId>com.baomidou</groupId>
    <artifactId>mybatis-plus-boot-starter</artifactId>
    <version>3.2.0</version>
</dependency>

创建实体类

public class User {

    private long id;
    private String name;
    private long age;
    private String email;
    private long managerId;
    private LocalDateTime createTime;

    ......
    // get 和 set 方法、toString 方法省略
}

创建持久层接口

@Mapper
public interface UserMapper extends BaseMapper<User> {
}

继承 BaseMapper 后不需要任何操作,就可以使用了!!!

测试

@SpringBootTest
class MybatisPlusApplicationTests {

    @Autowired
    private UserMapper userMapper;

    @Test
    public void selectList(){
        // 不添加任何条件查询
        List<User> users = userMapper.selectList(null);
        for (User user1: users){
            System.out.println(user1);
        }
    }
}

查询结果:

DEBUG==>  Preparing: SELECT id,create_time,name,manager_id,age,email FROM user 
DEBUG==> Parameters: 
TRACE<==    Columns: id, create_time, name, manager_id, age, email
TRACE<==        Row: 1087982257332887525, 2019-12-06 02:29:20, 刘明强, 1088248166370832385, 31, lmq@baomidou.com
TRACE<==        Row: 1087982257332887553, 2019-01-11 14:20:20, 大boss, null, 40, boss@baomidou.com
TRACE<==        Row: 1088248166370832385, 2019-02-05 11:12:22, 王天风, 1087982257332887553, 25, wtf@baomidou.com
TRACE<==        Row: 1088250446457389058, 2019-02-14 08:31:16, 李艺伟, 1088248166370832385, 28, lyw@baomidou.com
TRACE<==        Row: 1094590409767661570, 2019-01-14 09:15:15, 张雨琪, 1088248166370832385, 31, zjq@baomidou.com
TRACE<==        Row: 1094592041087729666, 2019-01-14 09:48:16, 刘红雨, 1088248166370832385, 32, lhm@baomidou.com
DEBUG<==      Total: 6
User{id=1087982257332887525, name='刘明强', age=31, email='lmq@baomidou.com', managerId=1088248166370832385, createTime=2019-12-06T02:29:20}
User{id=1087982257332887553, name='大boss', age=40, email='boss@baomidou.com', managerId=0, createTime=2019-01-11T14:20:20}
User{id=1088248166370832385, name='王天风', age=25, email='wtf@baomidou.com', managerId=1087982257332887553, createTime=2019-02-05T11:12:22}
User{id=1088250446457389058, name='李艺伟', age=28, email='lyw@baomidou.com', managerId=1088248166370832385, createTime=2019-02-14T08:31:16}
User{id=1094590409767661570, name='张雨琪', age=31, email='zjq@baomidou.com', managerId=1088248166370832385, createTime=2019-01-14T09:15:15}
User{id=1094592041087729666, name='刘红雨', age=32, email='lhm@baomidou.com', managerId=1088248166370832385, createTime=2019-01-14T09:48:16}

常用注解

  • 指定表名:@TableName
  • 指定主键:@TableId
  • 指定字段:@TableField

排除非表字段的方式

在实体类中,有的字段可能只用于暂时存储数据,并不需要将其插入数据库中,可使用以下方式达到插入时排除非表字段:

  • transient

    private transient String remark;
    
  • static

    private static String remark;
    
  • @TableField

    @TableField(exist=false)
    private String remark;
    

CRUD

新增

@Test
public void insert(){
    User user = new User();
    user.setId(1087982257332117525L);
    user.setName("张三");
    user.setAge(31);
    user.setEmail("zs@baomidou.com");
    user.setManagerId(1088248166370832385L);
    user.setCreateTime(LocalDateTime.now());
    int result = userMapper.insert(user);
    System.out.println(result);
}

运行结果:

DEBUG==>  Preparing: INSERT INTO user ( id, create_time, name, manager_id, age, email ) VALUES ( ?, ?, ?, ?, ?, ? ) 
DEBUG==> Parameters: 1087982257332117525(Long), 2019-12-09T14:38:34.424022600(LocalDateTime), 张三(String), 1088248166370832385(Long), 31(Long), zs@baomidou.com(String)
DEBUG<==    Updates: 1
1

查询

根据 id 查询

@Test
public void selectById(){
    User user = userMapper.selectById(1088248166370832385L);
    System.out.println(user);
}

查询结果:

DEBUG==>  Preparing: SELECT id,create_time,name,manager_id,age,email FROM user WHERE id=? 
DEBUG==> Parameters: 1088248166370832385(Long)
TRACE<==    Columns: id, create_time, name, manager_id, age, email
TRACE<==        Row: 1088248166370832385, 2019-02-05 11:12:22, 王天风, 1087982257332887553, 25, wtf@baomidou.com
DEBUG<==      Total: 1
User{id=1088248166370832385, name='王天风', age=25, email='wtf@baomidou.com', managerId=1087982257332887553, createTime=2019-02-05T11:12:22}

根据多个 id 查询

@Test
public void selectByIds() {
    List<Long> list = new ArrayList<>();
    list.add(1088248166370832385L);
    list.add(1087982257332887553L);
    List<User> users = userMapper.selectBatchIds(list);
    users.forEach(System.out::println);
}

查询结果:

DEBUG==>  Preparing: SELECT id,create_time,name,manager_id,age,email FROM user WHERE id IN ( ? , ? ) 
DEBUG==> Parameters: 1088248166370832385(Long), 1087982257332887553(Long)
TRACE<==    Columns: id, create_time, name, manager_id, age, email
TRACE<==        Row: 1087982257332887553, 2019-01-11 14:20:20, 大boss, null, 40, boss@baomidou.com
TRACE<==        Row: 1088248166370832385, 2019-02-05 11:12:22, 王天风, 1087982257332887553, 25, wtf@baomidou.com
DEBUG<==      Total: 2
User{id=1087982257332887553, name='大boss', age=40, email='boss@baomidou.com', managerId=0, createTime=2019-01-11T14:20:20}
User{id=1088248166370832385, name='王天风', age=25, email='wtf@baomidou.com', managerId=1087982257332887553, createTime=2019-02-05T11:12:22}

根据 map 查询

@Test
public void selectByMap() {
    Map<String, Object> map = new HashMap<>();
    // key 为表中列名
    map.put("name", "王天风");
    map.put("age", 25);
    List<User> users = userMapper.selectByMap(map);
    users.forEach(System.out::println);
}

查询结果:

DEBUG==>  Preparing: SELECT id,create_time,name,manager_id,age,email FROM user WHERE name = ? AND age = ? 
DEBUG==> Parameters: 王天风(String), 25(Integer)
TRACE<==    Columns: id, create_time, name, manager_id, age, email
TRACE<==        Row: 1088248166370832385, 2019-02-05 11:12:22, 王天风, 1087982257332887553, 25, wtf@baomidou.com
DEBUG<==      Total: 1
User{id=1088248166370832385, name='王天风', age=25, email='wtf@baomidou.com', managerId=1087982257332887553, createTime=2019-02-05T11:12:22}

条件构造器查询

QueryWrapper<User> wrapper = new QueryWrapper<>();
// 名字中包含雨并且年龄小于 40
// name like '%雨%' and age < 40
wrapper.like("name", "雨").lt("age", 40);
// 名字中包含雨年并且龄大于等于 20 且小于等于 40 并且 email 不为空
// name like '%雨%' and age between 20 and 40 and email is not null
wrapper.like("name", "雨").between("age", 20, 40).isNotNull("email");
// 名字为王姓或者年龄大于等于 25,按照年龄降序排列,年龄相同按照 id 升序排列
// name like '王%' or age >= 25 order by age desc, id asc
wrapper.likeRight("name", "王").or().ge("age", 25).orderByDesc("age").orderByAsc("id");
// 创建日期为2019年2月14日并且直属上级为名字为王姓
// date_format(create_time,'%Y-%m-%d')='2019-02-14' and manager_id in (select id from user where name like '王%')
wrapper.apply("date_format(create_time,'%Y-%m-%d') = {0}", "2019-02-14").inSql("manager_id", "select id from user where name like '王%'");
// 名字为王姓并且年龄小于40或邮箱不为空
// name like '王%' and (age < 40 or email is not null)
wrapper.likeRight("name", "王").and(wq->wq.lt("age", 40).or().isNotNull("email"));
// 名字为王姓或者年龄小于 40 并且年龄大于 20 并且邮箱不为空
// name like '王%' or (age < 40 and age > 20 and email is not null)
wrapper.likeRight("name", "王").or(wq->wq.lt("age", 40).gt("age", 20).isNotNull("email"));
// 年龄小于40或邮箱不为空并且名字为王姓
// (age < 40 or email is not null) and name like '王%'
wrapper.nested(wq->wq.lt("age", 40).or().isNotNull("email")).like("name", "王");
// 年龄为 30、31、34、35
// age in (30, 31, 34, 35)
wrapper.in("age", Arrays.asList(30, 31, 34, 35));
// 只返回满足条件的其中一条语句即可
// limit 1
wrapper.in("age", Arrays.asList(30, 31, 34, 35)).last("limit 1");
List<User> users = userMapper.selectList(wrapper);
users.forEach(System.out::println);

select 中字段不全出现的处理方法

// 第一种情况:select id,name
//             from user
//             where name like '%雨%' and age < 40
wrapper.select("id", "name").like("name", "雨").lt("age", 40);
// 第二种情况:select id,name,age,email
//             from user
//             where name like '%雨%' and age < 40
wrapper.like("name", "雨").lt("age", 40).select(
        User.class,
        info -> !info.getColumn().equals("create_time") && !info.getColumn().equals("manager_id")
);
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,014评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,796评论 3 386
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,484评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,830评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,946评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,114评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,182评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,927评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,369评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,678评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,832评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,533评论 4 335
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,166评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,885评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,128评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,659评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,738评论 2 351

推荐阅读更多精彩内容