电脑知识|欧美黑人一区二区三区|软件|欧美黑人一级爽快片淫片高清|系统|欧美黑人狂野猛交老妇|数据库|服务器|编程开发|网络运营|知识问答|技术教程文章 - 好吧啦网

您的位置:首頁技術(shù)文章
文章詳情頁

MyBatis Plus插件機(jī)制與執(zhí)行流程原理分析詳解

瀏覽:2日期:2023-10-22 17:17:04

MyBatis Plus插件

MyBatis Plus提供了分頁插件PaginationInterceptor、執(zhí)行分析插件SqlExplainInterceptor、性能分析插件PerformanceInterceptor以及樂觀鎖插件OptimisticLockerInterceptor。

Mybatis 通過插件 (Interceptor) 可以做到攔截四大對象相關(guān)方法的執(zhí)行 ,根據(jù)需求完成相關(guān)數(shù)據(jù)的動態(tài)改變。

四大對象是:

Executor StatementHandler ParameterHandler ResultSetHandler

四大對象的每個對象在創(chuàng)建時,都會執(zhí)行interceptorChain.pluginAll(),會經(jīng)過每個插件對象的 plugin()方法,目的是為當(dāng)前的四大對象創(chuàng)建代理。代理對象就可以攔截到四大對象相關(guān)方法的執(zhí)行,因為要執(zhí)行四大對象的方法需要經(jīng)過代理 。

① xml下插件的配置

如下所示:

<bean class='com.baomidou.mybatisplus.spring.MybatisSqlSessionFactoryBean'><!-- 數(shù)據(jù)源 --><property name='dataSource' ref='dataSource'></property><property name='configLocation' value='classpath:mybatis-config.xml'></property><!-- 別名處理 --><property name='typeAliasesPackage' value='com.jane.mp.beans'></property><!-- 注入全局MP策略配置 --><property name='globalConfig' ref='globalConfiguration'></property><!-- 插件注冊 --><property name='plugins'><list><!-- 注冊分頁插件 --><bean class='com.baomidou.mybatisplus.plugins.PaginationInterceptor'></bean><!-- 注冊執(zhí)行分析插件 --><bean class='com.baomidou.mybatisplus.plugins.SqlExplainInterceptor'><property name='stopProceed' value='true'></property></bean><!-- 注冊性能分析插件 --><bean class='com.baomidou.mybatisplus.plugins.PerformanceInterceptor'><property name='format' value='true'></property><!-- <property name='maxTime' value='5'></property> --></bean><!-- 注冊樂觀鎖插件 --><bean class='com.baomidou.mybatisplus.plugins.OptimisticLockerInterceptor'></bean></list></property></bean>

② springboot下注冊插件

這里以分頁插件為例:

