Mybatis 插件(plugins)

一、Mybatis 插件介绍

MyBatis 允许你在映射语句执行过程中的某一点进行拦截调用。默认情况下,MyBatis 允许使用插件来拦截的方法调用包括:

  • Executor (update, query, flushStatements, commit, rollback, getTransaction, close, isClosed)
  • ParameterHandler (getParameterObject, setParameters)
  • ResultSetHandler (handleResultSets, handleOutputParameters)
  • StatementHandler (prepare, parameterize, batch, update, query)

这些类中方法的细节可以通过查看每个方法的签名来发现,或者直接查看 MyBatis 发行包中的源代码。 如果你想做的不仅仅是监控方法的调用,那么你最好相当了解要重写的方法的行为。 因为在试图修改或重写已有方法的行为时,很可能会破坏 MyBatis 的核心模块。 这些都是更底层的类和方法,所以使用插件的时候要特别当心。
通过 MyBatis 提供的强大机制,使用插件是非常简单的,只需实现 Interceptor 接口,并指定想要拦截的方法签名即可。

// ExamplePlugin.java
@Intercepts({@Signature(
	type= Executor.class,
	method = "update",
	args = {MappedStatement.class, Object.class})})
public class ExamplePlugin implements Interceptor {
	private Properties properties = new Properties();
	public Object intercept(Invocation invocation) throws Throwable {
		// implement pre processing if need
		Object returnObject = invocation.proceed();
		// implement post processing if need
		return returnObject;
	}
	public void setProperties(Properties properties) {
    	this.properties = properties;
	}
}
<!-- mybatis-config.xml -->
<plugins>
  <plugin interceptor="org.mybatis.example.ExamplePlugin">
    <property name="someProperty" value="100"/>
  </plugin>
</plugins>

上面的插件将会拦截在 Executor 实例中所有的 “update” 方法调用, 这里的 Executor 是负责执行底层映射语句的内部对象。

注意:覆盖配置类
除了用插件来修改 MyBatis 核心行为以外,还可以通过完全覆盖配置类来达到目的。只需继承配置类后覆盖其中的某个方法,再把它传递到 SqlSessionFactoryBuilder.build(myConfig) 方法即可。再次重申,这可能会极大影响 MyBatis 的行为,务请慎之又慎。

二、自定义一个插件

数据脱敏:对用户信息中的身份证 id_card 进行加密:只显示前面三位和后面4位

1. 实现Interceptor接口

import org.apache.ibatis.executor.Executor;
import org.apache.ibatis.mapping.BoundSql;
import org.apache.ibatis.mapping.MappedStatement;
import org.apache.ibatis.mapping.ResultSetType;
import org.apache.ibatis.plugin.*;
import org.apache.ibatis.session.ResultHandler;
import org.apache.ibatis.session.RowBounds;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.util.StringUtils;
import java.util.List;
import java.util.Properties;
/**
 * 数据脱敏
 * 对用户信息中的身份证 id_card 进行加密:只显示前面三位和后面4位
 */
@Intercepts({
        @Signature(type = Executor.class, method = "query",
                args = {MappedStatement.class, Object.class, RowBounds.class, ResultHandler.class})
})
public class DataDesensitizationInterceptor implements Interceptor {
    private static final Logger logger = LoggerFactory.getLogger(DataDesensitizationInterceptor.class);
    @Override
    public Object intercept(Invocation invocation) throws Throwable {
        logger.info("-------------拦截目标对象的目标方法的执行-------------");
        Executor executor = (Executor) invocation.getTarget();
        Object[] args = invocation.getArgs();
        MappedStatement mappedStatement = (MappedStatement) args[0];
        Object parameter = args[1];
        BoundSql boundSql = mappedStatement.getBoundSql(parameter);
        String sql = boundSql.getSql();
        sql = sql.replaceAll("\n", " ").replaceAll(" +", " ").toLowerCase();
        RowBounds rowBounds = (RowBounds) args[2];
        ResultHandler resultHandler = (ResultHandler) args[3];
        Object obj = invocation.proceed();
        if (sql.indexOf("from user") > -1) {
            ResultSetType resultSetType = mappedStatement.getResultSetType();
            if (obj instanceof User) {
                User user = (User) obj;
                String idCard = user.getIdCard();
                if (!StringUtils.isEmpty(idCard)) {
                    user.setIdCard(idCard.substring(0, 3) + "***" + idCard.substring(idCard.length() - 4));
                }
                return user;
            } else if (obj instanceof List) {
                List list = (List) obj;
                for (Object item : list){
                    if (item instanceof User) {
                        User user = (User) item;
                        String idCard = user.getIdCard();
                        if (!StringUtils.isEmpty(idCard)) {
                            user.setIdCard(idCard.substring(0, 3) + "***" + idCard.substring(idCard.length() - 4));
                        }
                    }
                }
                return list;
            }
        }
        return obj;
    }
    @Override
    public Object plugin(Object target) {
        return Plugin.wrap(target, this);
    }
    @Override
    public void setProperties(Properties properties) {
        // 获取自定义配置参数
        logger.info("-------------获取自定义配置参数-------------");
    }
}

2. 注册插件

import org.mybatis.spring.annotation.MapperScan;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import java.util.Properties;

@Configuration
@MapperScan({"com.xxx.mapper"})
public class MybatisConfig {
    //注册插件
    @Bean
    public DataDesensitizationInterceptor dataDesensitizationInterceptor() {
        DataDesensitizationInterceptor dataDesensitizationInterceptor = new DataDesensitizationInterceptor();
        // 设置参数
        Properties properties = new Properties();
        dataDesensitizationInterceptor.setProperties(properties);
        return dataDesensitizationInterceptor;
    }
}

参考文档:

Mybatis 官网

  • 0
    点赞
  • 5
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值