动态代理

作者: 墨线宝 | 来源:发表于2021-01-14 21:12 被阅读0次

原文链接http://zhhll.icu/2020/11/12/java%E5%9F%BA%E7%A1%80/%E5%8F%8D%E5%B0%84/%E5%8A%A8%E6%80%81%E4%BB%A3%E7%90%86/

动态代理

动态代理有很多种方式,如jdk代理,cglib,ASM等

在说动态代理之前先说一下静态代理

静态代理

静态代理在使用时,需要定义接口或者父类,被代理对象和代理对象一起实现相同的接口或者继承相同的父类

静态代理使用的是组合模式,在代理类中包含有被代理类的对象

public class TestStaticProxy {
    public static void main(String[] args) {
        NikeClothFactory nikeClothFactory = new NikeClothFactory();
        ProxyFactory proxyFactory = new ProxyFactory(nikeClothFactory);
        proxyFactory.productCloth();
    }
}

// 接口
interface ClothFactory{
    void productCloth();
}

// 被代理类
class NikeClothFactory implements ClothFactory{

    @Override
    public void productCloth() {
        System.out.println("Nike工厂生产了一件Nike");
    }
}

// 代理类
class ProxyFactory implements ClothFactory{

    private ClothFactory clothFactory;

    public ProxyFactory(ClothFactory clothFactory){
        this.clothFactory = clothFactory;
    }
    @Override
    public void productCloth() {
        System.out.println("代理类开始执行,准备调用被代理类");
        clothFactory.productCloth();
    }
}

静态代理虽然可以在不修改目标对象功能的前提下对目标功能进行扩展,但是一旦接口增加方法,目标对象和代理类都要同时修改,而且代理对象和被代理对象要实现一样的接口,导致有很多的代理类,不便于维护

jdk动态代理

jdk动态代理的底层是用的是java的反射,但是jdk代理的前提是目标类必须实现接口

使用步骤

  • 首先实现一个InvocationHandler,方法调用会被转发到该类的invoke()方法
  • 调用时通过动态代理获取代理对象

核心方法为Proxy.newProxyInstance(ClassLoader,Class[],InvocationHandler)

三个参数分别表示

  • ClassLoader 当前目标对象使用的类加载器
  • Class[] 目标对象实现的接口类型
  • InvocationHandler 事件处理,执行目标对象的方法时,会触发事件处理器的方法,会把当前执行目标对象的方法作为参数传入
public class TestDynamicProxy {
    public static void main(String[] args) {
        RealSubject realSubject = new RealSubject();
        MyInvationHandler myInvationHandler = new MyInvationHandler();
        Object obj = myInvationHandler.blind(realSubject);
        Subject sub = (Subject) obj;
        sub.action();
    }
}

// 接口
interface Subject{
    void action();
}

// 真正执行的方法,被代理类
class RealSubject implements Subject{

    @Override
    public void action() {
        System.out.println("被代理类开始执行");
    }
}

class MyInvationHandler implements InvocationHandler{

    // 实现了接口的被代理类的对象的声明
    Object obj;
    // 被代理类的实例
    // 返回一个代理类的对象
    public Object blind(Object obj){
        this.obj = obj;
        // ①使用被代理类的类加载器②被代理类的接口③代理类的实例
        return Proxy.newProxyInstance(obj.getClass().getClassLoader(),obj.getClass().getInterfaces(),this);
    }

    /**
     * 当通过代理类的对象被重写的方法调用时,都会转换为对invoke方法的调用
     * @param proxy 正在返回的代理对象,一般情况下,在invoke方法中不使用该对象
     * @param method 正在被调用的方法 
     * @param args 调用方法时,传入的参数
     * @return
     * @throws Throwable
     */
    @Override
    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {

        return method.invoke(obj,args);
    }
}

cglib代理

静态代理和jdk动态代理都妖气目标对象一定要实现接口,但是有时候目标对象只是一个单独的对象,并没有实现任何接口,这个时候采用以目标对象子类的方式实现代理,该方法称为cglib代理

Cglib包的底层是通过使用一个字节码处理框架ASM来转换字节码并生成新的类,由于要生成子类,所以要被代理的类不可以被final修饰

需要引入cglib的包

<dependency>
  <groupId>cglib</groupId>
  <artifactId>cglib</artifactId>
  <version>3.3.0</version>
</dependency>

使用步骤

  • 首先实现MethodIntercepor,方法调用会被转发到intercept()方法
  • 使用CGLIB来获取代理对象
public class CglibProxy implements MethodInterceptor { 

    public Object getProxy(Class clazz){
      Enhancer enhancer = new Enhancer();
        // 指定代理类的父类
        enhancer.setSuperclass(clazz);
        // 设置Callback对象
        enhancer.setCallback(this);
        // 通过字节码技术动态创建子类实例
        return enhancer.create();
    }
    @Override
    public Object intercept(Object o, Method method, Object[] objects, MethodProxy methodProxy) throws Throwable {
        System.out.println("前置处理");
        // 调用MethodProxy.invokeSuper方法将调用转发给原始对象
        Object result = methodProxy.invokeSuper(o,objects);
        System.out.println("后置处理");
        return result;
    }
}


public class Test {

    public void print(){
        System.out.println("方法执行");
    }

    public static void main(String[] args) {
        CglibProxy proxy = new CglibProxy();
        Test proxyImp = (Test) proxy.getProxy(Test.class);
        proxyImp.print();

    }
}

由于本身的博客百度没有收录,博客地址http://zhhll.icu

相关文章

  • 面试系列~动态代理实现与原理

    动态代理有JDK动态代理, CGLIB动态代理, SpringAOP动态代理 一,JDK动态代理  jdk动态代理...

  • 编程常用的设计模式

    动态代理和静态代理 静态代理 动态代理 静态代理与动态代理的区别 JDK中的动态代理和CGLIB 实现动态代理的方...

  • Spring的AOP原理分析

    一 动态代理 动态代理分为JDK动态代理和CGLIB动态代理 jdk动态代理 被代理类(目标类)和代理类必须实现同...

  • 设计模式之代理模式

    代理分为静态代理和动态代理。 动态代理又包括基于JDK的动态代理、基于CGlib 的动态代理、基于Aspectj实...

  • Java高级主题(五)——动态代理

    代理可以分为静态代理、动态代理,动态代理又可以分为 jvm的动态代理 和 cglib的动态代理。像spring框架...

  • 动态代理

    动态代理分为两类:1、基于接口的动态代理; (JDK动态代理 )2、基于类的动态代理;(cglib动态代理)3、J...

  • 动态代理的两种方式

    静态代理就不说了,基本用到的都是动态代理。 Java中动态代理有JDK动态代理和CGLIB动态代理。 JDK代理的...

  • Java动态代理

    通过以下几种方式介绍动态代理 动态代理涉及到的类 动态代理用法 Proxy类解析 动态代理类解析 动态代理涉及到的...

  • Spring之代理模式

    九、代理模式 目录:静态代理、动态代理AOP的底层机制就是动态代理。代理模式分为静态代理和动态代理。接触aop之前...

  • Java 代理

    静态代理 动态代理 动态代理, 日志切片使用反射获得方法 动态代理, 自定义注解(对注解的方法,使用动态代理添加切...

网友评论

    本文标题:动态代理

    本文链接:https://www.haomeiwen.com/subject/agdbaktx.html