设计模式之代理模式的三种实现方式

本文详细介绍了设计模式中的代理模式,包括静态代理、JDK动态代理和CGlib代理的实现方式。静态代理需要为每个目标对象创建一个代理类,实现相同接口;JDK动态代理利用Proxy类生成代理对象,要求目标对象实现接口;CGlib代理通过创建目标对象的子类来实现,不需要接口,但需要引入额外的库。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

概念

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

在Java中代理模式的实现大概可分为三种:静态代理、动态代理、CGlib代理。

1. 静态代理

静态代理,代理类和被代理类(也即目标对象)需实现同一个接口,保证外部调用知道是哪个接口。代理类中维护目标对象,在方法里执行目标对象的方法。
缺点是,需要为每一个目标对象都构建一个代理类,目标对象数量多的话则代理类也多。

关键点:

  • 代理类需实现和目标对象同一个接口;

接口,代理类和被代理类需要共同实现的接口

public interface DemoService {
    void run();
}

被代理类,也即目标类

public class DemoServiceImpl implements DemoService{

    @Override
    public void run() {
        System.out.println("执行目标对象的方法。");
        try {
            Thread.sleep(100);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
    }
}

代理类,代理DemoServiceImpl的实现,执行业务逻辑

/**
 * 代理类,维护目标对象,执行业务逻辑 及 调用目标对象方法
 */
public class DemoServiceProxy implements DemoService {

    // 维护目标对象
    private DemoService service;

    DemoServiceProxy(DemoService service) {
        this.service = service;
    }

    @Override
    public void run() {
        System.out.println("代理类对象方法执行;" );
        long startTime = System.currentTimeMillis();

        // 执行目标方法
        service.run();

        // 记录耗时
        long endTime = System.currentTimeMillis();
        System.out.println("方法执行结束。耗时:" + (endTime - startTime) + "ms");
    }

}

调用的客户端

/**
 * 调用Demo的客户端类
 */
public class DemoClient {

    public static void main(String[] args) {

        // 目标对象
        DemoService service = new DemoServiceImpl();
        // 代理对象,把目标对象传给代理对象,建立代理关系
        DemoService serviceProxy = new DemoServiceProxy(service);
        // 执行方法
        serviceProxy.run();
    }
}

好处:目标对象只需要关注核心逻辑,业务逻辑交由代理类来实现,例如入参打印,耗时情况,以及异常处理等。

缺点:代理类需要实现接口的所有方法。

2. 动态代理

2.1 JDK 自带的实现

动态代理,代理类由Jdk的Proxy提供的newInstance方法可以生成某个对象的代理对象。该方法需要三个参数,分别是代理对象的类加载器、被代理类的接口、代理对象处理器。其中代理对象处理器为实现InvocationHandler接口,重写invoke方法。
被代理类需实现接口。
缺点是,目标对象需有接口。

关键点:

  • 代理拦截实现 InvocationHandler接口,重写invoke方法;
  • invoke方法有三个参数
    • proxy 代理对象实例
    • method 运行时调用的方法,也即目标对象需要执行的方法,Object obj = method.invoke(target);
    • args method 方法执行所需要的参数
  • 代理拦截中需维护目标对象target
  • 使用Proxy#newProxyInstance 获取代理对象,三个参数:
    • 目标对象的类加载器:target.getClass().getClassLoader();
    • 目标对象的接口:target.getClass().getInterfaces();
    • 实现InvocationHandler 的子类对象;

代理工厂以及代理实现

/**
 * 获取代理对象的工厂类
 */
public class DemoProxyFactory {

    // 目标对象。即被代理类
    private Object target;

    public DemoProxyFactory(Object target) {
        this.target = target;
    }

    public Object getProxyInstance() {
        InvocationHandler handler = new DemoServiceHandler();
        return Proxy.newProxyInstance(target.getClass().getClassLoader(), target.getClass().getInterfaces(),handler);
    }

    class DemoServiceHandler implements InvocationHandler {
        /**
         * 代理执行方法
         * @param proxy  代理对象实例
         * @param method 运行时调用的方法,也即目标对象需要执行的方法
         * @param args  method 方法执行所需要的参数
         * @return
         * @throws Throwable
         */
        @Override
        public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
            System.out.println("代理对象方法执行。proxy.class="+proxy.getClass().getCanonicalName());
            long startTime = System.currentTimeMillis();
            // 执行目标对象
            Object obj = method.invoke(target);
            // 记录耗时
            long endTime = System.currentTimeMillis();
            System.out.println("方法执行结束。耗时:" + (endTime - startTime) + "ms");
            return obj;
        }
    }

}

