代理模式是指,为其他对象提供一种代理以控制对这个对象的访问,在某些情况下,一个对象不适合或者不能直接引用另一个对象,而代理对象可以在客户类和目标对象之间起到中介的作用
使用代理模式的作用
功能增强:可以在完成目标对象的调用时,附加一些额外的功能,这些额外的功能叫做功能增强
控制访问:在代理中,控制是否可以调用目标对象的方法
实现代理的方式
静态代理
代理类是自己手工实现的,自己创建一个java类,代表代理类
所要代理的目标类是确定的
特点
- 优点:实现简单,容易理解
- 缺点:当目标类增加了,代理类可能也需要成倍的增加;当接口中的方法增加或修改,目标类,代理类都需要相应修改
步骤
- 创建一个接口,定义方法,表示代理类和目标类要做的事情
- 创建目标类,实现接口中的方法
- 创建代理类,实现接口中的方法,代理类直接调用目标类的方法,并在其周围进行功能增强、控制访问
- 创建一个类调用代理的方法
动态代理
静态代理中目标类很多的时候,可以使用动态代理,避免静态代理的缺点
动态代理中目标类即使很多,代理类数量可以很少,当修改了接口中的方法时,不会影响代理类
概念:在程序执行过程中,使用jdk的反射机制,创建代理类对象,并动态的指定要代理的目标类。换句话说,动态代理是一种创建java对象的能力,不用创建专门的代理类,就可以创建一个代理类对象
动态代理其实就是jdk运行期间,动态创建class字节码并加载到jvm
动态代理的实现方式:
使用JDK动态代理
jdk动态代理是基于java的反射机制实现的,使用jdk中接口和类实现代理对象的动态创建
反射包:java.lang.reflect ,里面有三个类:InvocationHandler、Method、Proxy
反射,Method类,表示类中的方法,通过Method可以执行某个方法
jdk动态代理的实现
InvocationHandler接口(调用处理器),只有一个方法invoke(),表示代理要干什么
invoke()表示代理对象要执行的功能代码。代理类要完成的功能就写在invoke()方法中
方法原型:public Object invoke (Object proxy, Method method, Object[] args)
参数:
Object proxy:jdk创建的代理对象,无需赋值
Method method:目标类中的方法,jdk提供的method对象
Object[] args:目标类中方法的参数,jdk提供的
使用:
创建类实现接口InvocationHandler
重写invoke()方法,把原来静态代理中代理类要完成的功能,写在这
Method类:表示方法的,确切的说就是目标类中的方法
作用:通过Method可以执行某个类的方法,Method.invoke()
method.invoke(目标对象,方法的参数);
invoke是Method类中的一个方法,表示执行方法的调用
参数:1、表示对象,要执行这个对象的方法;2、方法执行时的参数值
返回值:方法执行后的返回值
Proxy类:核心的对象,创建代理对象
之前创建对象都是new 类的构造方法,现在使用Proxy类的方法,代替new的使用
方法:静态方法:newProxyInstance()
作用:创建代理对象,等同于静态代理中的new代理对象
方法原型:public static Object newProxyInstance(ClassLoader loader, Class<?>[] interfaces, InnovationHandler h)
参数
ClassLoader loader类加载器,负责向内存中加载对象的,使用反射获取对象的ClassLoader,目标对象的类加载器
Class<?>[] interfaces:接口,目标对象实现的接口,也是反射获取的
InnovationHandler h:自己写的,代理类要完成的功能
返回值:代理对象
实现动态代理的步骤
- 创建接口,定义目标类要完成的功能
- 创建目标类,实现接口
- 创建InvocationHandler接口的实现类,在invoke方法中完成代理类的功能
- 调用目标方法
method.invoke(target, args);
- 增强功能
- 使用Proxy类的静态方法,创建代理对象,并把返回值转为接口类型
- (最后调用哪个方法就会传入哪个方法)
使用cglib动态代理
cglib全称code generate library,是一个开源项目,是一个强大的、高性能、高质量的code生成类库,它可以在运行期扩展Java类与实现Java接口
cglib是一个第三方工具库,它可以创建代理对象
cglib的原理是继承,cglib通过继承目标类,创建它的子类,在子类中重写父类中同名的方法,实现功能的修改
因为cglib是继承,重写方法,所以要求目标类不能是final的,方法也不能是final的
cglib要求目标类比较宽松,只要能继承就可以了
cglib在很多的框架中使用,比如mybatis、spring框架中都有使用
使用jdk的proxy实现代理,要求目标类与代理类实现相同的接口,若目标类不存在接口,则无法使用该方式实现,但对于无接口的类,要为其创建动态代理,就要使用cglib来实现。