代理Proxy:

Proxy代理模式是一种结构型设计模式,主要解决的问题是:在直接访问对象时带来的问题

代理是一种常用的设计模式,其目的就是为其他对象提供一个代理以控制对某个对象的访问。代理类负责为委托类预处理消息,过滤消息并转发消息,以及进行消息被委托类执行后的后续处理。

为了保持行为的一致性,代理类和委托类通常会实现相同的接口,所以在访问者看来两者没有丝毫的区别。通过代理类这中间一层,能有效控制对委托类对象的直接访问,也可以很好地隐藏和保护委托类对象,同时也为实施不同控制策略预留了空间,从而在设计上获得了更大的灵活性。

更通俗的说,代理解决的问题当两个类需要通信时,引入第三方代理类,将两个类的关系解耦,让我们只了解代理类即可,而且代理的出现还可以让我们完成与另一个类之间的关系的统一管理,但是切记,代理类和委托类要实现相同的接口,因为代理真正调用的还是委托类的方法。

使用场合举例:

如果需要委托类处理某一业务,那么我们就可以先在代理类中统一处理然后在调用具体实现类

按照代理的创建时期,代理类可以分为两种:

静态:由程序员创建代理类或特定工具自动生成源代码再对其编译。在程序运行前代理类的.class文件就已经存在了。

动态:在程序运行时运用反射机制动态创建而成。

下面分别用静态代理与动态代理演示一个示例:

添加打印日志的功能,即每个方法调用之前和调用之后写入日志

静态代理:

具体用户管理实现类

public class UserManagerImpl implements UserManager {  @Override  public void addUser(String userId, String userName) {  System.out.println("UserManagerImpl.addUser");  }  @Override  public void delUser(String userId) {  System.out.println("UserManagerImpl.delUser");  }  @Override  public String findUser(String userId) {  System.out.println("UserManagerImpl.findUser");  return "张三";  }  @Override  public void modifyUser(String userId, String userName) {  System.out.println("UserManagerImpl.modifyUser");  }
}  

代理类--代理用户管理实现类

public class UserManagerImplProxy implements UserManager {  // 目标对象  private UserManager userManager;  // 通过构造方法传入目标对象  public UserManagerImplProxy(UserManager userManager){  this.userManager=userManager;  }  @Override  public void addUser(String userId, String userName) {  try{  //添加打印日志的功能  //开始添加用户  System.out.println("start-->addUser()");  userManager.addUser(userId, userName);  //添加用户成功  System.out.println("success-->addUser()");  }catch(Exception e){  //添加用户失败  System.out.println("error-->addUser()");  }  }  @Override  public void delUser(String userId) {  userManager.delUser(userId);  }  @Override  public String findUser(String userId) {  userManager.findUser(userId);  return "张三";  }  @Override  public void modifyUser(String userId, String userName) {  userManager.modifyUser(userId,userName);  }  }  

客户端调用

public class Client {  public static void main(String[] args){  //UserManager userManager=new UserManagerImpl();  UserManager userManager=new UserManagerImplProxy(new UserManagerImpl());  userManager.addUser("1111", "张三");  }
}  

静态代理类优缺点

优点:

代理使客户端不需要知道实现类是什么,怎么做的,而客户端只需知道代理即可(解耦合),对于如上的客户端代码,newUserManagerImpl()可以应用工厂将它隐藏,如上只是举个例子而已。

缺点:

1)代理类和委托类实现了相同的接口,代理类通过委托类实现了相同的方法。这样就出现了大量的代码重复。如果接口增加一个方法,除了所有实现类需要实现这个方法外,所有代理类也需要实现此方法。增加了代码维护的复杂度。

2)代理对象只服务于一种类型的对象,如果要服务多类型的对象。势必要为每一种对象都进行代理,静态代理在程序规模稍大时就无法胜任了。如上的代码是只为UserManager类的访问提供了代理,但是如果还要为其他类如Department类提供代理的话,就需要我们再次添加代理Department的代理类。

举例说明:代理可以对实现类进行统一的管理,如在调用具体实现类之前,需要打印日志等信息,这样我们只需要添加一个代理类,在代理类中添加打印日志的功能,然后调用实现类,这样就避免了修改具体实现类。满足我们所说的开闭原则。但是如果想让每个实现类都添加打印日志的功能的话,就需要添加多个代理类,以及代理类中各个方法都需要添加打印日志功能(如上的代理方法中删除,修改,以及查询都需要添加上打印日志的功能)

