设计模式-代理模式

设计模式-代理模式

代理模式(Proxy Design Pattern)在不改变原有业务代码的情况下,通过引入代理类来给原始类附加功能,一般分为静态代理和动态代理。

案例分析

如果有一个 UserServiceImpl 类,我们希望统计其中的每个方法执行耗时。

java 复制代码
public class UserServiceImpl {

    public void test() {
        IntStream.rangeClosed(1, 100).forEach(i -> {
            if (i % 10 == 0) {
                try {
                    Thread.sleep(100);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
            }
        });
    }

}

我们很容易想到的就是创建一个 UserServiceImpl 的示例,在调用前后记录时间并计算差值

java 复制代码
public class Main {

    public static void main(String[] args) {
        long start = System.currentTimeMillis();
        UserServiceImpl userService = new UserServiceImpl();
        userService.test();
        long end = System.currentTimeMillis();
        System.out.println("test耗时:" + (end - start) + "ms");
    }

}

但是一方面如果方法很多,那么所有的位置都需要加上这样的代码,另一方面这些与业务无关的代码有可能未来取消或改动,如果每次变化都修改源代码,会使程序出错的可能性变大,因此我们希望不改动原始代码来实现。

基于继承的代理实现
java 复制代码
public class UserServiceImplProxy extends UserServiceImpl{

    @Override
    public void test() {
        long start = System.currentTimeMillis();
        super.test();
        long end = System.currentTimeMillis();
        System.out.println("test耗时:" + (end - start) + "ms");
    }

}

public class Main {

    public static void main(String[] args) {
        UserServiceImpl proxy = new UserServiceImplProxy();
        proxy.test();
    }

}

我们使 UserServiceImplProxy 类继承 UserServiceImpl 类,使用完全相同的接口,这样在创建对象时利用多态特性,父类对象指向子类实例,就可以在不改动原始代码的情况下增加记录接口调用时间的日志。

基于接口的代理实现

首先定义 UserService 接口及实现类 UserServiceImpl

java 复制代码
public interface UserService {

    void test();

}

public class UserServiceImpl implements UserService {

    @Override
    public void test() {
        IntStream.rangeClosed(1, 100).forEach(i -> {
            if (i % 10 == 0) {
                try {
                    Thread.sleep(100);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
            }
        });
    }

}

我们创建一个新的类 UserServiceImplProxy 也实现这个接口

java 复制代码
public class UserServiceImplProxy implements UserService {

    private UserServiceImpl userService;

    public UserServiceImplProxy(UserServiceImpl userService) {
        this.userService = userService;
    }

    @Override
    public void test() {
        long start = System.currentTimeMillis();
        userService.test();
        long end = System.currentTimeMillis();
        System.out.println("test耗时:" + (end - start) + "ms");
    }

}

在 Proxy 代理类中,通过注入或构造器等把默认实现类传入,这样也可以达到我们的目的。

动态代理

在上述的两种实现中,不难看出这些都需要我们创建一些新的类,当类的数量很多时这是个很大的工作量,而且会影响编译和启动速度,动态代理可以在运行的过程中动态生成这样的类。

java 复制代码
public interface UserService {
    void test();
}

public class UserServiceImpl implements UserService {
    @Override
    public void test() {
        IntStream.rangeClosed(1, 100).forEach(i -> {
            if (i % 10 == 0) {
                try {
                    Thread.sleep(100);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
            }
        });
    }
}

以下是生成动态代理对象的代码:

java 复制代码
public class DynamicProxy {

    public Object createProxy(Object proxiedObject) {
        Class<?>[] interfaces = proxiedObject.getClass().getInterfaces();
        DynamicProxyHandler handler = new DynamicProxyHandler(proxiedObject);
        return Proxy.newProxyInstance(proxiedObject.getClass().getClassLoader(), interfaces, handler);
    }

    private class DynamicProxyHandler implements InvocationHandler {

        private Object proxiedObject;

        public DynamicProxyHandler(Object proxiedObject) {
            this.proxiedObject = proxiedObject;
        }

        @Override
        public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
            long startTimestamp = System.currentTimeMillis();
            Object result = method.invoke(proxiedObject, args);
            long endTimeStamp = System.currentTimeMillis();
            long responseTime = endTimeStamp - startTimestamp;
            String apiName = proxiedObject.getClass().getName() + ":" + method.getName();
            System.out.println("执行:" + apiName + "耗时:" + responseTime + "ms");
            return result;
        }
    }

}

测试代码:

java 复制代码
public class Main {

    public static void main(String[] args) {
        DynamicProxy serviceImplProxy = new DynamicProxy();
        UserService userService = (UserService)serviceImplProxy.createProxy(new UserServiceImpl());
        userService.test();
    }

}

动态代理其实就是在运行过程中动态地创建原始类对应的代理类,然后在系统中用代理类替换掉原始类。

我们常说 Spring AOP 就是基于动态代理实现的,同时 Spring 的事务中也用到了动态代理,那么看完上述的实现过程自然也知道了为什么 private 修饰的方法事务无法失效了~

相关推荐
贵慜_Derek7 小时前
《从零实现 Agent 系统》连载 07|记忆系统:短期上下文 vs 长期外部记忆
人工智能·设计模式·架构
老码观察12 小时前
设计模式实战解读(一):单例模式——全局唯一实例的正确打开方式
单例模式·设计模式
老码观察12 小时前
设计模式实战解读(二):工厂模式——对象创建的解耦艺术
设计模式·log4j
看山是山_Lau14 小时前
原型模式:当复制比重新创建更高效时
设计模式·原型模式
用户3563029048714 小时前
【设计模式】观察者模式——事件通知机制
设计模式
追烽少年x14 小时前
STL中的设计模式(二)
c++·设计模式
行者-全栈开发15 小时前
SpringBoot AOP 面向切面编程实战|4大企业级案例+自定义注解+5个避坑指南
spring boot·aop·权限校验·自定义注解·日志记录·避坑指南·切面编程
悟051515 小时前
设计模式-模板模式
设计模式
BLSxiaopanlaile15 小时前
有关创建型的几个设计模式总结
设计模式
蜡笔小马16 小时前
14.C++设计模式-状态模式
c++·设计模式·状态模式