1. 首页 » 知识阅读

动态描写的句子(Mybatis)

Mybatis-Plus(简称MP)是一个 Mybatis 的增强工具,那么它是怎么增强的呢?其实就是它已经封装好了一些crud方法,开发就不需要再写xml了,直接调用这些方法就行,就类似于JPA。那么这篇文章就来阅读以下MP的具体实现,看看是怎样实现这些增强的。

Mybatis Plus 实现动态SQL语句的原理

入口类:MybatisSqlSessionFactoryBuilder

通过在入口类 MybatisSqlSessionFactoryBuilder#build方法中, 在应用启动时, 将mybatis plus(简称MP)自定义的动态配置xml文件注入到Mybatis中。

public classMybatisSqlSessionFactoryBuilderextendsSqlSessionFactoryBuilder{

public SqlSessionFactory build(Configuration configuration) {

// ... 省略若干行

if (globalConfig.isEnableSqlRunner) {

new SqlRunnerInjector.inject(configuration);

}

// ... 省略若干行

return sqlSessionFactory;

}

}

这里涉及到2个MP2个功能类

扩展继承自Mybatis的MybatisConfiguration类: MP动态脚本构建,注册,及其它逻辑判断。

SqlRunnerInjector: MP默认插入一些动态方法的xml 脚本方法。

MybatisConfiguration类

这里我们重点剖析MybatisConfiguration类,在MybatisConfiguration中,MP初始化了其自身的MybatisMapperRegistry,而MybatisMapperRegistry是MP加载自定义的SQL方法的注册器。

MybatisConfiguration中很多方法是使用MybatisMapperRegistry进行重写实现

其中有3个重载方法addMapper实现了注册MP动态脚本的功能。

public classMybatisConfigurationextendsConfiguration{

/**

* Mapper 注册

*/

protected final MybatisMapperRegistry mybatisMapperRegistry = new MybatisMapperRegistry(this);

// ....

/**

* 初始化调用

*/

publicMybatisConfiguration {

super;

this.mapUnderscoreToCamelCase = true;

languageRegistry.setDefaultDriverClass(MybatisXMLLanguageDriver.class);

}

/**

* MybatisPlus 加载 SQL 顺序:

*

1、加载 XML中的 SQL

*

2、加载 SqlProvider 中的 SQL

*

3、XmlSql 与 SqlProvider不能包含相同的 SQL

*

调整后的 SQL优先级:XmlSql > sqlProvider > CurdSql

*/

@Override

publicvoidaddMappedStatement(MappedStatement ms) {

// ...

}

// ... 省略若干行

/**

* 使用自己的 MybatisMapperRegistry

*/

@Override

public voidaddMapper(Class type) {

mybatisMapperRegistry.addMapper(type);

}

// .... 省略若干行

}

在MybatisMapperRegistry中,MP将mybatis的MapperAnnotationBuilder替换为MP自己的MybatisMapperAnnotationBuilder

public classMybatisMapperRegistryextendsMapperRegistry{

@Override

public voidaddMapper(Class type) {

// ... 省略若干行

MybatisMapperAnnotationBuilder parser = new MybatisMapperAnnotationBuilder(config, type);

parser.parse;

// ... 省略若干行

}

}

在MybatisMapperRegistry类的addMapper方法中,真正进入到MP的核心类MybatisMapperAnnotationBuilder,MybatisMapperAnnotationBuilder这个类是MP实现动态脚本的关键类。

MybatisMapperAnnotationBuilder动态构造

在MP的核心类MybatisMapperAnnotationBuilder的parser方法中,MP逐一遍历要加载的Mapper类,加载的方法包括下面几个

public classMybatisMapperAnnotationBuilderextendsMapperAnnotationBuilder{

@Override

publicvoidparse {

//... 省略若干行

for (Method method : type.getMethods) {

/** for循环代码, MP判断method方法是否是@Select @Insert等mybatis注解方法**/

parseStatement(method);

InterceptorIgnoreHelper.initSqlParserInfoCache(cache, mapperName, method);

SqlParserHelper.initSqlParserInfoCache(mapperName, method);

}

/** 这2行代码, MP注入默认的方法列表**/

if (GlobalConfigUtils.isSupperMapperChildren(configuration, type)) {

GlobalConfigUtils.getSqlInjector(configuration).inspectInject(assistant, type);

}

//... 省略若干行

}

@Override

publicvoidinspectInject(MapperBuilderAssistant builderAssistant, Class mapperClass) {

Class modelClass = extractModelClass(mapperClass);

//... 省略若干行

List methodList = this.getMethodList(mapperClass);

TableInfo tableInfo = TableInfoHelper.initTableInfo(builderAssistant, modelClass);

// 循环注入自定义方法

methodList.forEach(m -> m.inject(builderAssistant, mapperClass, modelClass, tableInfo));

mapperRegistryCache.add(className);

}

}