即静态代理类只能为特定的接口(Service)服务。如想要为多个接口服务则需要建立很多个代理类。

引入动态代理:

根据如上的介绍,你会发现每个代理类只能为一个接口服务,这样程序开发中必然会产生许多的代理类

所以我们就会想办法可以通过一个代理类完成全部的代理功能,那么我们就需要用动态代理

在上面的示例中,一个代理只能代理一种类型,而且是在编译器就已经确定被代理的对象。而动态代理是在运行时,通过反射机制实现动态代理,并且能够代理各种类型的对象

在Java中要想实现动态代理机制,需要java.lang.reflect.InvocationHandler接口和 java.lang.reflect.Proxy 类的支持

java.lang.reflect.InvocationHandler接口的定义如下:

//Object proxy:被代理的对象
//Method method:要调用的方法
//Object[] args:方法调用时所需要参数
public interface InvocationHandler {  public Object invoke(Object proxy, Method method, Object[] args) throws Throwable;
}

java.lang.reflect.Proxy类的定义如下:

//CLassLoader loader:类的加载器
//Class<?> interfaces:得到全部的接口
//InvocationHandler h:得到InvocationHandler接口的子类的实例
public static Object newProxyInstance(ClassLoader loader, Class<?>[] interfaces, InvocationHandler h) throws IllegalArgumentException

动态代理:

具体实现类

//CLassLoader loader:类的加载器
//Class<?> interfaces:得到全部的接口
//InvocationHandler h:得到InvocationHandler接口的子类的实例
public static Object newProxyInstance(ClassLoader loader, Class<?>[] interfaces, InvocationHandler h) throws IllegalArgumentException

动态创建代理对象的类

//动态代理类只能代理接口(不支持抽象类),代理类都需要实现InvocationHandler类,实现invoke方法。该invoke方法就是调用被代理接口的所有方法时需要调用的,该invoke方法返回的值是被代理接口的一个实现类  public class LogHandler implements InvocationHandler {  // 目标对象  private Object targetObject;  //绑定关系,也就是关联到哪个接口(与具体的实现类绑定)的哪些方法将被调用时,执行invoke方法。              public Object newProxyInstance(Object targetObject){  this.targetObject=targetObject;  //该方法用于为指定类装载器、一组接口及调用处理器生成动态代理类实例    //第一个参数指定产生代理对象的类加载器,需要将其指定为和目标对象同一个类加载器  //第二个参数要实现和目标对象一样的接口,所以只需要拿到目标对象的实现接口  //第三个参数表明这些被拦截的方法在被拦截时需要执行哪个InvocationHandler的invoke方法  //根据传入的目标返回一个代理对象  return Proxy.newProxyInstance(targetObject.getClass().getClassLoader(),  targetObject.getClass().getInterfaces(),this);  }  @Override  //关联的这个实现类的方法被调用时将被执行  /*InvocationHandler接口的方法,proxy表示代理,method表示原对象被调用的方法,args表示方法的参数*/  public Object invoke(Object proxy, Method method, Object[] args)  throws Throwable {  System.out.println("start-->>");  for(int i=0;i<args.length;i++){  System.out.println(args[i]);  }  Object ret=null;  try{  /*原对象方法调用前处理日志信息*/  System.out.println("satrt-->>");  //调用目标方法  ret=method.invoke(targetObject, args);  /*原对象方法调用后处理日志信息*/  System.out.println("success-->>");  }catch(Exception e){  e.printStackTrace();  System.out.println("error-->>");  throw e;  }  return ret;  }  }

被代理对象targetObject通过参数传递进来,我们通过targetObject.getClass().getClassLoader()获取ClassLoader对象,然后通过targetObject.getClass().getInterfaces()获取它实现的所有接口,然后将targetObject包装到实现了InvocationHandler接口的LogHandler对象中。通过newProxyInstance函数我们就获得了一个动态代理对象。

客户端代码

