理解代理模式,MyBatis原理就掌握了一半

298 阅读4分钟

以下文章来源于公众号“程序员求职之路”

在java程序员的世界里,最熟悉的开源软件除了 Spring,Tomcat,还有谁呢?当然是 Mybatis 了。MyBatis 是一个被广泛应用的持久化框架。代理模式可以认为是Mybatis的核心使用的模式。本文通过代理模式,带你掌握MyBatis原理。
​01
**什么是代理模式**

代理模式含义是为其他的对象提供一种代理以难以控制对这个对象的访问,简而言之即如果有一些我们不能做不想做的事情,可以委托别人去做。优点是可以不用修改源代码。

代理模式的角色分类有抽象主题(接口)、具体主题(真实类)和proxy(代理类)。

![](https://mmbiz.qpic.cn/mmbiz_png/ib8uEDL0WGNsZLGscLdgqxqrIrYibwgicIzuib8PmyMtw6xbdAvenJfa4cibaGEic9klxRPFhibBEhqyfffjYpmHqFIiaQ/640?wx_fmt=png "图片1.png")

代理模式的角色分类

02
**静态代理与动态代理**

代理模式又分为静态代理和动态代理,动态代理又有两种,一种是JDK动态代理,一种是Cglib。

![](https://mmbiz.qpic.cn/mmbiz_png/ib8uEDL0WGNsZLGscLdgqxqrIrYibwgicIzWkP1AqobNk2K5WiahicPy1uY9EhwKpA8paaF9Jj8YdsP24aszowY7SfA/640?wx_fmt=png "图片2.png")

静态代理与动态代理

静态代理是必须要有java源文件,通过java编译器转换为.calss文件,通过转换转为byte类加载器再进行加载。

动态代理直接在运行时就生成byte文件直接通过类加载器进行加载。

JDK动态代理要求必须是实现接口那种方式,否则不能进行动态代理。

Cglib支持不是接口的类能进行动态代理,但是Cglib不能代理被final修饰的方法。Springboot 2.x版本spring已经添加了cglib。

如果是实现接口类型的就用jdk动态代理,如果不是接口类型就用Cglib。

03
**为什么需要动态原理?**

原因是代码运行的时自动生成代理类,帮我们做额外我们想做的事情。

动态代理实现方法有JDK Proxy和用JDK里的工具自动生成动态代理类两种。

![](https://mmbiz.qpic.cn/mmbiz_png/ib8uEDL0WGNsZLGscLdgqxqrIrYibwgicIzb73aQUQZG2dfugqFibOf67vJ5sXvs4ciaKeJhfMsUYWdMDiaeOHwXoo6w/640?wx_fmt=png "20200516135030565.png")
![](https://mmbiz.qpic.cn/mmbiz_gif/ib8uEDL0WGNsZLGscLdgqxqrIrYibwgicIz54FHWWTPoRrwZMic9yfAYh947dDLT3LfChdhsNgGEr4buNE6ib05zuRg/640?wx_fmt=gif)

动态代理流程

Client假如就是main方法,doSomething就是调用任何方法它都会生成Proxy类,Proxy类,Proxy类调用invoke方法,走到我们自己定义的任何代理类方法调用invoke开始做增强代码逻辑的操作,然后再调用被代理对象的方法。

04
**JDK动态原理分析**

1首先对我们传进来的接口InvocationHandler全部克隆了一遍(必须实现Invocationhandler的接口)

![](https://mmbiz.qpic.cn/mmbiz_png/ib8uEDL0WGNsZLGscLdgqxqrIrYibwgicIzGRLUM5EhQq0ymoibat41IPmDjqeZSOzXyOQqo6ib78Zc47SDMNCgomZA/640?wx_fmt=png)
2生成一个代理类class对象(get proxy0生成对象)
![](https://mmbiz.qpic.cn/mmbiz_png/ib8uEDL0WGNsZLGscLdgqxqrIrYibwgicIzIsA6sMvhOE1MYIgLX4O37rc8lB48wZL4w64XbonSBKc1rZuxtImRGg/640?wx_fmt=png)

3根据代理类对象cl,获取构造器

![](https://mmbiz.qpic.cn/mmbiz_png/ib8uEDL0WGNsZLGscLdgqxqrIrYibwgicIzpm9hbWde9f6CBJFoh1LBcqmH4FYjRGicScHIk8zPwIPqyaUXr0gNfog/640?wx_fmt=png)
4根据构造器和InvocationHandler实现类,创建代理类实例(使用newInstance)
![](https://mmbiz.qpic.cn/mmbiz_png/ib8uEDL0WGNsZLGscLdgqxqrIrYibwgicIzzXLN5sbSbhvmMCRNMIick15Gibb5W32O9M4lYeDJIEEOVFM5SSfRfYqg/640?wx_fmt=png)
05
**MyBatis动态原理的使用**

mapper语句

这是一段Mybatis的一个查询数据库数据的代码,通过mapper找到配置文件的sql语句,执行sql语句获取数据,但是mapper.selectBlogById()它是没有实现类的,那是怎么实现操作方法的呢?

![](https://mmbiz.qpic.cn/mmbiz_png/ib8uEDL0WGNsZLGscLdgqxqrIrYibwgicIzibqtt5yvPfF12vx4f9OCXBiaIn3YXiaoo5ploLfBmak66lMKbXF1zMO3w/640?wx_fmt=png)

打印一下类信息,发现此时的mapper其实是动态代理出来的类了

![](https://mmbiz.qpic.cn/mmbiz_png/ib8uEDL0WGNsZLGscLdgqxqrIrYibwgicIzJVSmVbiaIC6df4AiakKZ5yVTt0RtTic3zsicwYEZZBfQNnniavPEQric79MA/640?wx_fmt=png)

发现输出的是jdk的动态代理

![](https://mmbiz.qpic.cn/mmbiz_png/ib8uEDL0WGNsZLGscLdgqxqrIrYibwgicIzKicxJtmKmYm4PvBeTy6N4vFL0bXYxCluQmols4jyTb82OAia5K5QVAAg/640?wx_fmt=png)

那MyBatis中的hadler是什么呢,是MapperProxy里的invoke方法,也就是说能够不实现接口就直接调用来运行sql语句是通过了jdk动态代理实现的,sql操作都在mapperMethod里的execute里实现的。

![](https://mmbiz.qpic.cn/mmbiz_png/ib8uEDL0WGNsZLGscLdgqxqrIrYibwgicIzItzFuONpsbUhzzUyKIibXOWxZCnuPwP0wtdgAlicWWBEzjiaPRJdrG9fA/640?wx_fmt=png)

Mybatis插件

Mybatis拦截器、分页插件等一些其他插件都是使用了动态代理,那么mybats里有专门的Plugin代理类, 下图二Plugin类的invoke

![](https://mmbiz.qpic.cn/mmbiz_png/ib8uEDL0WGNsZLGscLdgqxqrIrYibwgicIzrRe4a41nZ1HWolQpJBbh31HsqRm5h6Cib3Xv7rflRg5jz2dceMLAshQ/640?wx_fmt=png)
![](https://mmbiz.qpic.cn/mmbiz_png/ib8uEDL0WGNsZLGscLdgqxqrIrYibwgicIzaux5KjiceNZdPLvzQfJtn10dW3qN5mxTRiadKJuicb6x4ys8jzypVNZ2A/640?wx_fmt=png)

Mybatis的连接池

如果不指定spring管理数据库连接,mybatis也是有连接池的,而池的操作mybatis也是采用动态代理。因为connection的连接自己是不可能把自己还给池子里的,而connection本身没有池,那么只能通过代理方式增强连接池的功能,代理类帮它把connection放回池操作等等。

![](https://mmbiz.qpic.cn/mmbiz_png/ib8uEDL0WGNsZLGscLdgqxqrIrYibwgicIzxh9tygnX38WXYiaVIJzJOSgLzAibWiaibXib7KEmpMQpmiakubp9icYRwd1BQ/640?wx_fmt=png)

invoke的实现就是假如你要释放连接,那么判断方法是不是CLOSE,然后把连接放入dataSource容器,最后return也都是调回被代理方法的本身操作的方法

![](https://mmbiz.qpic.cn/mmbiz_png/ib8uEDL0WGNsZLGscLdgqxqrIrYibwgicIzzibG8qicLk9JNEROfYXAf713WcblniaHWH0kNzPwrWOUUicycUicDSuCozQ/640?wx_fmt=png)

Mybatis的日志

Mybatis会打印执行sql日记,肯定不能在业务里写,所以也需要代理模式

ConnectionLogger打印日志。

![](https://mmbiz.qpic.cn/mmbiz_png/ib8uEDL0WGNsZLGscLdgqxqrIrYibwgicIzzicxjQY7yKCbONOYiafkCJkK7L9XvKX21CXsq9ql5fQ3hIBbkxDiaGqpw/640?wx_fmt=png)
![](https://mmbiz.qpic.cn/mmbiz_gif/ib8uEDL0WGNtvibOhlFrhCmGq2ibia0F9Jic0dPUT9XUkzj1Gaic4Z5gGabacO2rnRknnXXTUiaH4PeVJe6KI1W8iauQzw/640?wx_fmt=gif)

代理模式在MyBatis中经常使用,希望这篇文章能让你掌握MyBatis原理,进一步提升操作数据库能力。