Bootstrap

【107期】面试官问:Mybatis Plus 实现动态 SQL 语句的原理,你知道吗?

/**

* 初始化调用

*/

public MybatisConfiguration() {

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

public void addMappedStatement(MappedStatement ms) {

// …

}

// … 省略若干行

/**

* 使用自己的 MybatisMapperRegistry

*/

@Override

public  void addMapper(Class type) {

mybatisMapperRegistry.addMapper(type);

}

// … 省略若干行

}

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

public class MybatisMapperRegistry extends MapperRegistry {

@Override

public  void addMapper(Class type) {

// … 省略若干行

MybatisMapperAnnotationBuilder parser = new MybatisMapperAnnotationBuilder(config, type);

parser.parse();

// … 省略若干行

}

}

在MybatisMapperRegistry类的addMapper方法中,真正进入到MP的核心类MybatisMapperAnnotationBuilder,MybatisMapperAnnotationBuilder这个类是MP实现动态脚本的关键类。更多面试题,见微信小程序Java精选面试题,内涵大量初级、中级、高级面试题,支持随时随地刷题。

MybatisMapperAnnotationBuilder动态构造

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

public class MybatisMapperAnnotationBuilder extends MapperAnnotationBuilder {

@Override

public void parse() {

//… 省略若干行

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

public void inspectInject(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 class DefaultSqlInjector extends AbstractSqlInjector {

@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 class SelectById extends AbstractMethod {

@Override

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

/** 定义 mybatis xml method id, 对应 <id=“xyz”> **/

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自有功能。更多面试题,见微信小程序Java精选面试题,内涵大量初级、中级、高级面试题,支持随时随地刷题。

总结一下

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

Spring Boot 短信验证码登录

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

public class YourSqlSessionFactoryBean extends SqlSessionFactoryBean implements ApplicationContextAware {

private Resource[] mapperLocations;

@Override

public void setMapperLocations(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实现动态语句的实现过程,并且给出一个可能的更便捷方法。

作者:稻草江南

自我介绍一下,小编13年上海交大毕业,曾经在小公司待过,也去过华为、OPPO等大厂,18年进入阿里一直到现在。

深知大多数Java工程师,想要提升技能,往往是自己摸索成长或者是报班学习,但对于培训机构动则几千的学费,着实压力不小。自己不成体系的自学效果低效又漫长,而且极易碰到天花板技术停滞不前!

因此收集整理了一份《2024年Java开发全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。img

既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上Java开发知识点,真正体系化!

由于文件比较大,这里只是将部分目录截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且会持续更新!

如果你觉得这些内容对你有帮助,可以扫码获取!!(备注Java获取)

img

面试结束复盘查漏补缺

每次面试都是检验自己知识与技术实力的一次机会,面试结束后建议大家及时总结复盘,查漏补缺,然后有针对性地进行学习,既能提高下一场面试的成功概率,还能增加自己的技术知识栈储备,可谓是一举两得。

以下最新总结的阿里P6资深Java必考题范围和答案,包含最全MySQL、Redis、Java并发编程等等面试题和答案,用于参考~

重要的事说三遍,关注+关注+关注!

历经30天,说说我的支付宝4面+美团4面+拼多多四面,侥幸全获Offer

image.png

更多笔记分享

历经30天,说说我的支付宝4面+美团4面+拼多多四面,侥幸全获Offer

《互联网大厂面试真题解析、进阶开发核心学习笔记、全套讲解视频、实战项目源码讲义》点击传送门即可获取!
得。

以下最新总结的阿里P6资深Java必考题范围和答案,包含最全MySQL、Redis、Java并发编程等等面试题和答案,用于参考~

重要的事说三遍,关注+关注+关注!

[外链图片转存中…(img-gc7qiaJF-1713520906152)]

[外链图片转存中…(img-gtZMldX9-1713520906153)]

更多笔记分享

[外链图片转存中…(img-O9SzcIPc-1713520906155)]

《互联网大厂面试真题解析、进阶开发核心学习笔记、全套讲解视频、实战项目源码讲义》点击传送门即可获取!

;