public class Client {  public static void main(String[] args){  LogHandler logHandler=new LogHandler();  UserManager userManager=(UserManager)logHandler.newProxyInstance(new UserManagerImpl());  //UserManager userManager=new UserManagerImpl();  userManager.addUser("1111", "张三");  }
}

可以看到,我们可以通过LogHandler代理不同类型的对象,如果我们把对外的接口都通过动态代理来实现,那么所有的函数调用最终都会经过invoke函数的转发,因此我们就可以在这里做一些自己想做的操作,比如日志系统、事务、拦截器、权限控制等。这也就是AOP(面向切面编程)的基本原理。

插曲:

AOP(AspectOrientedProgramming):将日志记录,性能统计,安全控制,事务处理,异常处理等代码从业务逻辑代码中划分出来,通过对这些行为的分离,我们希望可以将它们独立到非指导业务逻辑的方法中,进而改变这些行为的时候不影响业务逻辑的代码---解耦。

针对如上的示例解释:

我们来看上面的UserManagerImplProxy类,它的两个方法System.out.println("start-->addUser()")和System.out.println("success-->addUser()"),这是做核心动作之前和之后的两个截取段,正是这两个截取段,却是我们AOP的基础,在OOP里,System.out.println("start-->addUser()")、核心动作、System.out.println("success-->addUser()")这个三个动作在多个类里始终在一起,但他们所要完成的逻辑却是不同的,如System.out.println("start-->addUser()")里做的可能是权限的判断,在所有类中它都是做权限判断,而在每个类里核心动作却各不相同,System.out.println("success-->addUser()")可能做的是日志,在所有类里它都做日志。正是因为在所有的类里,核心代码之前的操作和核心代码之后的操作都做的是同样的逻辑,因此我们需要将它们提取出来,单独分析,设计和编码,这就是我们的AOP思想。一句话说,AOP只是在对OOP的基础上进行进一步抽象,使我们的类的职责更加单一。

动态代理优点:

动态代理与静态代理相比较,最大的好处是接口中声明的所有方法都被转移到调用处理器一个集中的方法中处理(InvocationHandler.invoke)。这样,在接口方法数量比较多的时候,我们可以进行灵活处理,而不需要像静态代理那样每一个方法进行中转。而且动态代理的应用使我们的类职责更加单一,复用性更强

总结:

其实所谓代理,就是一个人或者一个机构代表另一个人或者另一个机构采取行动。在一些情况下,一个客户不想或者不能够直接引用一个对象,而代理对象可以在客户端和目标对象之前起到中介的作用。

代理对象就是把被代理对象包装一层,在其内部做一些额外的工作,比如用户需要上facebook,而普通网络无法直接访问,网络代理帮助用户先FQ,然后再访问facebook。这就是代理的作用了。

纵观静态代理与动态代理,它们都能实现相同的功能,而我们看从静态代理到动态代理的这个过程,我们会发现其实动态代理只是对类做了进一步抽象和封装,使其复用性和易用性得到进一步提升而这不仅仅符合了面向对象的设计理念,其中还有AOP的身影,这也提供给我们对类抽象的一种参考。关于动态代理与AOP的关系,个人觉得AOP是一种思想,而动态代理是一种AOP思想的实现!

相关文章:java动态代理中的invoke方法是如何被自动调用的

参考:https://www.cnblogs.com/baizhanshi/p/6611164.html

静态代理和动态代理的区别和联系相关推荐

  1. 代理模式详解(静态代理和动态代理的区别以及联系)

    原文链接:https://www.cnblogs.com/takumicx/p/9285230.html 1. 前言 代理模式可以说是生活中处处可见.比如说在携程上定火车票,携程在这里就起到了一个代理 ...

  2. 静态代理,cglib动态代理,jdk动态代理区别以及流程详解

    1.静态代理 静态代理使用的是代理设计模式,不讲高大上的思想,我们直接实战 这是动物接口,其中有一个吃饭方法 这是其中的一只动物,实现了动物接口,覆盖了吃饭方法 现在我们思考,我想要给猫找一个代理,希 ...

  3. JAVA静态代理和动态代理的区别?

    代理简述 代理是英文 Proxy 翻译过来的.我们在生活中见到过的代理,大概最常见的就是朋友圈中卖面膜的同学了. 他们从厂家拿货,然后在朋友圈中宣传,然后卖给熟人. 按理说,顾客可以直接从厂家购买产品 ...

