静态代理、动态代理与Mybatis的理解
静态代理、动态代理与Mybatis的理解
这里的代理与设计模式中的代理模式密切相关,代理模式的主要作用是为其他对象提供一种控制对这个对象的访问方法,即在一个对象不适合或者不能直接引用另一个对象时,代理对象充当中介的作用。
现实生活中比较贴切的例子比如租房,被代理对象就是房东,代理对象就是中介,使用者就是租客,租客通过中介向房东租赁房屋,即使用者通过代理对象访问被代理对象。
一、直接调用
一般我们通过new关键字初始化对象来调用类中的方法
如下代码,创建Human接口,Student类实现了Human接口,在main函数中,通过new关键字来初始化Student对象来实现对Student类中
say()方法的调用
interface Human{
public void say();
}
class Student implements Human{
@Override
public void say() {
System.out.println("I'm a Student");
}
}
public class ProxyTest {
public static void main(String[] args) {
Human human = new Student();
human.say();
}
}
//输出
//I'm a Student
二、静态代理
实现静态代理有以下三个步骤:
创建接口,通过接口来实现对象的代理
创建该接口的实现类
创建Proxy代理类来调用我们需要的方法
interface Human{
public void say();
}
class Student implements Human{
@Override
public void say() {
System.out.println("I'm a Student");
}
}
class StudentProxy implements Human{
private Student student;
public StudentProxy(){}
public StudentProxy(Student student){
this.student = student;
}
private void begin(){
System.out.println("Begin");
}
private void end(){
System.out.println("End");
}
@Override
public void say() {
begin();
student.say();
end();
}
}
public class ProxyTest {
public static void main(String[] args) {
Student student = new Student();
StudentProxy studentProxy = new StudentProxy(student);
studentProxy.say();
}
}
//输出
//Begin
//I'm a Student
//End
在上述代码中,我们在没有修改Student类中say()方法的情况下,实现了在原来的say()方法前后分别执行sayHello()和sayBye()方法。由此引出代理模式的主要作用:
- 在不修改被代理对象的情况下,实现对被代理对象功能的增强
同时,静态代理也存在一些比较致命的缺点。想象这样一个场景:若新增一个Worker类实现了Human接口,我们应该如何去代理这个Worker类?比较容易想到的方法是扩大StudentProxy的代理范围,然后将Worker当作参数传入StudentProxy,然后继续使用StudentProxy类代理Worker对象。这样实现功能是没有问题的,但会存在如下问题:
- 当Human接口的实例中方法增加时,代理类中代码会变得非常冗长
- 当有其他不属于Human类的子类需要被代理时,需要新增一个新的代理类
由此引出动态代理
三、动态代理
使用动态代理时,我们不需要编写实现类,而是通过JDK提供的Proxy.newProxyInstance()创建一个Human接口的对象。
生成动态代理有以下几个步骤:
- 定义一个
InvocationHandler实例,它负责实现接口的方法调用; - 通过
Proxy.newProxyInstance()创建interface实例,它需要3个参数:- 使用的
ClassLoader,通常是接口类的ClassLoader; - 需要实现的接口数组,至少需要传入一个接口进去;
- 用来处理接口方法调用的
InvocationHandler实例。
- 使用的
- 将返回的
Object强制转型为接口。
interface Human{
public void say();
}
class Student implements Human{
@Override
public void say() {
System.out.println("I'm a Student");
}
@Override
public void eat() {
System.out.println("I eat something");
}
}
class MyInvocationHandler implements InvocationHandler {
private Object object;
public MyInvocationHandler(){}
public MyInvocationHandler(Object object){
this.object = object;
}
@Override
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
System.out.println("Begin");
Object invoke = method.invoke(object, args);
System.out.println("End");
return invoke;
}
}
public class ProxyTest {
public static void main(String[] args) {
MyInvocationHandler handler = new MyInvocationHandler(new Student());
Human human = (Human) Proxy.newProxyInstance(
Human.class.getClassLoader(),
new Class[] {Human.class},
handler);
human.say();
human.eat();
}
}
当Human接口的实例中方法增加时,如新增eat()方法时,只需要在Student类中直接实例化该方法即可。
当有其他不属于Human类的子类需要被代理时,只需要将传入MyInvocationHandler()中的new Student()替换为需要被代理的子类即可。
综上所述,通过动态代理基本可以解决静态代理的痛点。
四、Mybatis中的动态代理
在Springboot项目中配置Mybatis时,我们仅编写了Mapper接口,并未编写Mapper接口的实现类,那么当我们调用Mapper接口中方法时,是如何生成方法体的呢?
首先,项目在启动时生成MapperFactoryBean对象,通过factory.getObject()方法获取mapper的代理对象





将上述过程与动态代理的步骤进行对比,我们最终获取的是一个类似于动态代理例子中Human的代理对象,这里是MapperProxy的代理对象。至此,一个Mapper代理对象就生成完毕。
然后,当我们完成项目中Mybatis的相关配置后,使用我们Mapper接口中的数据库相关方法时,将调用之前生成的MapperProxy代理对象中invoke()方法。类比动态代理的例子,即调用MyInvocationHandler类中的invoke()方法。

//83行代码含义:如果method为Object中定义的方法(toString()、hash()...)则直接执行,这里我们要执行的是Mapper接口中定义的方法,显然返回为false
Object.class.equals(method.getDeclaringClass())
于是执行cachedInvoker(method)的invoke()方法

进入execute()方法,我们看到之前我们配置的mapper.xml在MapperMethod初始化时,被解析成了59行的command。在此处通过sqlSession对象实现了对数据库的操作。

