1.2万多行MyBatis源码,码好你知道里面用了多少种设计模式吗?
2.拿到美团、码好快手offer后收到阿里三面通知,码好竟然被mybatis挡住了去路
3.Mybatis:PageHelper分页插件源码及原理剖析
4.三万字带你彻底吃透MyBatis源码!码好!码好
5.MyBatis 码好tiny dnn 源码分析源码解析:映射文件的加载与解析(上)
6.mybatis插件机制源码解析
2万多行MyBatis源码,你知道里面用了多少种设计模式吗?
在MyBatis的码好两万多行的框架源码中,设计模式的码好巧妙使用是整个框架的精华。
MyBatis中主要使用了以下设计模式:工厂模式、码好单例模式、码好建造者模式、码好适配器模式、码好代理模式、码好组合模式、码好装饰器模式、码好模板模式、策略模式和迭代器模式。
具体来说,工厂模式用于SqlSessionFactory的创建,单例模式用于Configuration的管理,建造者模式用于ResultMap的构建,适配器模式用于统一日志接口,代理模式用于MapperProxy的实现,组合模式用于SQL标签的组合,装饰器模式用于二级缓存操作,模板模式用于定义SQL执行流程,策略模式用于多类型处理器的实现,迭代器模式用于字段解析的实现。
通过运用这些设计模式,MyBatis成功地实现了复杂场景的解耦,并将问题合理切割为若干子问题,以提高理解和解决的效率。
总的来说,MyBatis大约运用了种左右的设计模式,这使得框架在处理复杂问题时能够更加高效和灵活。
学习源码不仅可以帮助我们更好地理解设计模式和设计原则,更能够扩展我们的编码思维,积累实际应用的经验。
希望本文的显示驱动源码解读分享能够帮助到您,同时也推荐您阅读《手写MyBatis:渐进式源码实践》一书,了解更多关于MyBatis的知识。
拿到美团、快手offer后收到阿里三面通知,竟然被mybatis挡住了去路
一位求职者在年底之际收获了美团、快手等公司的录用通知,面对阿里三面的邀请,他抱有期待。然而,面试过程却出乎意料地受挫,主要原因是面试官针对MyBatis源码提出了深入的问题。其中包括设计模式的应用、调试模式的实现机制、数据库连接池操作、二级缓存功能的实现以及源码中涉及的术语如“缓存雪崩”。
面试官热衷于考察源码理解,因为这不仅能测试技术深度,还能展现开发者的思维方式和问题解决能力。正如淘宝创始人团队成员多隆,他凭借对源码的深入研究,不仅提升了工作效率,还帮助同事解决难题。掌握源码对于程序员意味着编写高效代码的经验、微观架构设计的提升、工作中的疑难杂症解决,以及学习大牛的思维模式。
然而,学习MyBatis源码并非易事,市面上的教材要么难以理解,要么过于专业。但好消息是,有一份详尽的学习指南,包含了学习文档、视频讲解和思维导图,从入门到精通,涵盖了高级用法和设计模式。对于有需要的朋友,可以通过此资源进行高效学习,性能验证指标源码点击此处获取:( 点这里)免费领取,我们非常诚挚地邀请您的支持!
此外,对于Java技术、面试问题和架构实战内容,我们也有更多精选资源:Java面试题和实战文档,欢迎感兴趣的朋友点击左下角支持我们。
Mybatis:PageHelper分页插件源码及原理剖析
PageHelper是一款强大的Mybatis分页插件,以其开源和免费的特性受到赞誉。其功能复杂性远超初印象,实现了物理分页的强大与彻底。核心在于保持分页插件的基本功能,同时提供智能参数以适应复杂场景。
基本使用与配置方面,PageHelper依赖于jsqlparser包,Maven会自动引入。关键参数dialect = mysql,与oracle等其他数据库对应参数可根据需要选择使用或避免使用,保持分页插件的简洁性。
源码分析中,PageHelper包含SqlUtil、SqlUtilConfig等类,负责数据库类型专用SQL工具的管理与执行,如count查询、分页查询等。自动检测当前数据库方言的机制和缓存策略提高了插件的灵活性与效率。
PageSqlSource类展示了如何解析SQL并添加分页参数,确保分页查询的正确执行。Parser类则负责解析SQL并添加所需分页参数,实现分页功能的关键步骤。
执行分页查询的`SqlUtil.doProcessPage()`方法,展示了创建count查询、执行分页查询、修改参数列表等关键步骤,确保分页查询的高效执行。
PageHelper提供了两种使用方式:直接通过RowBounds参数进行分页查询和使用静态方法`PageHelper.startPage()`。其原理在于使用ThreadLocal传递和保存Page对象,每次查询时需单独设置,灰度图纸源码大全以保持分页功能的灵活性。
最佳使用建议包括明确指定方言以及编写SQL分页业务与对应的count查询,避免简化操作以提高性能。推荐进一步阅读相关聚合内容,如SpringBoot、设计模式、Mybatis、多线程等,以深入理解其应用与优化。
三万字带你彻底吃透MyBatis源码!!
随着互联网的迅猛发展,MyBatis逐渐成为了Java开发者不可或缺的框架技术。许多大厂在面试中偏好问及MyBatis的底层原理及源码实现,这表明了其在技术栈中的重要性。本文旨在全面解析MyBatis源码,帮助开发者深入理解这一强大的框架。为了方便学习,推荐大家先收藏后仔细研读。
MyBatis源码在封装了JDBC之后,实现了对数据库操作的高级抽象。无论是获取连接、预编译语句、参数封装还是执行SQL,其核心步骤并未改变。
解析过程始于通过`ClassLoader.getResourceAsStream`方法获取配置文件路径。这个过程确保了MyBatis能正确加载配置信息,进而解析XML文件,构建配置中心。
解析XML文件的关键在于`parseConfiguration`和`mapperParser.parse`方法。前者用于解析配置文件中的`Environment`、`Setting`等信息,后者则专注于解析Mapper映射器,将其与工厂类进行绑定。
构建`SqlSessionFactory`的过程涉及解析Mapper映射器,生成`MappedStatement`对象,以及将接口类型与工厂类绑定。最终,jdk源码是否开放`DefaultSqlSessionFactory`被创建,用于管理会话生命周期。
会话的创建通过`openSession`方法完成,该方法实例化了`Executor`来执行SQL。`Executor`的配置则决定了事务管理和执行器类型。同时,`Transaction`的管理分为两种方式,以确保数据的一致性和完整性。
获取Mapper对象时,通过`mapperRegistry.getMapper`方法,该方法从`MapperRegistry`的`knownMappers`中获取接口类型和对应的工厂类。代理对象`MapperProxy`由JDK动态代理生成,用于执行实际的数据库操作。
执行SQL时,调用代理对象的`invoke`方法,进而调用`execute`方法。无论是查询还是其他操作,均遵循此流程。在查询场景下,`selectOne`与`selectList`功能实现相同,仅在参数处理上有所差异。
`MappedStatement`对象负责存储SQL信息,包括执行策略、参数类型等。`CacheKey`的生成则基于`BoundSql`内容,用于缓存结果,提高效率。
通过以上解析,我们可以看到MyBatis源码的简洁与高效。深入理解其结构与机制,不仅有助于提高开发效率,还能增强对数据库操作的理解。总的来说,MyBatis的源码并不复杂,只需耐心研读,两三天内即可掌握其核心。
MyBatis 源码解析:映射文件的加载与解析(上)
MyBatis 的映射文件是其核心组成部分,用于配置 SQL 语句、二级缓存及结果集映射等功能,是其区别于其他 ORM 框架的重要特色。 在解析映射文件时,MyBatis 通过调用 XMLMapperBuilder#parse 方法实现加载与解析操作。此方法首先判断映射文件是否已解析,若未解析则调用 XMLMapperBuilder#configurationElement 方法解析所有配置,并注册当前映射文件关联的 Mapper 接口。对于处理异常的标签,MyBatis 会记录至 Configuration 对象并尝试二次解析。 解析流程主要涉及以下几个关键步骤:缓存配置(cache 标签):MyBatis 采用缓存设计,分为一级缓存和二级缓存。解析 cache 标签时,首先获取相关属性配置,然后使用 CacheBuilder 创建缓存对象,并记录到 Configuration 对象。
缓存引用(cache-ref 标签):标签默认限定在 namespace 范围内,用于引用其它命名空间中的缓存对象。解析过程中记录引用关系,然后从 Configuration 中获取引用的缓存对象。
结果集映射(resultMap 标签):解析 resultMap 标签配置,构建 ResultMap 对象,并将其记录到 Configuration 中。
SQL 语句(sql 标签):通过 sql 标签配置复用的 SQL 语句片段,解析后记录至 Configuration 的 sqlFragments 属性中。
核心数据库操作(select / insert / update / delete 标签):解析这些标签时,构建 MappedStatement 对象并记录到 Configuration 中。
每个标签解析实现由 MyBatis 提供的多个方法执行,如 XMLMapperBuilder 的 configurationElement 方法和解析具体标签的子方法,如 cacheElement、sqlElement 等。解析过程中,MyBatis 会调用不同的构造器和工厂方法来创建、初始化和配置相应的对象。 在解析完成之后,MyBatis 将所有配置对象封装在 Configuration 对象中,该对象包含所有映射文件中定义的配置信息,供后续的 SQL 语句执行和映射操作使用。mybatis插件机制源码解析
引言
本篇源码解析基于MyBatis3.5.8版本。
首先需要说明的是,本篇文章不是mybatis插件开发的教程,而是从源码层面分析mybatis是如何支持用户自定义插件开发的。
mybatis的插件机制,让其扩展能力大大增加。比如我们项目中经常用到的PageHelper,这就是一款基于mybatis插件能力开发的产品,它的功能是让基于mybatis的数据库分页查询更容易使用。
当然基于插件我们还可以开发其它功能,比如在执行sql前打印日志、做权限控制等。
正文mybatis插件也叫mybatis拦截器,它支持从方法级别对mybatis进行拦截。整体架构图如下:
解释下几个相关概念:
Interceptor拦截器接口,用户自定义的拦截器就是实现该接口。
InterceptorChain拦截器链,其内部维护一个interceptorslist,表示拦截器链中所有的拦截器,并提供增加或获取拦截器链的方法。比如有个核心的方法是pluginAll。该方法用来生成代理对象。
Invocation拦截器执行时的上下文环境,其实就是目标方法的调用信息,包含目标对象、调用的方法信息、参数信息。核心方法是proceed。该方法的主要目的就是进行处理链的传播,执行完拦截器的方法后,最终需要调用目标方法的invoke方法。
mybatis支持在哪些地方进行拦截呢?你只需要在代码里搜索interceptorChain.pluginAll的使用位置就可以获取答案,一共有四处:
parameterHandler=(ParameterHandler)interceptorChain.pluginAll(parameterHandler);resultSetHandler=(ResultSetHandler)interceptorChain.pluginAll(resultSetHandler);statementHandler=(StatementHandler)interceptorChain.pluginAll(statementHandler);executor=(Executor)interceptorChain.pluginAll(executor);这四处实现的原理都是一样的,我们只需要选择一个进行分析就可以了。
我们先来看下自定义的插件是如何加载进来的,比如我们使用PageHelper插件,通常会在mybatis-config.xml中加入如下的配置:
<plugins><plugininterceptor="com.github.pagehelper.PageInterceptor"><!--configparamsasthefollowing--><propertyname="param1"value="value1"/></plugin></plugins>mybatis在创建SqlSessionFactory的时候会加载配置文件,
publicConfigurationparse(){ if(parsed){ thrownewBuilderException("EachXMLConfigBuildercanonlybeusedonce.");}parsed=true;parseConfiguration(parser.evalNode("/configuration"));returnconfiguration;}parseConfiguration方法会加载包括plugins在内的很多配置,
privatevoidparseConfiguration(XNoderoot){ try{ ...pluginElement(root.evalNode("plugins"));...}catch(Exceptione){ thrownewBuilderException("ErrorparsingSQLMapperConfiguration.Cause:"+e,e);}}privatevoidpluginElement(XNodeparent)throwsException{ if(parent!=null){ for(XNodechild:parent.getChildren()){ Stringinterceptor=child.getStringAttribute("interceptor");Propertiesproperties=child.getChildrenAsProperties();InterceptorinterceptorInstance=(Interceptor)resolveClass(interceptor).getDeclaredConstructor().newInstance();interceptorInstance.setProperties(properties);configuration.addInterceptor(interceptorInstance);}}}pluginElement干了几件事情:
创建Interceptor实例
设置实例的属性变量
添加到Configuration的interceptorChain拦截器链中
mybatis的插件是通过动态代理实现的,那肯定要生成代理对象,生成的逻辑就是前面提到的pluginAll方法,比如对于Executor生成代理对象就是,
executor=(Executor)interceptorChain.pluginAll(executor);接着看pluginAll方法,
/***该方法会遍历用户定义的插件实现类(Interceptor),并调用Interceptor的plugin方法,对target进行插件化处理,*即我们在实现自定义的Interceptor方法时,在plugin中需要根据自己的逻辑,对目标对象进行包装(代理),创建代理对象,*那我们就可以在该方法中使用Plugin#wrap来创建代理类。*/publicObjectpluginAll(Objecttarget){ for(Interceptorinterceptor:interceptors){ target=interceptor.plugin(target);}returntarget;}这里遍历所有我们定义的拦截器,调用拦截器的plugin方法生成代理对象。有人可能有疑问:如果有多个拦截器,target不是被覆盖了吗?
其实不会,所以如果有多个拦截器的话,生成的代理对象会被另一个代理对象代理,从而形成一个代理链条,执行的时候,依次执行所有拦截器的拦截逻辑代码。
plugin方法是接口Interceptor的默认实现类,
defaultObjectplugin(Objecttarget){ returnPlugin.wrap(target,this);}然后进入org.apache.ibatis.plugin.Plugin#wrap,
publicstaticObjectwrap(Objecttarget,Interceptorinterceptor){ Map<Class<?>,Set<Method>>signatureMap=getSignatureMap(interceptor);Class<?>type=target.getClass();Class<?>[]interfaces=getAllInterfaces(type,signatureMap);if(interfaces.length>0){ returnProxy.newProxyInstance(type.getClassLoader(),interfaces,newPlugin(target,interceptor,signatureMap));}returntarget;}首先是获取我们自己实现的Interceptor的方法签名映射表。然后获取需要代理的对象的Class上声明的所有接口。比如如果我们wrap的是Executor,就是Executor的所有接口。然后就是最关键的一步,用Proxy类创建一个代理对象(newProxyInstance)。
注意,newProxyInstance方法的第三个参数,接收的是一个InvocationHandler对象,表示的是当动态代理对象调用方法的时候会关联到哪一个InvocationHandler对象上,并最终由其调用。
我们这里传入的是Plugin类,故在动态运行过程中会执行Plugin的invoker方法。
如果对这一段不是很理解,建议先了解下java动态代理的原理。java动态代理机制中有两个重要的角色:InvocationHandler(接口)和Proxy(类),这个是背景知识需要掌握的。
我们在深入看下上面的getSignatureMap方法,
privatestaticMap<Class<?>,Set<Method>>getSignatureMap(Interceptorinterceptor){ //从Interceptor的类上获取Intercepts注解,说明我们自定义拦截器需要带注解InterceptsinterceptsAnnotation=interceptor.getClass().getAnnotation(Intercepts.class);//issue#if(interceptsAnnotation==null){ thrownewPluginException("No@Interceptsannotationwasfoundininterceptor"+interceptor.getClass().getName());}Signature[]sigs=interceptsAnnotation.value();Map<Class<?>,Set<Method>>signatureMap=newHashMap<>();//解析Interceptor的values属性(Signature[])数组,存入HashMap,Set<Method>>for(Signaturesig:sigs){ Set<Method>methods=MapUtil.computeIfAbsent(signatureMap,sig.type(),k->newHashSet<>());try{ Methodmethod=sig.type().getMethod(sig.method(),sig.args());methods.add(method);}catch(NoSuchMethodExceptione){ thrownewPluginException("Couldnotfindmethodon"+sig.type()+"named"+sig.method()+".Cause:"+e,e);}}returnsignatureMap;}首先需要从Interceptor的类上获取Intercepts注解,说明我们自定义拦截器需要带注解,比如PageHelper插件的定义如下:
<plugins><plugininterceptor="com.github.pagehelper.PageInterceptor"><!--configparamsasthefollowing--><propertyname="param1"value="value1"/></plugin></plugins>0所以我们可以知道,getSignatureMap其实就是拿到我们自定义拦截器声明需要拦截的类以及类对应的方法。
前面说过,当我们调用代理对象时,最终会执行Plugin类的invoker方法,我们看下Plugin的invoker方法,
<plugins><plugininterceptor="com.github.pagehelper.PageInterceptor"><!--configparamsasthefollowing--><propertyname="param1"value="value1"/></plugin></plugins>1Interceptor接口的intercept方法就是我们自定义拦截器需要实现的逻辑,其参数为Invocation,可从Invocation参数中拿到执行方法的对象,方法,方法参数,比如我们可以从statementHandler拿到SQL语句,实现自己的特殊逻辑。
在该方法的结束需要调用invocation#proceed()方法,进行拦截器链的传播。
参考:
blogs.com/chenpi/p/.html
mybatisçä¼ç¹
ãã1.ä¼ç¹
ããç®åï¼
ããæäºå¦ä¹ ï¼æäºä½¿ç¨ï¼éè¿ææ¡£åæºä»£ç ï¼å¯ä»¥æ¯è¾å®å ¨çææ¡å®ç设计æè·¯åå®ç°ã
ããå®ç¨ï¼
ããæä¾äºæ°æ®æ å°åè½ï¼æä¾äºå¯¹åºå±æ°æ®è®¿é®çå°è£ ï¼ä¾å¦ado.netï¼ï¼æä¾äºDAOæ¡æ¶ï¼å¯ä»¥ä½¿æ们æ´å®¹æçå¼ååé ç½®æ们çDALå±ã
ããçµæ´»ï¼
ããéè¿sqlåºæ¬ä¸å¯ä»¥å®ç°æ们ä¸ä½¿ç¨æ°æ®è®¿é®æ¡æ¶å¯ä»¥å®ç°çææåè½ï¼æ许æ´å¤ã
ããåè½å®æ´ï¼
ããæä¾äºè¿æ¥ç®¡çï¼ç¼åæ¯æï¼çº¿ç¨æ¯æï¼ï¼åå¸å¼ï¼äºç©ç®¡çï¼éè¿é ç½®ä½å ³ç³»å¯¹è±¡æ å°çæ°æ®è®¿é®å±éè¦è§£å³çé®é¢ãæä¾äºDAOæ¯æï¼å¹¶å¨DAOæ¡æ¶ä¸å°è£ äºADO.NETï¼NHibernateåDataMapperã
ããå¢å¼ºç³»ç»çå¯ç»´æ¤æ§ï¼
ããéè¿æä¾DALå±ï¼å°ä¸å¡é»è¾åæ°æ®è®¿é®é»è¾å离ï¼ä½¿ç³»ç»ç设计æ´æ¸ æ°ï¼æ´æç»´æ¤ï¼æ´æåå æµè¯ãsqlå代ç çå离ï¼æé«äºå¯ç»´æ¤æ§ã
ãã2.缺ç¹
ããæ»åæ§ï¼
ããè¿æ²¡ææ确对.NET2.0çæ¯æãææ°çæ¬å¨2.0ä¸ç¼è¯å¯ä»¥ï¼ä½æäºåå æµè¯ä¸è½éè¿ã
ããä¸æçï¼å·¥ç¨å®è·µè¾å°ï¼
ããIbatisNetå¨å®é 项ç®ä¸ç使ç¨è¾å°ã åªæ¯ç论ä¸å¯è¡.
ããåORMï¼å·¥å ·æ¯æè¾å°ï¼
ããéè¦æ们èªå·±åsqlï¼å¹¶ä¸.NETä¸è¿æªåç°å¯ä»¥èªå¨çæä¸å¡å±ç±»åé ç½®æ件çå·¥å ·ï¼è¿ç¹åNHibernateä¸ä¸æ ·ï¼NHibernateä¼ä¸ºæ们çæ°æ®åºç´æ¥äº§çsqlï¼å¹¶æä¸äºè¾ å©å·¥å ·ãå æ¤ä½¿ç¨Ibatisæ¯NHibernateè¦å¤åä¸äºå·¥ä½ã
Mybatis源码剖析(懒加载原理)
懒加载,即按需加载,旨在优化查询性能。以一个包含订单列表的User对象为例,当仅获取用户信息时,若启用懒加载模式,执行SQL不会查询订单列表。需获取订单列表时,才会发起数据库查询。实现方式包括在核心配置文件中设置或在相关映射文件中通过fetchType属性配置懒加载策略。
懒加载的配置如何加载到项目中呢?首先,这些配置保存在全局Configuration对象中,通常在解析核心配置文件的代码中实现。在settingsElement方法中,懒加载配置被保存在lazyLoadingEnabled属性中。对于resultMap标签中collection | association的fetchType属性,其配置通过解析mappers标签下的resultMap标签实现,最终调用buildResultMappingFromContext方法处理子标签。该方法结合全局配置判断是否需要执行懒加载。
懒加载的实现原理涉及动态代理。当调用代理对象的延迟加载属性方法时,如访问a.getB().getName(),代理对象会调用拦截器方法。若发现需要延迟加载,代理对象会单独发送SQL查询关联对象,加载数据后设置属性值,完成方法调用。简而言之,懒加载通过动态代理实现,拦截指定方法并执行数据加载。
深入剖析懒加载源码,会发现它涉及查询和数据处理的多步操作。查询完成后,结果集处理、列值获取、判断是否进行懒加载等步骤共同构建懒加载机制。动态代理在访问对象属性时触发,最终通过Javassist库创建代理对象,实现懒加载逻辑。当访问如userList2.get(0).getOrderList()时,若满足条件,代理对象会调用懒加载查询方法获取数据。判断懒加载条件的关键在于结果集处理阶段,通过访问映射关系和查询映射值来确定是否执行后续懒加载查询。
综上所述,Mybatis的懒加载机制通过动态代理和结果集处理实现,旨在优化性能,按需加载数据,提高查询效率。通过核心配置和映射文件中的配置,懒加载逻辑被加载到项目中,为开发者提供灵活的加载策略。