客户端调用

public class DemoClient {

    public static void main(String[] args) {

        DemoService service = new DemoServiceImpl();    // 目标对象
        DemoProxyFactory factory = new DemoProxyFactory(service);   // 构建代理工厂

        DemoService serviceProxy  = (DemoService)factory.getProxyInstance(); // 通过代理工厂获取代理类
        serviceProxy.run();
    }
}   

2.2 CGlib 代理的实现

CGlib代理是为了解决动态代理而生的,也被称为子类代理。通过实现MethodInterceptor接口,在内存中动态构建目标对象的子类,也即代理对象做扩展。其中动态构建目标对象子类通过Enhancer工具类构建,主要是设置父类,设置回调函数,创建子类。

关键点:

  • 引入cglib的包;
  • 实现cglib包里的接口MethodInterceptor,重写intercept方法;
  • intercept方法有四个参数:
    • Object o:cglib生成的代理类实例
    • Method method:目标方法,Object obj = method.invoke(target) 或 methodProxy.invokeSuper(target)
    • Object[] objects:目标方法的参数列表
    • MethodProxy methodProxy:生成的代理类对方法的代理引用
  • 需维护目标对象 target
  • 使用Enhancer工具类创建子类代理实例
    • 设置父类;
    • 设置回调对象,即实现MethodInterceptor接口的子类实例;
    • 创建子类代理对象

需要额外引入cglib的依赖

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

目标类

public class DemoCglibService {

    public void run() {
        System.out.println("执行目标对象的方法。");
        try {
            Thread.sleep(100);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
    }
}

方法拦截器的实现,

public class DemoCglibInterceptor implements MethodInterceptor {

    private Object target;

    public DemoCglibInterceptor(Object target) {
        this.target = target;
    }

    public Object getProxyInstance() {
        // 工具类
        Enhancer enhancer = new Enhancer();
        // 设置父类
        enhancer.setSuperclass(target.getClass());
        // 设置回调函数。回调到当前对象(因为当前对象就是callBack的实现
        enhancer.setCallback(this);

        return enhancer.create();   // 创建子类代理对象
    }



    @Override
    public Object intercept(Object o, Method method, Object[] objects, MethodProxy methodProxy) throws Throwable {
        System.out.println("代理对象方法执行。");
        long startTime = System.currentTimeMillis();
        // 执行目标对象
        Object obj = method.invoke(target);
        // 记录耗时
        long endTime = System.currentTimeMillis();
        System.out.println("方法执行结束。耗时:" + (endTime - startTime) + "ms");
        return obj;
    }
}

获取子类代理的工厂类

/**
 * 代理工厂
 */
public class DemoProxyFactory {

    /**
     * 获取子类代理对象实例
     * @param target  目标对象
     * @param callback 实现了 MethodInterceptor 的子类
     * @return
     */
    public static Object getProxyInstance(Object target, Callback callback) {
        // 工具类
        Enhancer enhancer = new Enhancer();
        // 设置父类
        enhancer.setSuperclass(target.getClass());
        // 设置回调函数。
        enhancer.setCallback(callback);

        return enhancer.create();   // 创建子类代理对象
    }
}

客户端调用

public class DemoClient {

    public static void main(String[] args) {

        System.out.println("方法一:使用方法拦截对象的方法获取代理对象");
        DemoCglibService service = new DemoCglibService();
        DemoCglibInterceptor cglibInterceptor = new DemoCglibInterceptor(service);

        DemoCglibService serviceProxy = (DemoCglibService) cglibInterceptor.getProxyInstance();
        serviceProxy.run();

        System.out.println("\n");

        System.out.println("方法二:使用代理工厂获取子类代理对象");
        DemoCglibService serviceProxy2 = (DemoCglibService) DemoProxyFactory.getProxyInstance(service,cglibInterceptor);
        serviceProxy2.run();
    }
}

注:如果使用Spring框架的话,就不需要额外引入cglib包了,因为Spring核心包已经包括了cglib功能,所以直接引入spring-core.jar即可。(Maven的传递依赖)

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值