Spring专题1: 静态代理和动态代理
- 合集目录
- Spring专题1: 静态代理和动态代理
为什么需要代理模式?
代理对象处于访问者和被访问者之间,可以隔离这两者之间的直接交互,访问者与代理对象打交道就好像在跟被访者者打交道一样,因为代理者通常几乎会拥有全部被代理者的职能,代理对象能够处理的访问请求就不必要劳烦被访问者来处理了.
- 代理对象可以减少被访问者的负担,
- 在转发访问请求之前或者之后加入特定的逻辑.比如安全限制,或者做一些中间操作
- 便于通过配置, 直接修改实际的实现类
- 便于测试
讲讲静态代理模式的优点及其瓶颈?
直接编写实现方法的代理类, 这种代理方式称为静态代理, 这也是效率最高的一种方式, 因为所有的类都是已经编写完成的, 客户端只需要取得代理对象并且执行即可.
静态代理虽然效率较高, 但其缺点在于要为每个接口实现一个代理类, 而这些代理类中的代码几乎是一致的, 这在大型系统中将会产生很大的维护问题.
对Java 接口代理模式的实现原理的理解?
使用 Proxy 和 InvocationHandler 来实现
对于接口A, 以及其实现类AImpl,
创建一个子类 MyInvocationHandler, 继承自 InvocationHandler , 其构造参数可以用 AImpl, 也可以不用
用 Proxy.newProxyInstance()
, 输入指定的接口和MyInvocationHandler实例, 创建其代理对象, 然后就可以用A接口方法进行调用, 调用时都会经过 MyInvocationHandler.invoke() 方法
如何使用 Java 反射实现动态代理?
如果使用java反射, 则可以使用这样的形式得到代理类
public class MyInvocationHandler implements InvocationHandler { // 目标对象(需要被代理的对象) private Object target; public void setTarget(Object target) { this.target = target; } /** * 执行代理对象的所有方法时都会被替换成执行如下的invoke方法 */ @Override public Object invoke(Object proxy, Method method, Object[] args) throws Throwable { // 实例化一个增强类 Extend extend = new Extend(); // 执行增强方法1(模拟spring前置增强) extend.ExtendMethod1(); Object result = method.invoke(target, args);// 【执行目标方法】 // 执行增强方法2(模拟spring后置增强) extend.ExtendMethod2(); return result; } } public class MyProxyFactory { /* * 获取一个目标对象(目标类必须实现接口)的代理对象 */ public static Object getProxy(Object target) { // 实例化一个InvocationHandler类, 并传入目标对象 MyInvocationHandler myInvocationHandler = new MyInvocationHandler(); myInvocationHandler.setTarget(target); // 生成代理对象 Object proxyObject = Proxy.newProxyInstance(target.getClass().getClassLoader(), target.getClass().getInterfaces(), myInvocationHandler); return proxyObject; } }
Java 接口代理模式的指定增强?
在 InvocationHandler.invoke() 中, 在执行对象方法的前后, 进行额外的操作
谈谈对Cglib 类增强动态代理的实现?
https://blog.csdn.net/yaomingyang/article/details/82762697
CGLIB是一个强大、高性能的字节码生成库, 它用于在运行时扩展Java类和实现接口;本质上它是通过动态的生成一个子类
去覆盖所要代理的类(非final修饰的类和方法).
Enhancer是一个非常重要的类, 它允许为非接口类型创建一个JAVA代理, Enhancer动态的创建给定类的子类并且拦截代理类的所有的方法, 和JDK动态代理不一样的是不管是接口还是类它都能正常工作.
- net.sf.cglib.proxy.Callback接口, 在cglib包中是一个很关键的接口, 所有被net.sf.cglib.proxy.Enhancer类调用的回调(callback)接口都要继承这个接口
- net.sf.cglib.proxy.MethodInterceptor接口, 是通用的回调(callback)类型, 他经常被AOP用来实现拦截(intercept)方法的调用
一个常见的通过 MethodInterceptor 和 Enhancer 实现的代理实现方式
public class ProxyFactory implements MethodInterceptor { private Object obj; //要代理的真实对象 public Object createProxy(Object target) { this.obj = target; Enhancer enhancer = new Enhancer(); enhancer.setSuperclass(this.obj.getClass()); //设置代理目标 enhancer.setCallback(this); //设置单一回调对象, 在调用中拦截对目标方法的调用 enhancer.setClassLoader(this.obj.getClass().getClassLoader()); //设置类加载器 return enhancer.create(); } /** * 方法描述 当对基于代理的方法回调时, 在调用原方法之前会调用该方法 * 拦截对目标方法的调用 */ @Override public Object intercept(Object obj, Method method, Object[] args, MethodProxy proxy) throws Throwable { Object result = null; try { before(); //前置通知 result = proxy.invokeSuper(obj, args); after(); //后置通知 } catch (Exception e) { exception(); //异常通知 } finally { beforeReturning(); //方法返回前通知 } return result; } private void before() { System.out.println("before method invoke..."); } private void after() { System.out.println("after method invoke..."); } private void exception() { System.out.println("exception method invoke..."); } private void beforeReturning() { System.out.println("beforeReturning method invoke..."); } }
讲解JDK 动态代理和 CGLIB 代理原理以及区别?
cglib生成的代理实际上是子类, 可以不需要接口.
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】凌霞软件回馈社区,博客园 & 1Panel & Halo 联合会员上线
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】博客园社区专享云产品让利特惠,阿里云新客6.5折上折
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· PowerShell开发游戏 · 打蜜蜂
· 在鹅厂做java开发是什么体验
· 百万级群聊的设计实践
· WPF到Web的无缝过渡:英雄联盟客户端的OpenSilver迁移实战
· 永远不要相信用户的输入:从 SQL 注入攻防看输入验证的重要性
2017-01-15 Linux中后台执行任务
2017-01-15 通过ionice和nice降低shell脚本运行的优先级
2015-01-15 Java语言中的volatile变量