Spring整合MyBatis之底层原理
作者 | 波哥
审校 | 孙淑娟
如果老铁们对Spring框架足够熟悉,整合MyBatis其实很容易理解,当然这里假定老铁们也已经熟悉了MyBatis框架。
在我们正常的应用开发过程中,使用MyBatis一般分为如下几个步骤:
1.在配置类上增加MapperScan注解,例如:@MapperScan(basePackages = {"com.test.dao"},annotationClass = Mapper.class);
2.在basePackages指定的目录下创建待MyBatis读取的接口文件,例如: @Mapper public interface TestMapper { ...... }1.2.3.4.
3.在Service或者其他地方使用该Mapper来操作数据库。
使用起来是很简单的,但是有没有老铁想过,为什么做了这么一个简单的配置,这个Mapper就能操作数据库了?按理说这个Mapper是个接口,应该是不能被创建才对啊!如果你有这个疑问,证明你是个爱思考的好童鞋。
咱们直接进入主题。Spring要与MyBatis整合,简单来说只要解决如下两个问题: 一、Spring如何知道哪些类应该被管理?
要让Spring去管理Bean的生命周期,首先需要对应的类被Spring扫描到,并且生成DeanDefinition,然后基于BeanDefinition生成Bean。下面对Spring生成BeanDefinition的方式做个小总结: 包含Component、Configuration、ComponentScan、Import、ImportResource注解的类; Import注解中指定的类、被Bean注解标注的方法所在的类; 实现了ImportBeanDefinitionRegistrar接口,并且在registerBeanDefinitions方法中调用registry直接注册的类; 实现了ImportSelector接口,并且在selectImports方法中返回的字符串对应的类; 直接调用register方法; 另外Spring还提供了一个扩展,可以让开发者自己指定需要被管理的类对应的类型:通过往includeFilters中添加注解类类型。
我们分析源码,第一步得找到它的入口,Spring整合MyBatis的入口,毫无疑问是MapperScan这个注解,在MapperScan注解上包含Import(MapperScannerRegistrar.class)注解,Spring整合MyBatis正是用了Import和ImportBeanDefinitionRegistrar的方式。我们先通过一张流程图来了解下整体流程,然后再慢慢品。
我们来看MapperScannerRegistrar这个类的继承关系图:
MapperScannerRegistrar是ImportBeanDefinitionRegistrar的实现类,Spring会去调用这个类的registerBeanDefinitions方法添加beanDefinition,这个方法中具体做了些什么呢:
获取MapperScan注解的配置信息,比如basePackages、annotationClass,basePackages表示需要扫描的路径,annotationClass则是指定了增加了这种注解类的类需要被Spring进行管理,比如增加了Mapper注解的类需要被Spring管理。
生成MapperScannerConfigurer这个类型的beanDefinition,并且把MapperScan注解的配置信息添加到该beanDefinition的属性集合中。
后续Spring就会基于这个MapperScannerConfigurer做一系列文章,看下它的继承关系:
它是BeanDefinitionRegistryPostProcessor的实现类,是一个BeanFactory后置处理器,Spring会调用该类的postProcessBeanDefinitionRegistry方法来添加beanDefinition的操作,MapperScannerConfigurer这个类中具体实现如下:
它定义了ClassPathMapperScanner这个扫描器,然后使用这个扫描器来扫描类,扫描哪些类呢?扫描有Mapper注解的类,看它的关系知道,它是ClassPathBeanDefinitionScanner的子类,而spring则是使用ClassPathBeanDefinitionScanner来进行扫描的。
为什么ClassPathMapperScanner能够扫描到带有Mapper注解的类呢?看上面代码,就是通过调用registerFilters方法来添加includeFilter(实际类型是:TypeFilter),这个就是Spring提供的扩展点,让咱们自己来指定需要被扫描的类,这里使用的是MappScan注解中annotationClass属性配置的注解类型,我们这里配置了Mapper,所以调用scan方法开启扫描后,Spring就会将包含Mapper注解的类扫描为BeanDefinition。注意这里的扫描能力还是调用Spring的扫描器来实现的,ClassPathMapperScanner并没有修改,只是当扫描完成后,ClassPathMapperScanner会对扫描出的BeanDefinition进行重新处理,主要是把原来的BeanClass修改成了MapperFactoryBean.class:
而这个MapperFactoryBean是FactoryBean的实现类,老铁们,FactoryBean这种Bean有什么特点?这个可是面试的高发点哦。
做个小小的总结:Spring扫描到有Mapper注解的类,生成BeanDefinition,并且将这一类BeanDefinition的BeanClass的值修改为MapperFactoryBean,也就是说它的类型不再是咱们自己编写的Mapper接口了,而是一个FactoryBean,这样Spring就能做妖了。 二、Mapper注解的类是接口
那如何实例化呢?
到这一步,其实老铁们也大概清楚了,Spring在实例化Mapper实例时,实际上首先会实例化MapperFactoryBean,然后再调用它的getObject方法。我们知道在Java里面接口是肯定不能被实例化的,那这个被实例化的对象只能是一个代理对象,所以我们有理由猜想这个getObject方法应该是用来创建代理对象的。要创建代理对象,得从以下两个方面着手: 1.准备工作
这里Spring准备的是接口类型和创建代理对象的代理工厂。具体如何准备的呢?来看上述MapperFactoryBean类型的整体继承关系:
它实现了InitializingBean,于是可以知道,在MapperFactoryBean初始化完成后,Spring会调用它的afterPropertiesSet方法,从而会执行到checkDaoConfig方法:
在该方法中调用configuration的addMapper方法,这个方法里面到底做了啥?
看出门道了吗?其实就是使用Mapper的接口类型作为key,MapperProxyFactory做为value,然后添加到mapperRegistry对象的Map集合中,注意这个type同时也是MapperProxyFactory对象的构造参数哦。 2.实例化
上述动作已经准备好了,接下来就应该是创建了。Spring在创建完成MapperFactoryBean对象后,最终会调用它的getObject方法来获得真实的对象:
getObject方法中,会调用getMapper方法,该方法中从knowMappers这个Map集合中拿到MapperProxyFactory对象,这个对象不就是我们在准备阶段添加的嘛!它就是用来创建代理对象的工厂。
从上面代码中也不难看出,确实是为咱们自己的接口创建了代理对象,而代理类的处理类则是MapperProxy对象,也就是说对所有接口对象的调用,都会进入MapperProxy的Invoke方法,至此Spring成功对接MyBatis。 作者介绍
波哥,互联行业从业10余年,先后担任项目总监及架构师。目前专攻技术,喜欢研究技术原理。技术全面,主攻java,精通JVM底层机制及Spring全家桶底层框架原理,熟练掌握当前主流的中间件、服务网格等技术原理。
新图发布!詹姆斯韦伯太空望远镜首次直接拍摄到系外行星图像Forthefirsttimeever,astronomersusedNASAsJamesWebbSpaceTelescopetotakeadirectimageofaplanet
营地炉石狂野日报20220913宇宙萨偶数术暗牧等今天的卡组是宇宙萨偶数术暗牧变速宇宙暗牧厚薄不同的3套宇宙猎开门法。AAEBAdOLBSj6Dva9AuTCAtEAovOAqvnAvPnAu3Aq2RA7WYA72ZA5ybAy
女科学家孙珍生命不息,科研事业不止文羊城晚报全媒体记者李钢图受访者供图她,是我国最了解深海地质构造的科学家之一。她,也是我国最懂得如何在深海钻探的女性。她,就是中国科学院南海海洋研究所研究员我国第一位担任国际大洋钻
李易峰事件女主亲爱的吃瓜群众大家好!相信大家对于李易峰的事都略知一二吧!高强度网上冲浪的时候又发现了一件大瓜事件,就在李易峰进去后,有一名网红出来说自己与李易峰交往了三年,三年期间李某出轨的时候
秋天是孩子骨骼发育的好季节,常吃6道家常菜,个头猛长,吃饭香秋天是孩子骨骼发育的好季节,常吃6道家常菜,个头猛长,吃饭香原创202208161051第一美食秋天是孩子骨骼发育的最佳时期,应该给孩子吃什么最佳?首先应该尽量减少孩子吃甜食零食和
王者荣耀9。14更新,韩信晋升T0野王,大乔只能传送一名队友了前言王者荣耀自公测到现在已经七年多,目前也来到了S28赛季,相信大家对于这款游戏都非常熟悉了吧?我们都知道,每次新赛季更新前,官方都会对抢先服进行更新,用以测试英雄的平衡性,而在S
别白费劲了!你用的瘦腿方法都是骗人的其实咱的灵感来源大家一看就知道了,出自倪妮的出圈名场面细雨红裙,这暗夜灯光氛围感拉爆了好吗!最让人挪不开眼的就是倪妮的这双大长腿,把她的韵味发挥到极致,所以说拥有一双修长笔直的腿确
(经济)欧盟提出紧急干预措施应对能源价格高涨新华社布鲁塞尔9月14日电(记者康逸)欧盟委员会14日提议对欧洲能源市场进行紧急干预,以缓解近期能源价格大幅上涨。此前备受关注的设置天然气价格上限的提议因争议较大未包括其中。欧盟委
别贪小便宜了!揭秘长期使用华强北耳机的危害听力受损或不可逆苹果AirPods等TWS蓝牙真无线耳机的爆火,不仅改变了我们使用耳机的方式,同时也让众多华强北生产的山寨耳机迎来了销量上的春天。凭借着一比一复刻知名品牌产品的外观功能,以及不到一
夜读最惬意的人生,是做最真实的自己努力地奔赴热爱。人这一生,无论怎么做,都不可能令所有人满意,索性,就将他人的评议搁置一边,做真实的自己吧。有位作家说生活是属于每个人自己的感受,不属于任何别人的看法。找到真正的自己
以太坊合并,为何二级价格不会暴跌?随着合并到来,以太坊价格正在上涨至临界水平以上,看涨势头可能会受到新的推动,最终将ETH推高至2,000美元以上。与此同时,市场参与者正在按市值推测第二种加密货币的近期未来。合并后
英雄传说黎之轨迹2绯红色的梦魇,与再次连接的羁绊欢迎关注,获取更多游戏评测资讯,入手与否不再犹豫喜欢的不妨点个赞唷()身为法老控以及老埃及人的我们又等到了一年一度的年货查收环节,也就在新年过后,英雄传说黎之轨迹2(下面简称黎轨2
被问100遍了的口红色号测色合集来了!几乎完全不挑皮不显黑的!被问100遍了的口红色号测色合集来了!几乎完全不挑皮不显黑的!欧莱雅619(秋叶)色系橘调枫叶色质地超级滋润超级舒服,沾杯苹果摄像头是傻子摄像头吧为啥拍出来的日光下都一毛一样!!!
写个代码有Bug,被公司起诉到法院这年头,给公司写个代码出现Bug还有这么大的风险?这个问题,就好比士兵需不需要承担战场失败的责任。不要误会,我不是说战败作为士兵就一定没有责任,更不是说士兵因为战败就要上军事法庭,
优维低代码UseProviderHook优维低代码技术专栏,是一个全新的技术为主的专栏,由优维技术委员会成员执笔,基于优维7年低代码技术研发及运维成果,主要介绍低代码相关的技术原理及架构逻辑,目的是给广大运维人提供一个技
看看人家阿里的低代码引擎,多优雅!特性引擎协议使用示例工程化配置cdn可选方式界面功能物料面板大纲面板源码面板Schema编辑编辑画布区域属性样式事件高级案例传送门LowCodeEngine是由阿里巴巴钉钉团队开源
今日雨水你不知道的中医养生秘诀今日雨水,是二十四节气中的第二个节气。雨水节气的到来,气温开始升高,日照降雨逐渐增加,此时肝气旺盛,寒湿困脾,湿邪难除,应该在养阳护肝的前提下,特别注意顾护脾胃。东阿双胶膏方研究院
独一代为父母养老问题网上抱团互助独生子女,自小便是家中独宠。随着时光流逝,父母年过花甲渐入古稀,独生子女的压力日渐凸显。在豆瓣有一个独生子女父母养老交流小组,里面有八万多个独生子女正在抱团互助惺惺相惜。该不该为赡
视评来了丨背蜀道难难于上青天?视频加载中孩子,乖,只要你背得下蜀道难,我们家就可以少给一张门票钱哦!最近剑门关景区搞了个活动,但凡背诵李白蜀道难全文者,均可获赠一张门票,这个活动还挺受欢迎的,据说目前已经有36
抗震救灾岂能打政治牌?(观象台)一场百年不遇的特大地震让土耳其和叙利亚两国陷入世纪灾难。数万人伤亡,数千座建筑垮塌,无数家庭一夜之间破碎大灾当前,团结互助显得尤为珍贵,也尤为迫切。然而,令人震惊的是,某些国家非但
河北保定富豪榜大洗牌,魏建军身价缩水880亿,一位80后闯进前三素有京畿重地,九省通衢美誉的保定市位于河北省中部,地处华北平原腹地,是京津冀地区的中心城市之一,同时也是河北重要的制造业基地。保定历史悠久,自古人杰地灵,建城至今已有三千余年,是中
赧水情连载缘份1937年七七事变,中国掀起了史无前例的抗日高潮,国共第二次合作,工农红军改编成八路军和新四军奔赴敌后配合国军共同抗战。日寇来势汹汹,南京武汉先后沦陷,国民政府迁都重庆,大批重要的