JAVA代理模式

什么是代理模式

代理类和被代理类实现同一个接口,也就是说他们有着相同的功能,区别在于,被代理类持有代理类的引用,被代理类中所有功能的实现都是由代理类来完成的

为什么要使用代理模式

一个优秀的软件设计,对于类似功能的模块,其中相同的逻辑,应该做到抽取与封装,来保证代码的简洁和可维护性,而对于其中不同的部分,则应该提供相应的方法来进行个性化定制.
现在有三个功能相似的类,他们有很多类似的方法,首先,把这些方法抽取出来,变成一个接口,然后写一个代理类实现这个接口,所有相同的逻辑,都抽取出来放在这个代理类中,三个类再调用这个代理类来完成相应的逻辑,不同的部分则正好可以在方法中自定义
还有第二种情况,一个接口中,有很多的方法,这些方法的实现中有很多的相同之处,比如retrofit的service接口,里面的方法都是网络请求的方法,比如我们现在要对所有的网络请求加缓存,难道所有的方法中都要写一次吗,显然不符合我们的设计规范,这里就可以用动态代理.

如何设计一个静态代理模式

  • 首先定义一个接口

      public interface NetWorkService {
          //获取用户名的网络请求
          String getUserName(int userId);
      
          //给文章点赞的网络请求
          Boolean likedThisArticle(int articleId);
      }
    
  • 接着是代理类

      public class NetWorkServiceImp implements NetWorkService {
          @Override
          public String getUserName(int userId) {
              //这里就来一个假的实现吧
              return "KingSlayer";
          }
      
          @Override
          public Boolean likedThisArticle(int articleId) {
              return true;
          }
      }
    
  • 然后是被代理类

      public class NetWorkServiceProxy implements NetWorkService {
          private NetWorkService netWorkServiceImp;
      
          public void attach(NetWorkService netWorkServiceImp) {
              this.netWorkServiceImp = netWorkServiceImp;
          }
      
          @Override
          public String getUserName(int userId) {
              //这里提出一个新的要求,将请求结果存入缓存
              String userName = netWorkServiceImp.getUserName(userId);
              CacheUtils.put(String.valueOf(userId), userName);
              return userName;
          }
      
          @Override
          public Boolean likedThisArticle(int articleId) {
              Boolean isActionSuccess = netWorkServiceImp.likedThisArticle(articleId);
              CacheUtils.put(String.valueOf(articleId), isActionSuccess);
              return isActionSuccess;
          }
      }
    

总结:上面这个静态代理完成之后,如果有别的类也需要这两个请求方法,只需要实现这个接口,然后绑定NetWorkServiceImp的对象就好了,非常的方便简洁,也利于维护

如何设计一个动态代理模式

  • 首先我们来看这个静态方法

      public static Object newProxyInstance(ClassLoader loader,
                                            Class<?>[] interfaces,
                                            InvocationHandler h)
              throws IllegalArgumentException {
          Objects.requireNonNull(h);
      
          final Class<?>[] intfs = interfaces.clone();
          // Android-changed: sm is always null
          // final SecurityManager sm = System.getSecurityManager();
          // if (sm != null) {
          //     checkProxyAccess(Reflection.getCallerClass(), loader, intfs);
          // }
      
          /*
           * Look up or generate the designated proxy class.
           */
          Class<?> cl = getProxyClass0(loader, intfs);
      
          /*
           * Invoke its constructor with the designated invocation handler.
           */
          try {
              // Android-changed: sm is always null
              // if (sm != null) {
              //     checkNewProxyPermission(Reflection.getCallerClass(), cl);
              // }
      
              final Constructor<?> cons = cl.getConstructor(constructorParams);
              final InvocationHandler ih = h;
              if (!Modifier.isPublic(cl.getModifiers())) {
                  // Android-changed: Removed AccessController.doPrivileged
                  cons.setAccessible(true);
              }
              return cons.newInstance(new Object[]{h});
          } catch (IllegalAccessException | InstantiationException e) {
              throw new InternalError(e.toString(), e);
          } catch (InvocationTargetException e) {
              Throwable t = e.getCause();
              if (t instanceof RuntimeException) {
                  throw (RuntimeException) t;
              } else {
                  throw new InternalError(t.toString(), t);
              }
          } catch (NoSuchMethodException e) {
              throw new InternalError(e.toString(), e);
          }
      }
    
  • 我们关注一下这个方法的第三个参数

      public class NetWorkServiceProxyHandler implements InvocationHandler {
      
          private NetWorkService netWorkService;
      
          public NetWorkServiceProxyHandler(NetWorkService netWorkService) {
              this.netWorkService = netWorkService;
          }
      
          @Override
          public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
              Object invoke = method.invoke(netWorkService, args);
              CacheUtils.put(String.valueOf(args[0]), invoke);
              return invoke;
          }
      }
    

在这里,我们创建一个类实现InvocationHandler接口,在这个类中,有一个invoke方法,我们拿到代理类,通过反射调用接口中所有的方法,同时也正好在这里把每个请求方法的结果都存入缓存中,就不用每个请求方法都写一遍了,我觉得这正是retrofit中通过动态代理获取service对象的原因.

©著作权归作者所有,转载或内容合作请联系作者
【社区内容提示】社区部分内容疑似由AI辅助生成,浏览时请结合常识与多方信息审慎甄别。
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

相关阅读更多精彩内容

  • 版权声明:本文为博主原创文章,未经博主允许不得转载 前言 Java 代理模式在 Android 中有很多的应用。比...
    cc荣宣阅读 947评论 0 7
  • 事例 小张是一个普普通通的码农,每天勤勤恳恳地码代码。某天中午小张刚要去吃饭,一个电话打到了他的手机上。“是XX公...
    余平的余_余平的平阅读 516评论 0 0
  • 代理模式 代理模式就是给某个对象提供一个代理对象,并由代理对象控制对于原对象的访问,即客户不直接操控原对象,而是通...
    雨中独奏阅读 313评论 1 1
  • 每一个宝宝都是带着祝福被生下来的,但和他今后是否辛福并没有什么关系
    姜渔阅读 177评论 0 0
  • 就在全国30多个省市全部出分的今天,天津市高校依然众志成城,万众一心坚决不出分。南开研究生院全日制硕士研究生的...
    totorous阅读 242评论 0 2

友情链接更多精彩内容