代理模式(十五)

相信自己,请一定要相信自己

上一章简单介绍了享元模式(十四), 如果没有看过, 请观看上一章

一. 代理模式

引用 菜鸟教程里面的代理模式介绍: https://www.runoob.com/design-pattern/proxy-pattern.html

在代理模式(Proxy Pattern)中,一个类代表另一个类的功能。这种类型的设计模式属于结构型模式。

在代理模式中,我们创建具有现有对象的对象,以便向外界提供功能接口。

一.一 介绍

意图: 为其他对象提供一种代理以控制对这个对象的访问。

主要解决: 在直接访问对象时带来的问题,比如说:要访问的对象在远程的机器上。在面向对象系统中,有些对象由于某些原因(比如对象创建开销很大,或者某些操作需要安全控制,或者需要进程外的访问),直接访问会给使用者或者系统结构带来很多麻烦,我们可以在访问此对象时加上一个对此对象的访问层。

何时使用: 想在访问一个类时做一些控制。

如何解决: 增加中间层。

关键代码: 实现与被代理类组合。

应用实例: 1、Windows 里面的快捷方式。 2、猪八戒去找高翠兰结果是孙悟空变的,可以这样理解:把高翠兰的外貌抽象出来,高翠兰本人和孙悟空都实现了这个接口,猪八戒访问高翠兰的时候看不出来这个是孙悟空,所以说孙悟空是高翠兰代理类。 3、买火车票不一定在火车站买,也可以去代售点。 4、一张支票或银行存单是账户中资金的代理。支票在市场交易中用来代替现金,并提供对签发人账号上资金的控制。 5、spring aop。

优点: 1、职责清晰。 2、高扩展性。 3、智能化。

缺点: 1、由于在客户端和真实主题之间增加了代理对象,因此有些类型的代理模式可能会造成请求的处理速度变慢。 2、实现代理模式需要额外的工作,有些代理模式的实现非常复杂。

使用场景: 按职责来划分,通常有以下使用场景: 1、远程代理。 2、虚拟代理。 3、Copy-on-Write 代理。 4、保护(Protect or Access)代理。 5、Cache代理。 6、防火墙(Firewall)代理。 7、同步化(Synchronization)代理。 8、智能引用(Smart Reference)代理。

注意事项: 1、和适配器模式的区别:适配器模式主要改变所考虑对象的接口,而代理模式不能改变所代理类的接口。 2、和装饰器模式的区别:装饰器模式为了增强功能,而代理模式是为了加以控制。

在这里插入图片描述

二. 代理模式

有 静态代理, 动态代理 (JDK代理) 和 Cglib 代理 ( 可以在内存动态的创建对象)

二.一 静态代理

二.一.一 教师讲课的接口

public interface ITeacher {public void talk();
}

二.一.二 一个教师讲课的接口实现

@Slf4j
public class RealTeacher implements ITeacher{@Overridepublic void talk() {log.info("老师正在讲课");}
}

二.一.三 不使用代理

 @Testpublic void noProxyTest() {ITeacher iTeacher = new RealTeacher();iTeacher.talk();}

INFO [main] (RealTeacher.java:16) - 老师正在讲课

二.一.四 使用代理

@Slf4j
public class TeacherProxy implements ITeacher{private ITeacher iTeacher ;public TeacherProxy (ITeacher iTeacher) {this.iTeacher = iTeacher;}@Overridepublic void talk() {log.info("老师提前备课--> 用户 prepare 操作");iTeacher.talk();log.info("老师课后总结--> 用户 after 操作");}}

二.一.五 代理测试方法

@Testpublic void oneTest() {RealTeacher realTeacher = new RealTeacher();TeacherProxy teacherProxy = new TeacherProxy(realTeacher);teacherProxy.talk();}

image-20230615145810997

二.二 动态代理

二.二.一 接口

public interface ITeacher2 {public void talk();public void hello (String name);
}

二.二.二 接口实现

@Slf4j
public class RealTeacher2 implements ITeacher2{@Overridepublic void talk() {log.info("老师正在讲课");}@Overridepublic void hello(String name) {log.info("你好啊 {}", name);}
}

