• 使用SQL注入器就可以自定义例如selectById的默认方法。

    实现步骤
    Step1:创建定义方法的类;
    Step2:创建注入器;
    Step3:在Mapper中加入自定义方法。

    自定义注入器的简单使用

    第一步:创建定义方法的类

    public class DeleteAllMethod  extends AbstractMethod {
    @Override
    public MappedStatement injectMappedStatement(Class<?> mapperClass, Class<?> modelClass, TableInfo tableInfo) {
    //执行的SQL
    String sql = "delete from " + tableInfo.getTableName();
    //mapper接口方法名
    String method = "deleteAll";
    SqlSource sqlSource = languageDriver.createSqlSource(configuration, sql, modelClass);
    return addDeleteMappedStatement(mapperClass, method, sqlSource);
    }
    }

    第二步:创建注入器

    @Component
    public class MySqlInjector extends DefaultSqlInjector { @Override
    public List<AbstractMethod> getMethodList(Class<?> mapperClass) {
    //MP自定义的SQL语句,如果不添加,MP自定义的语句就不能用了
    List<AbstractMethod> methodList = super.getMethodList(mapperClass);
    methodList.add(new DeleteAllMethod());
    return methodList;
    } }

    DefaultSqlInjector、AbstractSqlInjector、ISqlInjector都能继承。

    第三步:在UserMapper中加入自定义方法deleteAll

        /**
    * 删除所有数据
    * @return 影响行数
    */
    int deleteAll();

    第四步:测试

        @Test
    public void deleteAll() {
    userMapper.deleteAll();
    }

    结果如下:

    这里需要注意:

    1. 不知道大家还记不记得之前逻辑删除,3.1.2之前的版本配置了一个SQL注入器。自定义SQL注入器不可以和逻辑删除一起配置,因为不能同时制定两个sql注入器,所以报错。有一个办法就是MySqlinjector不要继承DefaultSqlInjector直接继承LogicSqlInjector就可以了,这样配置这一个sql注入器,就既能使用逻辑删除又能加入自定义方法了。

    2. 如果报以下错误

    当删除所有数据时,由于有的数据行有外键的约束,不允许你进行物理删除。使用语句删除外键即可,alter table user drop foreign key manager_fk;。或者把外键字段原来默认的Restrict改成CASCADE。

    同时附上初始化数据语句,可以多次尝试

    #初始化数据:
    INSERT INTO user (id, name, age, email, manager_id
    , create_time)
    VALUES (1087982257332887553, '大boss', 40, 'boss@baomidou.com', NULL
    , '2019-01-11 14:20:20'),
    (1088248166370832385, '王天风', 25, 'wtf@baomidou.com', 1087982257332887553
    , '2019-02-05 11:12:22'),
    (1088250446457389058, '李艺伟', 28, 'lyw@baomidou.com', 1088248166370832385
    , '2019-02-14 08:31:16'),
    (1094590409767661570, '张雨琪', 31, 'zjq@baomidou.com', 1088248166370832385
    , '2019-01-14 09:15:15'),
    (1094592041087729666, '刘红雨', 32, 'lhm@baomidou.com', 1088248166370832385
    , '2019-01-14 09:48:16');

    优化

    如果每张表都需要删除全部,那我们为每个Mapper都写一个deleteAll()方法,非常的繁琐。这时候我们可以自己建立一个MyMapper接口,MyMapper接口继承BaseMapper,而类似于UserMapper的自定义Mapper全部继承MyMapper,这样我们就只需要在MyMapper中编写一个deleteAll方法,再由UserMpper和其他Mapper继承就可以实现deleteAll()方法的重用。

    选装件InsertBatchSomeColumn

    选择件其实也是自定义SQL注入器,但是是由官方提供的。InsertBatchSomeColumn的主要功能是批量新增数据,自选字段insert。接下来让我们看看是如何实现的吧。
    第一步:注入器中加入方法

        @Override
    public List<AbstractMethod> getMethodList(Class<?> mapperClass) {
    //MP自定义的SQL语句,如果不添加,MP自定义的语句就不能用了
    List<AbstractMethod> methodList = super.getMethodList(mapperClass);
    methodList.add(new DeleteAllMethod());
    //加入选装件InsertBatchSomeColumn,同时排除逻辑删除字段
    methodList.add(new InsertBatchSomeColumn(t->!t.isLogicDelete()));
    return methodList;
    }

    第二步:在MyMapper中加入insertBatchSomeColumn方法

    public interface MyMapper<T> extends BaseMapper<T> {
    /**
    * 删除所有数据
    * @return 影响行数
    */
    int deleteAll(); int insertBatchSomeColumn(List<T> list);
    }

    第三步:测试

     @Test
    public void insertBatchSomeColumn() { User user = new User();
    user.setName("王聚义");
    user.setAge(26);
    user.setEmail("wjy@163.com");
    user.setManagerId(1088248166370832385L); User user2 = new User();
    user2.setName("王收义");
    user2.setAge(28);
    user2.setEmail("wsy@163.com");
    user2.setManagerId(1088248166370832385L); List<User> list = new ArrayList<>();
    list.add(user);
    list.add(user2); userMapper.insertBatchSomeColumn(list);
    }

    可以看出批量插入成功。

    需要注意的是,User对象中为Null的值会覆盖没有被排除字段的默认值。例如version的默认值为1,但是我们插入的User的version为null,如果version这个字段在注入器组件中没有被排除,结果数据库中的值应该为null。

    同时作者在注解中也提示,选装器只在mysql中测试过,所以慎用。

    选装件LogicDeleteByIdWithFill

    LogicDeleteByIdWithFill的主要功能是根据id逻辑删除数据并带字段填充功能。例如在删除的时候需要添加操作人。

    第一步:注入器中加入方法

     methodList.add(new LogicDeleteByIdWithFill());

    第二步:在MyMapper中加入方法

        int deleteByIdWithFill(T entity);

    第三步:测试

     @Test
    public void deleteByIdWithFill() {
    User user = new User();
    user.setId(1346432120618147841L);
    user.setUpdateTime(LocalDateTime.now());
    userMapper.deleteByIdWithFill(user);
    }

    注意,如果想在删除的同时修改数据,被修改的数据必须有自动填充标识,不然无法完成更新。结果如下:

    选装件AlwaysUpdateSomeColumnById

    AlwaysUpdateSomeColumnById的主要功能是根据id更新固定的某些字段。

    第一步:注入器中加入方法

            methodList.add(new AlwaysUpdateSomeColumnById(t->!t.getColumn().equals("name")));

    第二步:在MyMapper中加入方法

     int alwaysUpdateSomeColumnById(@Param(Constants.ENTITY) T entity);

    第三步:测试

        @Test
    public void alwaysUpdateSomeColumnById() {
    User user = new User();
    user.setId(1346432120618147841L);
    user.setName("王地风");
    user.setAge(16);
    userMapper.alwaysUpdateSomeColumnById(user);
    }

    结果如下:

  • 相关阅读:

【Mybatis-Plus进阶学习(八)】SQL注入器的更多相关文章

  1. Mybatis基础进阶学习2

    Mybatis基础进阶学习2 1.测试基本结构 2.三个POJO package com.pojo; import java.io.Serializable; import java.util.Dat ...

  2. Java数据持久层框架 MyBatis之API学习八(Java API详解)

    对于MyBatis的学习而言,最好去MyBatis的官方文档:http://www.mybatis.org/mybatis-3/zh/index.html 对于语言的学习而言,马上上手去编程,多多练习 ...

  3. 【译】SQL Server索引进阶第八篇:唯一索引

    原文:[译]SQL Server索引进阶第八篇:唯一索引     索引设计是数据库设计中比较重要的一个环节,对数据库的性能其中至关重要的作用,但是索引的设计却又不是那么容易的事情,性能也不是那么轻易就 ...

  4. (转)MyBatis框架的学习(四)——Mapper.xml文件中的输入和输出映射以及动态sql

    http://blog.csdn.net/yerenyuan_pku/article/details/71893689 前面对MyBatis框架的学习中,我们对Mapper.xml映射文件多少有些了解 ...

  5. mybatis源码学习:基于动态代理实现查询全过程

    前文传送门: mybatis源码学习:从SqlSessionFactory到代理对象的生成 mybatis源码学习:一级缓存和二级缓存分析 下面这条语句,将会调用代理对象的方法,并执行查询过程,我们一 ...

  6. mybatis源码学习--spring+mybatis注解方式为什么mybatis的dao接口不需要实现类

    相信大家在刚开始学习mybatis注解方式,或者spring+mybatis注解方式的时候,一定会有一个疑问,为什么mybatis的dao接口只需要一个接口,不需要实现类,就可以正常使用,笔者最开始的 ...

  7. Java数据持久层框架 MyBatis之API学习一(简介)

    对于MyBatis的学习而言,最好去MyBatis的官方文档:http://www.mybatis.org/mybatis-3/zh/index.html 对于语言的学习而言,马上上手去编程,多多练习 ...

  8. Mybatis 源码学习系列

    前言 很久以前,我们学习了Java,从一个控制台的 Hello world .开始,我们进入了面向对象的世界. 然后由学习了SQL语言,可以写出SQL语句来将尘封在硬盘之下的数据库数据,展现出来. 后 ...

  9. MyBatis基础入门《八》查询参数传入Map

    MyBatis基础入门<八>查询参数传入Map 描述: 在执行select查询数据的时候,方法传入的参数是java.util.Map类型. 接口方法: xml文件 注意: 书写SQL语句的 ...

  10. SVG 学习<八> SVG的路径——path(2)贝塞尔曲线命令、光滑贝塞尔曲线命令

    目录 SVG 学习<一>基础图形及线段 SVG 学习<二>进阶 SVG世界,视野,视窗 stroke属性 svg分组 SVG 学习<三>渐变 SVG 学习<四 ...