@Bean public PaginationInterceptor paginationInterceptor() { PaginationInterceptor paginationInterceptor = new PaginationInterceptor(); // 設(shè)置請求的頁面大于最大頁后操作, true調(diào)回到首頁,false 繼續(xù)請求 默認(rèn)false // paginationInterceptor.setOverflow(false); // 設(shè)置最大單頁限制數(shù)量,默認(rèn) 500 條,-1 不受限制 // paginationInterceptor.setLimit(500); // 開啟 count 的 join 優(yōu)化,只針對部分 left join paginationInterceptor.setCountSqlParser(new JsqlParserCountOptimize(true)); return paginationInterceptor; }

③ SqlExplainInterceptor

SQL執(zhí)行分析攔截器,全類名是com.baomidou.mybatisplus.plugins.SqlExplainInterceptor,只支持 MySQL5.6.3以上版本。

該插件的作用是分析 DELETE UPDATE語句 ,防止小白或者惡意進(jìn)行DELETE UPDATE全表操作,不建議在生產(chǎn)環(huán)境中使用會造成性能下降,

在插件的底層通過SQL語句分析命令 Explain 分析當(dāng)前的 SQL語句,根據(jù)結(jié)果集中的 Extra列來斷定當(dāng)前是否全表操作。

④ 性能分析插件

性能分析攔截器,全類名是com.baomidou.mybatisplus.plugins.PerformanceInterceptor,用于輸出每條 SQL 語句及其執(zhí)行時間。SQL性能執(zhí)行分析 ,開發(fā)環(huán)境使用 超過指定時間,停止運(yùn)行。

⑤ 樂觀鎖插件

全類名是com.baomidou.mybatisplus.plugins.OptimisticLockerInterceptor。如果想實現(xiàn)如下需求 : 當(dāng)要更新一條記錄的時候,希望這條記錄沒有被別人更新,就可以使用該插件進(jìn)行判斷。

樂觀鎖的實現(xiàn)原理(@Version 用于注解實體字段,必須要有) :

取出記錄時,獲取當(dāng)前 version 更新時,帶上這個version 執(zhí)行更新時,set version = yourVersion+1 where version = yourVersion 如果 version不對,就更新失敗

【2】獲取sqlSessionFactoryBean

如下圖所示,在系統(tǒng)啟動時會初始化定義的bean。DefaultListableBeanFactory.preInstantiateSingletons方法中會從beanDefinitionNames中獲取bean name然后依次創(chuàng)建。

這里可以看到RootBeanDefinition是com.baomidou.mybatisplus.spring.MybatisSqlSessionFactoryBean。

MyBatis Plus插件機(jī)制與執(zhí)行流程原理分析詳解

① 獲取bean的過程中bean屬性

如下所示,在getBean過程中可以看到bean的屬性:

MyBatis Plus插件機(jī)制與執(zhí)行流程原理分析詳解

② createBean

第一次獲取bean的時候會走到AbstractAutowireCapableBeanFactory.createBean進(jìn)行bean的創(chuàng)建。

/** 創(chuàng)建一個bean實例,為bean實例設(shè)置屬性值,調(diào)用post-processors-bean后置處理器 */@Overrideprotected Object createBean(String beanName, RootBeanDefinition mbd, Object[] args) throws BeanCreationException {//...暫時忽略其他代碼 try {// 這里會首先觸發(fā)BeanPostProcessors ,如果這里能獲取到bean則直接返回,不再走doCreateBean Object bean = resolveBeforeInstantiation(beanName, mbdToUse); if (bean != null) { return bean; } }//...暫時忽略其他代碼

在AbstractAutowireCapableBeanFactory.resolveBeforeInstantiation中就會分別執(zhí)行bean后置處理器的前置和后置方法。

protected Object resolveBeforeInstantiation(String beanName, RootBeanDefinition mbd) { Object bean = null; if (!Boolean.FALSE.equals(mbd.beforeInstantiationResolved)){ // Make sure bean class is actually resolved at this point. if (!mbd.isSynthetic() && hasInstantiationAwareBeanPostProcessors()){ Class<?> targetType = determineTargetType(beanName, mbd); if (targetType != null) { bean = applyBeanPostProcessorsBeforeInstantiation(targetType, beanName); if (bean != null){ bean = applyBeanPostProcessorsAfterInitialization(bean, beanName); } } } mbd.beforeInstantiationResolved = (bean != null); } return bean;}

執(zhí)行后置處理器的前置方法如下所示:

MyBatis Plus插件機(jī)制與執(zhí)行流程原理分析詳解

③ doCreateBean

AbstractAutowireCapableBeanFactory.doCreateBean是創(chuàng)建bean的核心方法,這會為bean屬性賦值并會觸發(fā)bean后置處理器、InitializingBean以及自定init方法等。

protected Object doCreateBean(final String beanName, final RootBeanDefinition mbd, final Object[] args) throws BeanCreationException {// Instantiate the bean.--實例化beanBeanWrapper instanceWrapper = null;if (mbd.isSingleton()) { instanceWrapper = this.factoryBeanInstanceCache.remove(beanName);}if (instanceWrapper == null) {//獲取bean的包裝對象-這里很重要 instanceWrapper = createBeanInstance(beanName, mbd, args);}final Object bean = (instanceWrapper != null ? instanceWrapper.getWrappedInstance() : null);Class<?> beanType = (instanceWrapper != null ? instanceWrapper.getWrappedClass() : null);mbd.resolvedTargetType = beanType;// Allow post-processors to modify the merged bean definition.synchronized (mbd.postProcessingLock) { if (!mbd.postProcessed) { try {//調(diào)用MergedBeanDefinitionPostProcessors的postProcessMergedBeanDefinition方法 applyMergedBeanDefinitionPostProcessors(mbd, beanType, beanName); } catch (Throwable ex) { throw new BeanCreationException(mbd.getResourceDescription(), beanName, 'Post-processing of merged bean definition failed', ex); } mbd.postProcessed = true; }}//...//這里暫時忽略其他代碼// Initialize the bean instance.--初始化bean實例Object exposedObject = bean;try {//如下方法很重要 populateBean(beanName, mbd, instanceWrapper); if (exposedObject != null) { exposedObject = initializeBean(beanName, exposedObject, mbd); }}//...

④ populateBean

顧名思義,為bean實例屬性賦值。

AbstractAutowireCapableBeanFactory.populateBean

protected void populateBean(String beanName, RootBeanDefinition mbd, BeanWrapper bw) {//獲取屬性值列表PropertyValues pvs = mbd.getPropertyValues();//...該種符號表示暫時忽略其他代碼//在為bean屬性賦值前,給InstantiationAwareBeanPostProcessors 機(jī)會修改bean的狀態(tài)//應(yīng)用場景如支持字段注入boolean continueWithPropertyPopulation = true;//循環(huán)調(diào)用InstantiationAwareBeanPostProcessors 的postProcessAfterInstantiation方法if (!mbd.isSynthetic() && hasInstantiationAwareBeanPostProcessors()) { for (BeanPostProcessor bp : getBeanPostProcessors()) { if (bp instanceof InstantiationAwareBeanPostProcessor) { InstantiationAwareBeanPostProcessor ibp = (InstantiationAwareBeanPostProcessor) bp; if (!ibp.postProcessAfterInstantiation(bw.getWrappedInstance(), beanName)) { continueWithPropertyPopulation = false; break; } } }}if (!continueWithPropertyPopulation) { return;}//解析autowire注解字段,進(jìn)行主動注入if (mbd.getResolvedAutowireMode() == RootBeanDefinition.AUTOWIRE_BY_NAME || mbd.getResolvedAutowireMode() == RootBeanDefinition.AUTOWIRE_BY_TYPE) { MutablePropertyValues newPvs = new MutablePropertyValues(pvs); // Add property values based on autowire by name if applicable. if (mbd.getResolvedAutowireMode() == RootBeanDefinition.AUTOWIRE_BY_NAME) { autowireByName(beanName, mbd, bw, newPvs); } // Add property values based on autowire by type if applicable. if (mbd.getResolvedAutowireMode() == RootBeanDefinition.AUTOWIRE_BY_TYPE) { autowireByType(beanName, mbd, bw, newPvs); } pvs = newPvs;}boolean hasInstAwareBpps = hasInstantiationAwareBeanPostProcessors();boolean needsDepCheck = (mbd.getDependencyCheck() != RootBeanDefinition.DEPENDENCY_CHECK_NONE);//循環(huán)調(diào)用InstantiationAwareBeanPostProcessors 的postProcessPropertyValues方法if (hasInstAwareBpps || needsDepCheck) { PropertyDescriptor[] filteredPds = filterPropertyDescriptorsForDependencyCheck(bw, mbd.allowCaching); if (hasInstAwareBpps) { for (BeanPostProcessor bp : getBeanPostProcessors()) { if (bp instanceof InstantiationAwareBeanPostProcessor) { InstantiationAwareBeanPostProcessor ibp = (InstantiationAwareBeanPostProcessor) bp; pvs = ibp.postProcessPropertyValues(pvs, filteredPds, bw.getWrappedInstance(), beanName); if (pvs == null) { return; } } } } if (needsDepCheck) { checkDependencies(beanName, mbd, filteredPds, pvs); }}//在這里為屬性賦值,會進(jìn)行類型轉(zhuǎn)換,這里注意關(guān)鍵詞deep copy//如果是引用類型且bean沒有存在,則會進(jìn)行bean的創(chuàng)建過程applyPropertyValues(beanName, mbd, bw, pvs);}

如下圖所示在創(chuàng)建sqlSessionFactoryBean過程中會創(chuàng)建其屬性globalConfiguration對象:

MyBatis Plus插件機(jī)制與執(zhí)行流程原理分析詳解

如下圖所示在創(chuàng)建sqlSessionFactoryBean過程中(從左側(cè)的方法順序就可以看出來)會創(chuàng)建其屬性PaginationInterceptor對象:

MyBatis Plus插件機(jī)制與執(zhí)行流程原理分析詳解

如下圖所示在創(chuàng)建sqlSessionFactoryBean過程中(從左側(cè)的方法順序就可以看出來)會創(chuàng)建其屬性SqlExplainInterceptor對象:

MyBatis Plus插件機(jī)制與執(zhí)行流程原理分析詳解

如下圖所示在創(chuàng)建sqlSessionFactoryBean過程中(從左側(cè)的方法順序就可以看出來)會創(chuàng)建其屬性PerformanceInterceptor對象:

MyBatis Plus插件機(jī)制與執(zhí)行流程原理分析詳解

如下圖所示在創(chuàng)建sqlSessionFactoryBean過程中(從左側(cè)的方法順序就可以看出來)會創(chuàng)建其屬性O(shè)ptimisticLockerInterceptor對象:

MyBatis Plus插件機(jī)制與執(zhí)行流程原理分析詳解

⑤ initializeBean

AbstractAutowireCapableBeanFactory.initializeBean源碼如下:

protected Object initializeBean(final String beanName, final Object bean, RootBeanDefinition mbd){ if (System.getSecurityManager() != null) { AccessController.doPrivileged(new PrivilegedAction<Object>() { @Override public Object run() { invokeAwareMethods(beanName, bean); return null; } }, getAccessControlContext()); } else { //調(diào)用意識/通知方法 invokeAwareMethods(beanName, bean); } Object wrappedBean = bean; if (mbd == null || !mbd.isSynthetic()) { //調(diào)用bean后置處理器的前置方法 wrappedBean = applyBeanPostProcessorsBeforeInitialization(wrappedBean, beanName); } //調(diào)用初始化方法 try { invokeInitMethods(beanName, wrappedBean, mbd); } catch (Throwable ex) { throw new BeanCreationException( (mbd != null ? mbd.getResourceDescription() : null), beanName, 'Invocation of init method failed', ex); }

AbstractAutowireCapableBeanFactory.invokeInitMethods方法源碼如下:

protected void invokeInitMethods(String beanName, final Object bean, RootBeanDefinition mbd) throws Throwable { boolean isInitializingBean = (bean instanceof InitializingBean); if (isInitializingBean && (mbd == null || !mbd.isExternallyManagedInitMethod('afterPropertiesSet'))) { if (logger.isDebugEnabled()) { logger.debug('Invoking afterPropertiesSet() on bean with name ’' + beanName + '’'); } //調(diào)用InitializingBean.afterPropertiesSet if (System.getSecurityManager() != null) { try { AccessController.doPrivileged(new PrivilegedExceptionAction<Object>() { @Override public Object run() throws Exception { ((InitializingBean) bean).afterPropertiesSet(); return null; } }, getAccessControlContext()); } catch (PrivilegedActionException pae) { throw pae.getException(); } } else { ((InitializingBean) bean).afterPropertiesSet(); } }//調(diào)用自定義初始化方法 if (mbd != null) { String initMethodName = mbd.getInitMethodName(); if (initMethodName != null && !(isInitializingBean && 'afterPropertiesSet'.equals(initMethodName)) && !mbd.isExternallyManagedInitMethod(initMethodName)) { invokeCustomInitMethod(beanName, bean, mbd); } }}

如下圖所示,MybatisSqlSessionFactoryBean同樣實現(xiàn)了InitializingBean接口。那么我們就需要注意其afterPropertiesSet方法了。

MyBatis Plus插件機(jī)制與執(zhí)行流程原理分析詳解

⑥ MybatisSqlSessionFactoryBean.afterPropertiesSet

如下所示,代碼很簡短只是創(chuàng)建了sqlSessionFactory。

@Overridepublic void afterPropertiesSet() throws Exception{ notNull(dataSource, 'Property ’dataSource’ is required'); //sqlSessionFactoryBuilder在populateBean的applyPropertyValues過程中已經(jīng)存在! notNull(sqlSessionFactoryBuilder, 'Property ’sqlSessionFactoryBuilder’ is required'); state((configuration == null && configLocation == null) || !(configuration != null && configLocation != null), 'Property ’configuration’ and ’configLocation’ can not specified with together'); this.sqlSessionFactory = buildSqlSessionFactory(); }

進(jìn)入afterPropertiesSet()方法前MybatisSqlSessionFactoryBean如下所示:

MyBatis Plus插件機(jī)制與執(zhí)行流程原理分析詳解

我們看一下sqlSessionFactory,這是一段很長的過程:

protected SqlSessionFactory buildSqlSessionFactory() throws Exception{ Configuration configuration; // TODO 加載自定義 MybatisXmlConfigBuilder MybatisXMLConfigBuilder xmlConfigBuilder = null; if (this.configuration != null) { configuration = this.configuration; if (configuration.getVariables() == null) { configuration.setVariables(this.configurationProperties); } else if (this.configurationProperties != null) { configuration.getVariables().putAll(this.configurationProperties); } } else if (this.configLocation != null) { //通常如果配置了configLocation會從這里創(chuàng)建MybatisXMLConfigBuilder, //其構(gòu)造方法又創(chuàng)建了MybatisConfiguration xmlConfigBuilder = new MybatisXMLConfigBuilder(this.configLocation.getInputStream(), null, this.configurationProperties); configuration = xmlConfigBuilder.getConfiguration(); } else { if (LOGGER.isDebugEnabled()) { LOGGER.debug('Property ’configuration’ or ’configLocation’ not specified, using default MyBatis Configuration'); } // TODO 使用自定義配置 configuration = new MybatisConfiguration(); if (this.configurationProperties != null) { configuration.setVariables(this.configurationProperties); } } if (this.objectFactory != null) { configuration.setObjectFactory(this.objectFactory); } if (this.objectWrapperFactory != null) { configuration.setObjectWrapperFactory(this.objectWrapperFactory); } if (this.vfs != null) { configuration.setVfsImpl(this.vfs); } if (hasLength(this.typeAliasesPackage)) { // TODO 支持自定義通配符 String[] typeAliasPackageArray; if (typeAliasesPackage.contains('*') && !typeAliasesPackage.contains(',') && !typeAliasesPackage.contains(';')) { typeAliasPackageArray = PackageHelper.convertTypeAliasesPackage(typeAliasesPackage); } else { typeAliasPackageArray = tokenizeToStringArray(this.typeAliasesPackage, ConfigurableApplicationContext.CONFIG_LOCATION_DELIMITERS); } if (typeAliasPackageArray == null) { throw new MybatisPlusException('not find typeAliasesPackage:' + typeAliasesPackage); } for (String packageToScan : typeAliasPackageArray) { configuration.getTypeAliasRegistry().registerAliases(packageToScan, typeAliasesSuperType == null ? Object.class : typeAliasesSuperType); if (LOGGER.isDebugEnabled()) { LOGGER.debug('Scanned package: ’' + packageToScan + '’ for aliases'); } } } // TODO 自定義枚舉類掃描處理 if (hasLength(this.typeEnumsPackage)) { Set<Class> classes = null; if (typeEnumsPackage.contains('*') && !typeEnumsPackage.contains(',') && !typeEnumsPackage.contains(';')) { classes = PackageHelper.scanTypePackage(typeEnumsPackage); } else { String[] typeEnumsPackageArray = tokenizeToStringArray(this.typeEnumsPackage, ConfigurableApplicationContext.CONFIG_LOCATION_DELIMITERS); if (typeEnumsPackageArray == null) { throw new MybatisPlusException('not find typeEnumsPackage:' + typeEnumsPackage); } classes = new HashSet<Class>(); for (String typePackage : typeEnumsPackageArray) { classes.addAll(PackageHelper.scanTypePackage(typePackage)); } } // 取得類型轉(zhuǎn)換注冊器 TypeHandlerRegistry typeHandlerRegistry = configuration.getTypeHandlerRegistry(); for (Class cls : classes) { if (cls.isEnum()) { if (IEnum.class.isAssignableFrom(cls)) { typeHandlerRegistry.register(cls.getName(), com.baomidou.mybatisplus.handlers.EnumTypeHandler.class.getCanonicalName()); } else { // 使用原生 EnumOrdinalTypeHandler typeHandlerRegistry.register(cls.getName(), org.apache.ibatis.type.EnumOrdinalTypeHandler.class.getCanonicalName()); } } } } if (!isEmpty(this.typeAliases)) { for (Class<?> typeAlias : this.typeAliases) { configuration.getTypeAliasRegistry().registerAlias(typeAlias); if (LOGGER.isDebugEnabled()) { LOGGER.debug('Registered type alias: ’' + typeAlias + '’'); } } } if (!isEmpty(this.plugins)) { for (Interceptor plugin : this.plugins) { configuration.addInterceptor(plugin); if (LOGGER.isDebugEnabled()) { LOGGER.debug('Registered plugin: ’' + plugin + '’'); } } } if (hasLength(this.typeHandlersPackage)) { String[] typeHandlersPackageArray = tokenizeToStringArray(this.typeHandlersPackage, ConfigurableApplicationContext.CONFIG_LOCATION_DELIMITERS); for (String packageToScan : typeHandlersPackageArray){ configuration.getTypeHandlerRegistry().register(packageToScan); if (LOGGER.isDebugEnabled()) { LOGGER.debug('Scanned package: ’' + packageToScan + '’ for type handlers'); } } } if (!isEmpty(this.typeHandlers)) { for (TypeHandler<?> typeHandler : this.typeHandlers) { configuration.getTypeHandlerRegistry().register(typeHandler); if (LOGGER.isDebugEnabled()) { LOGGER.debug('Registered type handler: ’' + typeHandler + '’'); } } } if (this.databaseIdProvider != null) {//fix #64 set databaseId before parse mapper xmls try { configuration.setDatabaseId(this.databaseIdProvider.getDatabaseId(this.dataSource)); } catch (SQLException e) { throw new NestedIOException('Failed getting a databaseId', e); } } if (this.cache != null) { configuration.addCache(this.cache); } if (xmlConfigBuilder != null) { try { xmlConfigBuilder.parse(); if (LOGGER.isDebugEnabled()) { LOGGER.debug('Parsed configuration file: ’' + this.configLocation + '’'); } } catch (Exception ex) { throw new NestedIOException('Failed to parse config resource: ' + this.configLocation, ex); } finally { ErrorContext.instance().reset(); } } if (this.transactionFactory == null) { this.transactionFactory = new SpringManagedTransactionFactory(); } configuration.setEnvironment(new Environment(this.environment, this.transactionFactory, this.dataSource)); // 設(shè)置元數(shù)據(jù)相關(guān) GlobalConfigUtils.setMetaData(dataSource, globalConfig); SqlSessionFactory sqlSessionFactory = this.sqlSessionFactoryBuilder.build(configuration); // TODO SqlRunner SqlRunner.FACTORY = sqlSessionFactory; // TODO 緩存 sqlSessionFactory globalConfig.setSqlSessionFactory(sqlSessionFactory); // TODO 設(shè)置全局參數(shù)屬性 globalConfig.signGlobalConfig(sqlSessionFactory); if (!isEmpty(this.mapperLocations)) { if (globalConfig.isRefresh()) { //TODO 設(shè)置自動刷新配置 減少配置 new MybatisMapperRefresh(this.mapperLocations, sqlSessionFactory, 2, 2, true); } for (Resource mapperLocation : this.mapperLocations) { if (mapperLocation == null) { continue; } try { // TODO 這里也換了噢噢噢噢 XMLMapperBuilder xmlMapperBuilder = new XMLMapperBuilder(mapperLocation.getInputStream(), configuration, mapperLocation.toString(), configuration.getSqlFragments()); xmlMapperBuilder.parse(); } catch (Exception e) { throw new NestedIOException('Failed to parse mapping resource: ’' + mapperLocation + '’', e); } finally { ErrorContext.instance().reset(); } if (LOGGER.isDebugEnabled()) { LOGGER.debug('Parsed mapper file: ’' + mapperLocation + '’'); } } } else { if (LOGGER.isDebugEnabled()) { LOGGER.debug('Property ’mapperLocations’ was not specified or no matching resources found'); } } return sqlSessionFactory; }

上面代碼主要做了如下事情:

獲取MybatisXMLConfigBuilder對象 獲取Configuration對象-MybatisConfiguration 配置對象Configuration添加插件configuration.addInterceptor(plugin); xmlConfigBuilder.parse()對configuration做進(jìn)一步處理 獲取SpringManagedTransactionFactory用來創(chuàng)建SpringManagedTransaction 獲取一個DefaultSqlSessionFactory實例對象 globalConfig.setSqlSessionFactory(sqlSessionFactory)中會創(chuàng)建MybatisSqlSessionTemplate 解析mapperLocation對應(yīng)的一個個mapper配置文件,使用助手builderAssistant的addMappedStatement方法將一個個結(jié)點添加配置對象中 其他屬性設(shè)置等等

也就是說,在MybatisSqlSessionFactoryBean.afterPropertiesSet方法執(zhí)行結(jié)束后,SqlSessionFactory、SqlSessionTemplate、Configuration等都已存在!

【3】查詢執(zhí)行流程分析

示例代碼如下:

List<Employee > emps = employeeMapper.selectPage(page, null);

如下圖所示,此時我們獲取到的employeeMapper其實是個代理對象:

MyBatis Plus插件機(jī)制與執(zhí)行流程原理分析詳解

MyBatis Plus插件機(jī)制與執(zhí)行流程原理分析詳解

總結(jié)

到此這篇關(guān)于MyBatis Plus插件機(jī)制與執(zhí)行流程原理分析的文章就介紹到這了,更多相關(guān)MyBatis Plus插件機(jī)制內(nèi)容請搜索好吧啦網(wǎng)以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持好吧啦網(wǎng)!

標(biāo)簽: Mybatis 數(shù)據(jù)庫
相關(guān)文章:
主站蜘蛛池模板: 飞行者联盟-飞机模拟机_无人机_低空经济_航空技术交流平台 | 东莞ERP软件_广州云ERP_中山ERP_台湾工厂erp系统-广东顺景软件科技有限公司 | 自动记录数据电子台秤,记忆储存重量电子桌称,设定时间记录电子秤-昆山巨天 | 广州番禺搬家公司_天河黄埔搬家公司_企业工厂搬迁_日式搬家_广州搬家公司_厚道搬迁搬家公司 | 传动滚筒_厂家-淄博海恒机械制造厂| 佛山商标注册_商标注册代理|专利注册申请_商标注册公司_鸿邦知识产权 | 天坛家具官网| 超声骨密度仪-动脉硬化检测仪器-人体成分分析仪厂家/品牌/价格_南京科力悦 | 汽车水泵_汽车水泵厂家-瑞安市骏迪汽车配件有限公司 | HEYL硬度计量泵-荧光法在线溶解氧仪-净时测控技术(上海)有限公司 | 聚合氯化铝厂家-聚合氯化铝铁价格-河南洁康环保科技 | 异噻唑啉酮-均三嗪-三丹油-1227-中北杀菌剂厂家 | 不锈钢复合板|钛复合板|金属复合板|南钢集团安徽金元素复合材料有限公司-官网 | 篷房[仓储-婚庆-展览-活动]生产厂家-江苏正德装配式帐篷有限公司 | 恒湿机_除湿加湿一体机_恒湿净化消毒一体机厂家-杭州英腾电器有限公司 | 数显恒温油浴-电砂浴-高温油浴振荡器-常州迈科诺仪器有限公司 | 换链神器官网-友情链接交换、购买交易于一体的站长平台 | 医院专用门厂家报价-医用病房门尺寸大全-抗菌木门品牌推荐 | 全自动真空上料机_粉末真空上料机_气动真空上料机-南京奥威环保科技设备有限公司 | 南京兰江泵业有限公司-水解酸化池潜水搅拌机-絮凝反应池搅拌机-好氧区潜水推进器 | 闭端端子|弹簧螺式接线头|防水接线头|插线式接线头|端子台|电源线扣+护线套|印刷电路板型端子台|金笔电子代理商-上海拓胜电气有限公司 | 不锈钢监控杆_监控立杆厂家-廊坊耀星光电科技有限公司 | 澳门精准正版免费大全,2025新澳门全年免费,新澳天天开奖免费资料大全最新,新澳2025今晚开奖资料,新澳马今天最快最新图库 | 2025福建平潭岛旅游攻略|蓝眼泪,景点,住宿攻略-趣平潭网 | 绿萝净除甲醛|深圳除甲醛公司|测甲醛怎么收费|培训机构|电影院|办公室|车内|室内除甲醛案例|原理|方法|价格立马咨询 | 南京展台搭建-南京展会设计-南京展览设计公司-南京展厅展示设计-南京汇雅展览工程有限公司 | 贴板式电磁阀-不锈钢-气动上展式放料阀-上海弗雷西阀门有限公司 工业机械三维动画制作 环保设备原理三维演示动画 自动化装配产线三维动画制作公司-南京燃动数字 | 焊缝跟踪系统_激光位移传感器_激光焊缝跟踪传感器-创想智控 | 液压升降平台_剪叉式液压/导轨式升降机_传菜机定做「宁波日腾升降机厂家」 | 隧道窑炉,隧道窑炉厂家-山东艾瑶国际贸易| 编织人生 - 权威手工编织网站,编织爱好者学习毛衣编织的门户网站,织毛衣就上编织人生网-编织人生 | 莱州网络公司|莱州网站建设|莱州网站优化|莱州阿里巴巴-莱州唯佳网络科技有限公司 | 发电机价格|发电机组价格|柴油发电机价格|柴油发电机组价格网 | 焊接减速机箱体,减速机箱体加工-淄博博山泽坤机械厂 | 幂简集成 - 品种超全的API接口平台, 一站搜索、试用、集成国内外API接口 | 24位ADC|8位MCU-芯易德科技有限公司 | SDG吸附剂,SDG酸气吸附剂,干式酸性气体吸收剂生产厂家,超过20年生产使用经验。 - 富莱尔环保设备公司(原名天津市武清县环保设备厂) | 桁架机器人_桁架机械手_上下料机械手_数控车床机械手-苏州清智科技装备制造有限公司 | 交通气象站_能见度检测仪_路面状况监测站- 天合环境科技 | 自动售货机_无人售货机_专业的自动售货机运营商_免费投放售货机-广州富宏主官网 | 臻知网大型互动问答社区-你的问题将在这里得到解答!-无锡据风网络科技有限公司 |