二.二.三 JDK 代理实现

@Slf4j
public class TeacherProxy2 {private Object target ;public TeacherProxy2 (Object target) {this.target = target;}/**获取动态代理*/public Object getProxyInstance() {return Proxy.newProxyInstance(target.getClass().getClassLoader(),target.getClass().getInterfaces(),new InvocationHandler() {@Overridepublic Object invoke(Object proxy, Method method, Object[] args) throws Throwable {log.info("JDK代理开始");Object returnVal = method.invoke(target, args);log.info("JDK代理结束");return returnVal;}});}}

二.二.四 代理测试

   @Testpublic void twoTest() {ITeacher2 teacher2 = new RealTeacher2();// 处理转换代理信息ITeacher2 target = (ITeacher2)(new TeacherProxy2(teacher2).getProxyInstance());// 进行处理target.hello("张三");}

image-20230615150102593

二.三 Cglib 代理

二.三.一 一个普通的类

@Slf4j
public class RealTeacher3 {public void talk() {log.info("老师正在讲课");}public void hello(String name) {log.info("你好啊 {}", name);}
}

二.三.二 对该类进行代理 TeacherProxy3

@Slf4j
public class TeacherProxy3 implements MethodInterceptor {private Object target;public TeacherProxy3(Object target) {this.target = target;}public Object getProxyInstance () {// 创建工具类 EnhancerEnhancer enhancer = new Enhancer();// 设置父类enhancer.setSuperclass(target.getClass());// 设置回调函数, 主要是这个。enhancer.setCallback(this);// 创建子类对象,即代理对象return enhancer.create();}@Overridepublic Object intercept(Object o, Method method, Object[] objects, MethodProxy methodProxy) throws Throwable {log.info("CGLIB 代理开始");Object returnVal = methodProxy.invoke(target, objects);log.info("CGLIB 代理结束");return returnVal;}}

二.三.三 代理测试

@Testpublic void threeTest() {RealTeacher3 realTeacher3 = new RealTeacher3();// 处理转换代理信息RealTeacher3 target = (RealTeacher3)new TeacherProxy3(realTeacher3).getProxyInstance();// 进行处理target.hello("张三");}

image-20230615150340055

三. Hutool 的 JDK 代理

可以使用 hutool 的 ProxyUtil ProxyFactory 进行代理

@Testpublic void fourTest() {ITeacher2 teacher2 = new RealTeacher2();ITeacher2 proxy = ProxyUtil.proxy(teacher2, new SimpleAspect());proxy.hello("hutool 工具类的代理");ITeacher2 proxy2 = ProxyFactory.createProxy(teacher2, new Aspect() {@Overridepublic boolean before(Object target, Method method, Object[] args) {log.info(">>>> 代理之前执行的操作");return true;}@Overridepublic boolean after(Object target, Method method, Object[] args, Object returnVal) {log.info(">>>> 代理之后执行的操作");return true;}@Overridepublic boolean afterException(Object target, Method method, Object[] args, Throwable e) {log.info(">>>> 异常时执行的操作");return true;}});proxy2.hello("hutool 工厂的代理");}

image-20230615150904736

CGLIB创建的动态代理对象比JDK创建的动态代理对象的性能更高,但是CGLIB创建代理对象时所花费的时间却比JDK多得多。

所以对于单例的对象,因为无需频繁创建对象,用CGLIB合适,反之使用JDK方式要更为合适一些。

同时由于CGLib由于是采用动态创建子类的方法,对于final修饰的方法无法进行代理


本章节的代码放置在 github 上:


https://github.com/yuejianli/DesignPattern/tree/develop/Proxy


谢谢您的观看,如果喜欢,请关注我,再次感谢 !!!


本文来自互联网用户投稿,文章观点仅代表作者本人,不代表本站立场,不承担相关法律责任。如若转载,请注明出处。 如若内容造成侵权/违法违规/事实不符,请点击【内容举报】进行投诉反馈!

相关文章

立即
投稿

微信公众账号

微信扫一扫加关注

返回
顶部