Bootstrap

动态代理.

学习视频链接:B站 动力节点

java的设计模式中有一项设计模式叫做代理模式,所谓代理模式,就是通过代理方来操作目标对象,而不是自己直接调用。

一、代理模式

代理模式:为其它对象提供一种代理以控制对这个对象的访问。在某些情况下,一个对象不适合或者不能直接引用另一个对象,而代理对象可以在客户类和目标对象之间起到中介的作用。

换句话说,使用代理对象,是为了在不修改目标对象的基础上,增强主业务逻辑。

客户类真正想要访问的对象是目标对象,但客户类真正可以访问的对象是代理对象。客户类对目标对象的访问是通过访问代理对象来实现的。当然,代理类与目标类要实现同一个接口。

例如:假设有一个A类,本来是要调用C类的方法,完成某个功能,但是C不让A调用。

	A不能调用C的方法,那么就在A和C之间创建一个B代理(C让B访问)
	A访问B,B再访问C

二、代理模式的作用

  1. 功能增强:在原有的功能上增加了额外的功能。新增加的功能,叫做功能增强。
  2. 控制访问:代理类不允许访问目标对象。

三、实现代理的方式

1、静态代理

静态代理是指,代理类在程序运行前就已经定义好.java 源文件,其与目标类的关系在
程序运行前就已经确立。在程序运行前代理类已经编译为.class 文件。

  • 代理类是手工实现的,创建一个java类,表示代理类。
  • 同时所要代理的目标类是确定的。

特点:

  1. 实现简单
  2. 容易理解
模拟用户购买商品。
	用户是客户端类
	商家:代理,代理某个品牌的商品
	厂家:目标类
	三者的关系:用户(客户)->商家(代理)->厂家(目标)
	商家和厂家都是卖商品的,完成的功能都是一致的。
实现步骤:
1.创建一个接口,定义卖商品的方法,表示厂家和商家要完成的功能。
2.创建厂家类,实现1步骤的接口。
3.创建商家类(代理),也要实现1步骤的接口。
4.创建客户端类,调用商家的方法,买一件商品。

**缺点:**当项目中,目标类和代理类过多时,

  1. 当目标类增加了,代理类可能也需要成倍增加。代理类数量过多。
  2. 当接口中功能增加或者修改了,会影响众多的实现类、厂家类,代理也需要修改,影响比较多。

2、动态代理

当静态代理中目标类很多时,可以使用动态代理,避免静态代理的缺点。

动态代理中目标类即使很多,代理类数量可以很少,当修改了接口中的方法时,不会影响代理类。

动态代理是指代理类对象在程序运行时由 JVM 根据反射机制动态生成的。动态代理不
需要定义代理类的.java 源文件。

在程序执行过程中,使用jdk的反射机制,创建代理类对象,并动态地指定要代理的目标类。

换句话说,动态代理是一种创建java对象的能力,让你不用创建类,就能创建代理类对象。

四、动态代理

1、动态代理的实现

  • jdk动态代理:使用反射包中的类和接口实现动态代理的功能。

    反射包java.lang.reflect,里面有三个类:InvocationHandler,Method,Proxy

  • cglib动态代理:cglib是第三方的工具库,创建代理对象。

    cglib的原理是继承,cglib通过集成目标类,创建它的子类,在子类中重写父类中同名的方法,实现功能的修改。

    因为cglib是继承、重写方法,所以要求目标类不能是final的,方法也不能是final的。

    cglib的要求目标类比较宽松,只要能继承就可以了。cglib在很多的框架中使用,比如mybatis,spring框架中都有使用。

2、jdk动态代理的实现

反射, Method类,表示方法。类中的方法。 通过Method可以执行某个方法。

反射包java.lang.reflect,里面有三个类:InvocationHandler,Method,Proxy

  • InvocationHandler接口(调用处理器),就一个方法invoke()

    invoke()表示代理对象要执行的功能代码。代理类要完成的功能就写在invoke()方法中。

    代理类要完成的功能:

    1. 调用目标方法,执行目标方法的功能。
    2. 功能增强,在目标方法调用时,增加功能。
    方法原型 public Object invoke(Object proxy,Method method,Object[] args)
    参数:
    Object proxy:jdk创建的代理对象,无需赋值。(proxy:代理)
    Method method:目标类中的方法,jdk提供method对象
    Object[] args:目标类中方法的参数,jdk提供的。
    
    InvocationHandler 接口:表示你的代理要干什么。
    怎么用? 
    1.创建类实现接口InvocationHandler
    2.重写invoke()方法, 把原来静态代理中代理类要完成的功能,写在这。
    
  • Method类:表示方法,确切的说就是目标类中的方法。

    作用:通过Method可以执行某个目标类的方法,Method.invoke()

    ​ method.invoke(目标对象,方法的参数)

  • Proxy类:核心的对象,创建代理对象。以前创建对象都是new类的构造方法,现在使用Proxy类中的方法,代替new的使用。

    方法:静态方法 newProxyInStance()

    作用:创建代理对象

    public static Object newProxyInstance(ClassLoader loader,Class<?>[] interfaces,InvocationHandler h)
    参数:
        1.ClassLoader loader类加载器,负责向内存中加载对象。使用反射获取对象的ClassLoader类a,a.getClass().getClassLoader(),目标对象的类加载器。
        2.Class<?>[] interfaces:接口,目标对象实现的接口,也是反射获取的。
        3.InvocationHandler h:自己写的,代理类要完成的功能。
    返回值:就是代理对象。
    

实现动态代理的步骤:

  1. 创建接口,定义目标类要完成的功能
  2. 创建目标类实现接口
  3. 创建InvocationHandler接口的实现类,在invoke方法中完成代理类的功能
    • 调用目标方法
    • 增强功能
  4. 使用Proxy类的静态方法,创建代理对象。 并把返回值转为接口类型。

3、cgLib代理

CGLIB(Code Beneration Library)是一个开源项目。是一个强大的、高性能、高质量的Code生成类库,它可以在运行期扩展Java类实现Java接口。它广泛的被许多AOP的框架使用,例如Spring AOP。

使用JDK的Proxy实现代理,要求目标类与代理类实现相同的接口。若目标类不存在接口,则无法使用该方式实现。

但对于无接口的类,要为其创建动态代理,就要使用CGLIB来实现。CGLIB代理的生成原理是生成目标类的子类,而子类是增强过的,这个子类对象就是代理对象。所以,使用CGLIB生成动态代理,要求目标类必须能过够被继承,即不能是final的类。

;