代理模式--【学习笔记】
创始人
2024-04-22 15:37:50

什么是代理模式?
代理是一种模式,提供了对目标对象的间接访问方式,即通过代理对象访问目标对象.如此便于在目标实现的基础上增加额外的功能操作,以满足自身的业务需求.

在这里插入图片描述

代理模式又分为静态代理,动态代理

静态代理模式

编写代理类, 要求: 代理类与目标类实现相同的接口, 代理类维护一个目标类对象, 目标方法由目标对象完成, 代理类作为额外的功能

图示:
在这里插入图片描述

代码演示:
目标类接口

/*** 工作接口*/
public interface Work {//工作的方法public int work();
}

目标类:

/*** 目标类*/
public class Worker implements  Work{@Overridepublic int work() {System.out.println("工人正在工作...");return 0;}
}

代理类:

/*** 代理类  代理的Worker类*   能够代理实现Work接口所有的实现类*/
public class AWorkerProxy implements Work {//目标类对象private Work  worker;  // 组合//构造public AWorkerProxy(Work  worker){this.worker = worker;}//set方式@Overridepublic int work() {System.out.println("验证身份...");System.out.println("打上班卡");//调用目标类的目标方法int rs = worker.work();System.out.println("打下班卡");return rs;}
}

优点:
1.实现了不改变目录类前提, 增强目录的类的方法
2相对于继承来实现: 静态代理不需要使用继承, 节省类继承资源, 静态代理增强的某一接口所有实现类, 继承方式, 只能增强某一个类

缺点:
1.只能增强某一接口, 如果要增强多个不同接口的实现类, 创建多个代理类, 代理类暴增
2. 代码重复
3. 增强代码直接写在代理类方法中, 硬编码, 后期修改, 违背开闭原则

动态代理:基于JDK提供: Proxy

可以解决静态代理只能增强某一个接口,代码重复的问题;

首先可以看一下创建一个对象的过程

  1. 先编写Person.java
  2. 编译Person.java,生成Person.class 使用 javac
  3. 通过类加载器, 把Person.class 加载到内存
  4. 运行代码: java --> main(){new Person();}

图示:
在这里插入图片描述

根据图中的步骤2,执行java命令会启动JVM将字节码文件加载进内存,然后再根据Class对象来创建对象

图示:
在这里插入图片描述
顺序反过来看,如果我们想得到一个代理类对象,就需要有一个代理类的Class对象。
构造这个Class对象的思路:
1.首先我们要根据目标类的接口来创建一个架构一样的Class对象(空壳架构,JDK提供了Proxy类实现这个功能)
2.再将目标类中要增强的方法,增强类(用来给目标增加功能)中的方法整合在一起
3.最后再更具这个代理Class对象,创建实例

在这里插入图片描述
(图中未加上增强类的方法)

内部设计思路:
每次调用代理对象的方法都会调用invoke(),且invoke()的返回值就是代理方法的返回值。
在这里插入图片描述

代码演示:

 private  Object getProxy(Object target,Advice advice) throws  Exception{//第一步:通过Proxy类,创建一个和目标类接口一样的架构的Class对象//参数1:类加载器  参数2:目标类的接口Class proxyClass = Proxy.getProxyClass(target.getClass().getClassLoader(),target.getClass().getInterfaces());//创建对象需要用到构造方法,但是Proxy生成的代理类对象的无参构造方法是私有的//只能调用调用下面的带参方法来创建实例Constructor constructor = proxyClass.getDeclaredConstructor(InvocationHandler.class);//创建代理类实例Object proxy = constructor.newInstance(new InvocationHandler(){//method调用方法//proxy: 代理类对象//args: 方法参数@Overridepublic Object invoke(Object proxy, Method method, Object[] args) throws Throwable {//第二步:将目标类中方法,和增强类中的方法组合在一起,形成一个新的方法advice.before();//调用目标对象的目标方法Object rs =  method.invoke(target,args);advice.after();return rs;}});//第三步:返回代理类对象(经过增强的代理类)return proxy;}

注意:newInstance方法,返回指定接口的实体类,必须用接口来接收返回值对象,不能用实现类来接收。不然会报错:
为什么这么做?个人理解:
代理类是根据目标类的接口而生成的,可以认为它和这个目标类没有一点关系。如果把这个代理类看作是目标类接口的一个子类,那么目标类和代理类都是接口的子类,他们之间是不能进行强转的,所以不能用实现类来接收返回值。

在这里插入图片描述

优点:

  1. 不改变目标类的代码, 对目标类的方法增强
  2. 代理各种不同的接口

缺点:

  1. 违背开闭原则: 增强代码虽然封装在一个一个增强类中,但是需要在invoke()方法,调用增强类的方法, 硬编码, 修改,需要修改invoke()方法
  2. 要求目标类必须实现某个接口, 基于接口的代理

解决方案:

解决1的方案: 使用配置文件/注解 替换硬编码 --> AOP(底层基于动态代理实现)

解决2的方案: 动态代理的第二种实现: 第三方: cglib, 代理类是目标类的子类, 这个类可以实现接口,也可以不实现接口

相关内容

热门资讯

北京的名胜古迹 北京最著名的景... 北京从元代开始,逐渐走上帝国首都的道路,先是成为大辽朝五大首都之一的南京城,随着金灭辽,金代从海陵王...
苗族的传统节日 贵州苗族节日有... 【岜沙苗族芦笙节】岜沙,苗语叫“分送”,距从江县城7.5公里,是世界上最崇拜树木并以树为神的枪手部落...
长白山自助游攻略 吉林长白山游... 昨天介绍了西坡的景点详细请看链接:一个人的旅行,据说能看到长白山天池全凭运气,您的运气如何?今日介绍...
世界上最漂亮的人 世界上最漂亮... 此前在某网上,选出了全球265万颜值姣好的女性。从这些数量庞大的女性群体中,人们投票选出了心目中最美...
阿西吧是什么意思 阿西吧相当于... 即使你没有受到过任何外语培训,你也懂四国语言。汉语:你好英语:Shit韩语:阿西吧(아,씨발! )日...