设计模式之代理模式(静态&动态)代理

深藏阁楼爱情的钟 2024-04-23 20:15 183阅读 0赞

前言:

二十三种设计模式中的一种,属于结构型模式。它的作用就是通过提供一个代理类,让我们在调用目标

方法的时候,不再是直接对目标方法进行调用,而是通过代理类间接调用。让不属于目标方法核心逻辑

的代码从目标方法中剥离出来——解耦。调用目标方法时先调用代理对象的方法,减少对目标方法的调

用和打扰,同时让附加功能能够集中在一起也有利于统一维护。

使用代理前

如下,我们创建了一个接口,然后实现接口方法,模拟了计算器的加法,传入两个数字返回结果。

  1. public interface Calculator {
  2. int add(int i, int j);
  3. }
  4. public class CalculatorImpl implements Calculator{
  5. @Override
  6. public int add(int i, int j) {
  7. int result = i + j;
  8. System.out.println("i+j="+result);
  9. return result;
  10. }
  11. }

4bc48d0810294ac5ba41de0f3e4cf71d.png

就是简单的调用方法然后返回数据

使用代理后

9c70052676bb476fb54cc119051efb5d.png

使用了代理就是相当于代理帮我们去调用方法,可以理解为生活中明星的经纪人

静态代理

  1. public class CalculatorStaticProxy implements Calculator{
  2. private CalculatorImpl target;
  3. public CalculatorStaticProxy(CalculatorImpl target) {
  4. this.target = target;
  5. }
  6. @Override
  7. public int add(int i, int j) {
  8. System.out.println("日志,方法:add,参数:"+i+","+j);
  9. int result = target.add(i, j);
  10. System.out.println("日志,方法:add,结果:" + result);
  11. return result;
  12. }

比如我们要在刚刚的add方法中加入一些日志功能,先创建一个代理类,实现接口的add方法,我们在接口类中声明一个CalculatorImpl类型的target,将它私有化,然后给一个构造方法,我们在日志中间加入内部方法,用构造方法传进来的CalculatorImpl类型的target来调用int result = target.add(i, j);即CalculatorImpl类的?

  1. public class CalculatorImpl implements Calculator{
  2. @Override
  3. public int add(int i, int j) {
  4. int result = i + j;
  5. System.out.println("i+j="+result);
  6. return result;
  7. }
  8. }

我们在测试类里测试一下

  1. public class ProxyTest {
  2. @Test
  3. public void testProxy(){
  4. CalculatorStaticProxy proxy = new CalculatorStaticProxy(new CalculatorImpl());
  5. proxy.add(1,2);
  6. }
  7. }

b83fdfcd9a5d4196810e471fcf9f8e16.png

静态代理确实实现了解耦,但是由于代码都写死了,完全不具备任何的灵活性。就拿日志功能来

说,将来其他地方也需要附加日志,那还得再声明更多个静态代理类,那就产生了大量重复的代

码,日志功能还是分散的,没有统一管理。

提出进一步的需求:将日志功能集中到一个代理类中,将来有任何日志需求,都通过这一个代理

类来实现。这就需要使用动态代理技术了

动态代理

动态代理通过自己的方式创建出代理对象,实际操作时候直接操作代理对象即可,解决了代码冗余的弊端。市面上常见的两种动态代理方式,jdk动态代理和cglib动态代理。

这里演示的是jdk动态代理。

JDK动态代理是jdk自带的,是通过java.lang.reflect.Proxy创建的代理对象,它的创建方式是通过传入java.lang.reflect.InvocationHandler匿名内部类的方式,在通过反射创建代理对象时让代理对象同时实现被代理的接口和java.lang.reflect.InvocationHandler接口,从而达到一个动态代理的目的。因此,jdk动态代理需要被代理对象实现一个接口。

创建一个代理工厂,之前创建对象都是new一个类的构造方法(),现在我们是使用Proxy类的方法,让他代替我们new一个类

  1. public class ProxyFactory {
  2. private Object target;
  3. public ProxyFactory(Object target) {
  4. this.target = target;
  5. }
  6. public Object getProxy() {
  7. //classLoader:加载动态生成的代理类的类加载器
  8. ClassLoader classLoader = this.getClass().getClassLoader();
  9. //interfaces:目标对象实现的所有接口的class对象所组成的数组
  10. Class<?>[] interfaces = target.getClass().getInterfaces();
  11. /**invocationHandler:设置代理对象实现目标对象方法的过程,即代理类中如何重写接
  12. 口中的抽象方法*/
  13. InvocationHandler handler = new InvocationHandler() {
  14. @Override
  15. public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
  16. /**
  17. * proxy:代理对象
  18. * method:代理对象需要实现的方法,即其中需要重写的方法
  19. * args:method所对应方法的参数*/
  20. System.out.println("日志,方法:" + method.getName() + ",参数:" + Arrays.toString(args));
  21. Object result = method.invoke(target, args);
  22. System.out.println("日志,方法:" + method.getName() + ",参数:" + result);
  23. return result;
  24. }
  25. };
  26. return Proxy.newProxyInstance(classLoader, interfaces, handler);
  27. }
  28. }

测试一下

  1. @Test
  2. public void testProxy2(){
  3. ProxyFactory proxyFactory = new ProxyFactory(new CalculatorImpl());
  4. Calculator proxy =(Calculator) proxyFactory.getProxy();
  5. proxy.add(1,2);
  6. }
  7. }

a54d20fedafa4e23ad443b5d8d2f64c0.png

发表评论

表情:
评论列表 (有 0 条评论,183人围观)

还没有评论,来说两句吧...

相关阅读