随机推荐

  1. 封装一个Promise.all 的函数

    // 1. 准备三个异步函数 const promise1 = Promise.resolve('prom11ise1'); const promise2 = new Promise(function ...

  2. 深入理解Java并发读写锁——ReentrantReadWriteLock

    ReentrantReadWriteLock使用场景 ReentrantReadWriteLock 是 Java 的一种读写锁,它允许多个读线程同时访问,但只允许一个写线程访问(会阻塞所有的读写线程) ...

  3. KubeSphere 3.2.1 正式发布,多项功能优化来袭!

    KubeSphere 从诞生第一天起就 100% 开源,并以社区的方式成长,在各位社区小伙伴的共同努力下,KubeSphere 迅速走向全球,成为全世界最受欢迎的开源容器平台之一. 经过 3 年的发展 ...

  4. Nuget包本地调试以及自动打包上传

    项目过程中,经常需要打包Nuget包,并且引用本地Nuget包调试,完成后上传,因此做了点配置,分享给大家.如果大家有更好的方法欢迎分享. 1. 使用生成后事件自动打包 项目文件中本身是可以配置生成时 ...

  5. 人形机器人是未来?6只手臂加AI模型,异形机器人重塑种植业。

    图源:reddit user IlustriousTea 近日,一则视频在媒体上引起了人们的讨论.国外一处苹果园里,机械嗡鸣声中,六只机械手熟练且快速地采摘成熟的苹果. 这是2018年于美国加利福尼亚 ...

  6. Java面试题中高级进阶(JVM篇Java内存)

    前言 本来想着给自己放松一下,刷刷博客,突然被几道面试题难倒!说说Java内存结构?说说对象分配规则?描述一下JVM加载class文件的原理机制?似乎有点模糊了,那就大概看一下面试题吧.好记性不如烂键 ...

  7. TXT文本Log日志分割工具(附工具链接)

    前言 相信大家也会像我一样,生产出现了问题,拿下来的日志,用文本编辑器打开直接卡死,甚至说非常卡,查起来非常麻烦且费时间 当当当当 ~~~~~~ 又小,免费非安装的TXT文件分割器就此诞生 链接地址: ...

  8. Hadoop习题汇总

    目录 选择 单选 多选 判断 填空 简答 选择 单选 查看HDFS系统版本的Shell命令,以下正确的是(). hdfs -ver hdfs version (答案) dfsadmin version ...

  9. WiFi基础(七):WiFi漫游与WiFi组网

    liwen01 2024.10.27 前言 无线 WiFi 的优点是方便.灵活,可以接入各种设备.缺点就是信号容易被干扰.信号覆盖范围有限.下面几个问题应该很多人都有遇到过: 为何很多洗手间的 WiF ...

  10. Python预安装包制作

    预编译安装包 在Linux服务器上,经常会安装Python.Redis.Nginx等服务,不管离线.在线都需要编译.编译之前还需要安装一些依赖的环境,比如,openssl.gcc.g++等,但是mak ...