热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

精讲代理设计模式

代理设计模式为其他对象提供一种代理以控制对这个对象的访问。代理模式实现原理代理模式主要包含三个角色,即抽象主题角色(Subject)、委托类角色(被代理角色ÿ

代理设计模式

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

代理模式实现原理

代理模式主要包含三个角色,即抽象主题角色(Subject)、委托类角色(被代理角色,Proxied)以及代理类角色(Proxy),如上图所示:
在这里插入图片描述

抽象主题角色:可以是接口,也可以是抽象类;
委托类角色:真实主题角色,业务逻辑的具体执行者;
代理类角色:内部含有对真实对象RealSubject的引用,负责对真实主题角色的调用,并在真实主题角色处理前后做预处理和后处理。

代理模式应用场景

SpringAop、日志收集、权限控制、过滤器、RPC远程调用

静态代理与动态代理区别

静态代理需要自己写代理类,而动态代理不需要写代理类。

JDK动态代理与CGLIB实现区别


  • JDK动态代理底层实现

动态代理的类实现接口
JDK的动态代理使用Java的反射技术生成动态代理类,只能代理实现了接口的类, 没有实现接口的类不能实现动态代理。

  • CGLIB动态代理底层实现

运行时动态的生成一个被代理类的子类
(通过ASM字节码处理框架实现),子类重写了被代理类中所有非final的方法,在子类中采用方法拦截的技术拦截所有父类方法的调用,不需要被代理类对象实现接口,从而CGLIB动态代理效率比Jdk动态代理反射技术效率要高。

jdk动态代理代码

public interface OrderService {void order();
}
public class OrderServiceImpl implements OrderService {public void order() {System.out.println("修改数据库订单操作..");}
}public class JdkInvocationHandler implements InvocationHandler {
/*** 目标代理对象*/
public Object target;public JdkInvocationHandler(Object target) {this.target = target;
}@Override
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {System.out.println(">>>日志收集开始>>>>");// 执行代理对象方法Object reuslt = method.invoke(target, args);System.out.println(">>>日志收集结束>>>>");return reuslt;
}/*** 获取代理对象接口** @param * @return*/
public T getProxy() {return (T) Proxy.newProxyInstance(target.getClass().getClassLoader(), target.getClass().getInterfaces(), this);
}public static void main(String[] args) {// 1.获取代理的生成的class文件System.getProperties().put("sun.misc.ProxyGenerator.saveGeneratedFiles", "true");JdkInvocationHandler jdkInvocationHandler = new JdkInvocationHandler(new OrderServiceImpl());OrderService proxy = jdkInvocationHandler.getProxy();proxy.order();
}
}

Cglib动态代理代码

public class OrderServiceImpl {public void order() {System.out.println("用户下单操作..");}
}public class CglibMethodInterceptor implements MethodInterceptor {
@Override
public Object intercept(Object obj, Method method, Object[] args, MethodProxy proxy) throws Throwable {System.out.println("<<<<<日志收集开始...>>>>>>>");Object reuslt &#61; proxy.invokeSuper(obj, args);System.out.println("<<<<<日志收集结束...>>>>>>>");return reuslt;
}public static void main(String[] args) {// 1.获取代理的生成的class文件保存到D:\\codeSystem.setProperty(DebuggingClassWriter.DEBUG_LOCATION_PROPERTY, "D:\\code");CglibMethodInterceptor cglibMethodInterceptor &#61; new CglibMethodInterceptor();Enhancer enhancer &#61; new Enhancer();// 设置代理类的付类enhancer.setSuperclass(OrderServiceImpl.class);// 设置回调对象enhancer.setCallback(cglibMethodInterceptor);// 创建代理对象OrderServiceImpl orderServiceImpl &#61; (OrderServiceImpl) enhancer.create();orderServiceImpl.order();}
}maven依赖cglibcglib3.2.12

http://www.mayikt.com


推荐阅读
author-avatar
Money00Fish
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有