  4. Java设计模式学习06——静态代理与动态代理(转)

    原地址:http://blog.csdn.net/xu__cg/article/details/52970885 一.代理模式 为某个对象提供一个代理,从而控制这个代理的访问.代理类和委托类具有共同的 ...

  5. SpringAOP静态代理和动态代理

    代理模式应用于SpringAOP. SpringAOP.即面向切面的编程范式.多用于权限控制,异常处理,日志管理等场景.主要作用是分离功能性需求和非功能性需求,减少代码冗余.我们在使用的时候可以集中处 ...

  6. Java静态代理、动态代理与CGLib代理

    java的动态代理举足轻重,它同反射原理一直是许多框架的底层实现.今天唠一下. 一.代理模式 代理模式是一种设计模式,提供了对目标对象额外的访问方式,即通过代理对象访问目标对象,这样可以在不修改原目标 ...

  7. 【学习笔记】结合代码理解设计模式 —— 代理模式(静态代理、动态代理、延伸)

    文章目录 什么是代理模式 一. 代理模式简介 二. 静态代理模式 三. 动态代理模式 万能模版 前言:笔记基于狂神设计模式视频.<大话设计模式>观后而写 (最近一直在更新之前的刷题博客,今 ...

  8. 红橙Darren视频笔记 代理模式 动态代理和静态代理

    红橙Darren视频笔记 代理模式 动态代理和静态代理(Android API 25) 关于代理模式我之前有过相关的介绍: https://blog.csdn.net/u011109881/artic ...

  9. Spring系列之静态代理、动态代理、cglib代理与Spring AOP的处理

    本章内容 代理的概念及理解 如何实现静态代理 如何实现动态代理 静态代理与动态代理有什么区别与优缺点 JDK动态代理如何实现 cglib动态代理如何实现 JDK动态代理与cglib动态代理的区别 Sp ...

  10. Java设计模式(五)代理设计模式—静态代理—JDK动态代理—Cglib动态代理

    文章目录 什么是代理模式 代理模式应用场景 代理的分类 静态代理 什么是静态代理 深入解析静态代理 小结 动态代理 什么是动态代理 JDK动态代理 原理和实现方式 代码实现 优缺点 Cglib动态代理 ...

最新文章

  1. NVIDIA® TensorRT™ supports different data formats
  2. windows socket编程入门示例3
  3. 苹果手机怎么投屏 如何操作
  4. Ajax 模糊查询的简单实现
  5. Android学习拾遗
  6. pvrect r语言 聚类_R语言实现KEGG通路富集可视化
  7. 机器学习之路:python 集成分类器 随机森林分类RandomForestClassifier 梯度提升决策树分类GradientBoostingClassifier 预测泰坦尼克号幸存者...
  8. 列车控制matlab仿真,基于matlab的列车纵向碰撞建模仿真研究
  9. 转:java中获取实体类中的get、set方法名
  10. STM8L USART串口使用
  11. 《大型网站技术架构》读书笔记
  12. union并不绝对比or的执行效率高
  13. 智商高的人情商都低?這個人來告訴你
  14. 设计师专属的导航网站
  15. 启动车子温车_车子启动后水温上的很快是什么原?
  16. SiamFC代码配置复现 matlab版本
  17. 通过外网访问局域网中的IPCamera
  18. 无监督学习——K均值聚类的Python实现
  19. oracle小机系统,【案例分享】Oracle系统参数过小导致数据库宕机
  20. 打破富不过三代神话 荣智健中兴百年家业

热门文章

  1. ADNI数据集下载(详细)
  2. the JDBC Driver has been forcibly unregistered问题解决
  3. cf两边黑屏怎么解决win10_电脑黑屏怎么解决
  4. 用计算机清点木材的数量,计算机在木材工业中的应用.doc
  5. 查看npy文件中存的是什么
  6. win10 linux efi分区大小,EFI模式 win10+Ubuntu16.04双系统
  7. 微信小程序挑战赛:全校级人脸门禁系统
  8. 计算机组成体系结构复习笔记
  9. 2021年程序员平均工资
  10. java数据类型之间的转换_Java数据类型之间的转换(转)