至此,我们对Mybatis的数据库操作流程已经有了大致了解。回到开头的问题:为什么仅编写了Mapper接口,并未编写Mapper接口的实现类,仍然可以实现我们的功能?这与我们之前的动态代理例子有什么区别呢?
研究代码我们发现,我们并没有直接使用method.invoke()方法来调用实现类中的方法,而是调用了cachedInvoker(method)的invoke()方法解析我们配置的Mapper.xml,并通过sqlSession实现了数据库操作,这个invoke()方法相当于Mybatis自定义的方法。因此,这里的invoke()方法具体执行的逻辑是根据Mapper.xml配置来生成的,这个Mapper.xml配置可以理解为Mapper接口的实现类。
静态代理、动态代理与Mybatis的理解的更多相关文章
- 轻松理解 Java 静态代理/动态代理
目录 什么是代理模式 定义 代理模式的主要角色 优点 缺点 静态代理 动态代理 JDK原生动态代理 例子 分析 小结 CGLIB动态代理 例子 分析 final类型 其他方案 尾声 理解Java动态代 ...
- 8、Spring教程之静态代理/动态代理
为什么要学习代理模式,因为AOP的底层机制就是动态代理! 代理模式: 静态代理 动态代理 学习aop之前 , 我们要先了解一下代理模式! 静态代理 静态代理角色分析 抽象角色 : 一般使用接口或者抽象 ...
- java静态和动态代理原理
一.代理概念 为某个对象提供一个代理,以控制对这个对象的访问. 代理类和委托类有共同的父类或父接口,这样在任何使用委托类对象的地方都可以用代理对象替代.代理类负责请求的预处理.过滤.将请求分派给委托类 ...
- 【SSH系列】静态代理&&动态代理
从设计模式说起 代理模式是二十三中设计模式中的一种,代理模式就是指由一个代理主题来操作真实的主题,真实的主题执行具体的业务操作,而代理主题负责其她相关业务,简而言之,代理模式可以由以下三个部分组成: ...
- 静态代理,动态代理,Cglib代理详解
一.静态代理 新建一个接口 定义一个玩家方法: package com."".proxy.staticc; public interface Iplayer { public vo ...
- Java代理模式/静态代理/动态代理
代理模式:即Proxy Pattern,常用的设计模式之一.代理模式的主要作用是为其他对象提供一种代理以控制对这个对象的访问. 代理概念 :为某个对象提供一个代理,以控制对这个对象的访问. 代理类和委 ...
- 啰里吧嗦式讲解java静态代理动态代理模式
一.为啥写这个 文章写的比较啰嗦,有些东西可以不看,因为想看懂框架, 想了解SSH或者SSM框架的设计原理和设计思路, 又去重新看了一遍反射和注解, 然后看别人的博客说想要看懂框架得先看懂设计模式,于 ...
- 静态代理&动态代理
原文地址:http://blog.csdn.net/partner4java/article/details/7048879 静态AOP和动态AOP. 静态代理: 代理对象与被代理对象必须实现同一个接 ...
- Java 的静态代理 动态代理(JDK和cglib)
转载:http://www.cnblogs.com/jqyp/archive/2010/08/20/1805041.html JAVA的动态代理 代理模式 代理模式是常用的java设计模式,他的特征是 ...
- spring aop,静态及动态代理例子
@Aspect@Componentpublic class AopText { @Pointcut("execution(public * com.llf.service.*Service. ...
随机推荐
- Mybatis-sql语句的抽取
1.抽取之前的UserMapper.xml <?xml version="1.0" encoding="UTF-8" ?> <!DOCTYPE ...
- Pinpoint介绍及docker安装方式
一.介绍 Pinpoint是用Java编写的大型分布式系统的APM(Application Performance Management应用程序性能管理)工具,受Dapper论文的启发,Pinpoin ...
- Promise的then和catch如何影响状态的变化
记住两句话: 1.then正常返回resolved,里面有报错返回reject 1 const p1 = Promise.resolve().then(()=>{ 2 console.log(1 ...
- 【原创】浅谈指针(十二)关于static(上)
0.前言 这个系列基本上是一月一更到两月一更 今天写一篇关于static的,内含大量干货,做好准备 1.基础知识的回顾 1.1.内存的种类 一般来说,我们之前已经讲过的变量(或者说是内存)可以大体分为 ...
- LVM从VG中删除PV及删除未知PV
当我们的硬盘发被删除掉了,我们的PV卷会变成[unknown] 一.首先我们要备份我们的文件,然后再删除lv分区 二. VG中去除PV unknown device:
- XCTF练习题---MISC---Excaliflag
XCTF练习题---MISC---Excaliflag flag:3DS{Gr4b_Only_th1s_B1ts} 解题步骤: 1.观察题目,下载附件 2.拿到手以后是一张图片,话不多说,直接上Ste ...
- [洛谷] P2010 [NOIP2016 普及组] 回文日期
点击查看代码 #include<bits/stdc++.h> using namespace std; int data1, data2, ans = 0, sum; int d[13] ...
- python3 获取函数变量
Python 3.8可以使用f字符串调试功能: 1 test_dict = {1: "1", 2: "2", 3: "3"} 2 print ...
- MySQL性能优化 - 别再只会说加索引了
MySQL性能优化 MySQL性能优化我们可以从以下四个维度考虑:硬件升级.系统配置.表结构设计.SQL语句和索引. 从成本上来说:硬件升级>系统配置>表结构设计>SQL语句及索引, ...
- OAuth 2.1 框架
OAuth 2.1 Draft 当前版本:v2-1-05 失效时间:2022/09/08 本文对部分原文翻译,同时加了一些笔记,以便理解. 单词 译意 identifiler 识别码 Resource ...