public classDefaultSqlInjectorextendsAbstractSqlInjector{

@Override

public List getMethodList(Class mapperClass) {

return Stream.of(

new Insert,

//... 省略若干行

new SelectPage

).collect(toList);

}

}

在MybatisMapperAnnotationBuilder中,MP真正将框架自定义的动态SQL语句注册到Mybatis引擎中。而AbstractMethod则履行了具体方法的SQL语句构造。

具体的AbstractMethod实例类,构造具体的方法SQL语句

以 SelectById 这个类为例说明下

/**

* 根据ID 查询一条数据

*/

public classSelectByIdextendsAbstractMethod{

@Override

public MappedStatement injectMappedStatement(Class mapperClass, Class modelClass, TableInfo tableInfo) {

/** 定义 mybatis xml method id, 对应 **/

SqlMethod sqlMethod = SqlMethod.SELECT_BY_ID;

/** 构造id对应的具体xml片段 **/

SqlSource sqlSource = new RawSqlSource(configuration, String.format(sqlMethod.getSql,

sqlSelectColumns(tableInfo, false),

tableInfo.getTableName, tableInfo.getKeyColumn, tableInfo.getKeyProperty,

tableInfo.getLogicDeleteSql(true, true)), Object.class);

/** 将xml method方法添加到mybatis的MappedStatement中 **/

return this.addSelectMappedStatementForTable(mapperClass, getMethod(sqlMethod), sqlSource, tableInfo);

}

}

至此,MP完成了在启动时加载自定义的方法xml配置的过程,后面的就是mybatis ${变量}#{变量}的动态替换和预编译,已经进入mybatis自有功能。

总结一下

MP总共改写和替换了mybatis的十多个类,主要如下图所示:

Mybatis Plus 实现动态SQL语句的原理

总体上来说,MP实现mybatis的增强,手段略显繁琐和不够直观,其实根据MybatisMapperAnnotationBuilder构造出自定义方法的xml文件,将其转换为mybatis的Resource资源,可以只继承重写一个Mybatis类:SqlSessionFactoryBean 比如如下:

public classYourSqlSessionFactoryBeanextendsSqlSessionFactoryBeanimplementsApplicationContextAware{

private Resource mapperLocations;

@Override

publicvoidsetMapperLocations(Resource... mapperLocations) {

super.setMapperLocations(mapperLocations);

/** 暂存使用mybatis原生定义的mapper xml文件路径**/

this.mapperLocations = mapperLocations;

}

/**

* {@inheritDoc}

*/

@Override

public void afterPropertiesSet throws Exception {

ConfigurableListableBeanFactory beanFactory = getBeanFactory;

/** 只需要通过将自定义的方法构造成xml resource和原生定义的Resource一起注入到mybatis中即可, 这样就可以实现MP的自定义动态SQL和原生SQL的共生关系**/

this.setMapperLocations(InjectMapper.getMapperResource(this.dbType, beanFactory, this.mapperLocations));

super.afterPropertiesSet;

}

}

在这边文章中,简单介绍了MP实现动态语句的实现过程,并且给出一个可能的更便捷方法。

来源:juejin.cn/post/6883081187103866894

#投 稿 通 道#

让你的博客被更多人看到

如果你在 CSDN、博客园、掘金等平台有写技术博客的习惯,想让自己的原创博客被更多人看到,可以来 Java后端 投稿。

Java后端 鼓励读者投稿个人技术博客、面试经验、教程。不管是入门的图文教程、还是热门技术讲解,只要你喜欢写东西,我们欢迎你来投稿。

声明:本文由"麦兜"发布,不代表"知识分享"立场,转载联系作者并注明出处:https://www.029ipr.com/zhishi/3633.html