@RunWith(SpringRunner.class)
@SpringBootTest
public class MybatisPlusApplicationTests {
@Autowired
private UserMapper userMapper;
@Test
public void insertTest() {
User user = new User();
user.setAge(18);
user.setEmail("[email protected]");
user.setName("Helen");
int result = userMapper.insert(user);
System.out.println("添加信息为 " + user);
System.out.println("影响行数为 " + result);
}
}
上一步操作中,new出来的对象并没有设置主键,但是插入进去之后是18位的id
原因是:MyBatis-Plus默认的主键策略是:ID_WORKER
全局唯一ID
因为创建表的时候没有设置主键自增,所以采用MyBatis-Plus的方式设置自增
@TableId(type = IdType.AUTO)
private Long id;
#全局设置主键生成策略
mybatis-plus.global-config.db-config.id-type=auto
注意:update时生成的sql自动是动态sql:UPDATE user SET age=? WHERE id=?
@Test
public void updateUserById() {
User user = new User();
user.setId(1L);
user.setName("小毛");
int result = userMapper.updateById(user);
System.out.println("影响行数为 " + result);
}
项目中经常会遇到一些数据,每次都使用相同的方式填充,例如记录的创建时间,更新时间等。
我们可以使用MyBatis Plus的自动填充功能,完成这些字段的赋值工作:
在User表中添加datetime类型的新的字段 create_time、update_time
@TableField(fill = FieldFill.INSERT) //执行添加的时候填充数据
private Date createTime;
@TableField(fill = FieldFill.INSERT_UPDATE) //执行更新操作的时候填充数据
private Date updateTime;
注意:不要忘记添加 @Component 注解
package com.zhanc.mybatistest.handler;
import com.baomidou.mybatisplus.core.handlers.MetaObjectHandler;
import org.apache.ibatis.reflection.MetaObject;
import org.springframework.stereotype.Component;
import java.util.Date;
@Component
public class MyMetaObjectHandler implements MetaObjectHandler {
@Override
public void insertFill(MetaObject metaObject) {
//第一个参数 填充的类的字段(不是sql表)
//第二个参数 当前系统时间
//第三个参数 传进来的参数
this.setFieldValByName("createTime", new Date(), metaObject);
this.setFieldValByName("updateTime", new Date(), metaObject);
}
@Override
public void updateFill(MetaObject metaObject) {
this.setFieldValByName("updateTime", new Date(), metaObject);
}
}
主要适用场景: 当要更新一条记录的时候,希望这条记录没有被别人更新,也就是说实现线程安全的数据更新
乐观锁实现方式:
ALTER TABLE `user` ADD COLUMN `version` INT;
并添加 @Version
注解
注意此处依然使用了填充注解@TableField
,请思考,为什么更新的时候不自动填充
@Version //表明是 乐观锁 的字段
@TableField(fill = FieldFill.INSERT)
private Integer version;
@Override
public void insertFill(MetaObject metaObject) {
//第一个参数 填充的类的字段(不是sql表)
//第二个参数 当前系统时间
//第三个参数 传进来的参数
......
this.setFieldValByName("version", 1, metaObject);//添加时 自动填充版本号
}
特别说明:
创建配置类
package com.zhanc.mybatistest.config;
import com.baomidou.mybatisplus.core.injector.ISqlInjector;
import com.baomidou.mybatisplus.extension.injector.LogicSqlInjector;
import com.baomidou.mybatisplus.extension.plugins.OptimisticLockerInterceptor;
import com.baomidou.mybatisplus.extension.plugins.PaginationInterceptor;
import org.mybatis.spring.annotation.MapperScan;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.stereotype.Component;
import org.springframework.transaction.annotation.EnableTransactionManagement;
@Configuration //表明是一个配置类
@EnableTransactionManagement //启用相关的注解
@MapperScan("com.zhanc.mybatistest.mapper") //这里写入后主启动类不用再写
public class MybatisPlusConfig {
/**
* 乐观锁插件
*/
@Bean
public OptimisticLockerInterceptor optimisticLockerInterceptor() {
return new OptimisticLockerInterceptor();
}
}
注意: 使用乐观锁必须先查询
@Test
public void updateUserById2Version() {
//查询数据
//此处id是数据库表里之前插入的数据
User user = userMapper.selectById(1285855724705980418L);
System.out.println(user);
//修改数据
user.setName("阿创好帅");
//执行更新
int result = userMapper.updateById(user);
System.out.println("影响行数为 " + result);
}
@Test
public void updateUserById2Version() {
//查询数据
//此处id是数据库表里之前插入的数据
User user = userMapper.selectById(1285855724705980418L);
System.out.println(user);
//修改数据
user.setName("阿创好帅");
//模拟取出数据后,数据库中version实际数据比取出的值大,即已被其它线程修改并更新了version
user.setVersion(user.getVersion() - 1);
//执行更新
int result = userMapper.updateById(user);
System.out.println("影响行数为 " + result);
}
@Test
public void updateUserById2Version() {
User user = userMapper.selectById(1285855724705980418L);
System.out.println(user);
}
完成了动态sql的foreach的功能
@Test
public void selectByIds() {
List<User> users = userMapper.selectBatchIds(Arrays.asList(1, 2, 3));
users.stream().forEach(System.out::println);
}
通过map封装查询条件
注意: map中的key对应的是数据库中的列名。例如数据库user_id,实体类是userId,这时map的key需要填写user_id
@Test
public void selectByMap() {//map查询是精确查询
HashMap<String, Object> map = new HashMap<>();
map.put("name", "小毛");//此处是在拼sql,以数据库表的字段为准
map.put("age", 22);
List<User> list = userMapper.selectByMap(map);
list.stream().forEach(System.out::println);
}
MyBatis Plus自带分页插件,只要简单的配置即可实现分页功能
此时可以删除主类中的 @MapperScan
扫描注解
/**
* 分页插件
*/
@Bean
public PaginationInterceptor paginationInterceptor() {
return new PaginationInterceptor();
}
测试: 最终通过page对象获取相关数据
@Test
public void testSelectPage() {
Page<User> page = new Page<>(1,5);//第几页,每页大小
userMapper.selectPage(page, null);
page.getRecords().forEach(System.out::println);
System.out.println(page.getCurrent());//打印当前页
System.out.println(page.getPages());//打印所有的页
System.out.println(page.getSize());//打印每一页的大小
System.out.println(page.getTotal());//打印数据总量
System.out.println(page.hasNext());//当前页是否有下一页
System.out.println(page.hasPrevious());//当前页是否有前一页
}
注意:此时控制台sql语句打印:SELECT id,name,age,email,create_time,update_time FROM user LIMIT 0,5
@Test
public void testSelectMapsPage() {
Page<User> page = new Page<>(1, 5);
IPage<Map<String, Object>> mapIPage = userMapper.selectMapsPage(page, null);
//注意:此行必须使用 mapIPage 获取记录列表,否则会有数据类型转换错误
mapIPage.getRecords().forEach(System.out::println);
System.out.println(page.getCurrent());
System.out.println(page.getPages());
System.out.println(page.getSize());
System.out.println(page.getTotal());
System.out.println(page.hasNext());
System.out.println(page.hasPrevious());
}
@Test
public void deleteById() {
int i = userMapper.deleteById(1L);
System.out.println("影响行数 " + i);
}
//批量删除
@Test
public void deletes() {
int i = userMapper.deleteBatchIds(Arrays.asList(1L, 2L));
System.out.println(i);
}
@Test
public void testDeleteByMap() {
HashMap<String, Object> map = new HashMap<>();
map.put("name", "Helen");
map.put("age", 18);
int result = userMapper.deleteByMap(map);
System.out.println(result);
}
ALTER TABLE `user` ADD COLUMN `deleted` boolean
并加上 @TableLogic
注解 和 @TableField(fill = FieldFill.INSERT)
注解
@TableLogic
@TableField(fill = FieldFill.INSERT)
private Integer deleted;
@Override
public void insertFill(MetaObject metaObject) {
//第一个参数 填充的类的字段(不是sql表)
//第二个参数 当前系统时间
//第三个参数 传进来的参数
......
this.setFieldValByName("deleted", 0, metaObject);
}
此为默认值,如果你的默认值和mp默认的一样,该配置可无
#被删除是1
mybatis-plus.global-config.db-config.logic-delete-value=1
#未删除是0
mybatis-plus.global-config.db-config.logic-not-delete-value=0
@Bean
public ISqlInjector sqlInjector() {
return new LogicSqlInjector();
}
@Test
public void deleteById() {
int i = userMapper.deleteById(1L);
System.out.println("影响行数 " + i);
}
测试后分析打印的sql语句,包含 WHERE deleted=0
SELECT id,name,age,email,create_time,update_time,deleted FROM user WHERE deleted=0