java基础之代理模式
静态代理
代理类与被代理类实现同一个接口或者父类,代理类中存在被代理类的引用,代理类中重写的方法内部调用添加的逻辑以及被代理类的方法。
要实现同一个接口是为了让代理类能调用与被代理类的方法名一致的方法,这样在调用代理类对象时同样只需要调用被代理类同名方法即可。
jdk动态代理
代理对象,不需要实现接口
代理对象的生成,是利用JDK的API,动态的在内存中构建代理对象(需要我们指定创建代理对象/目标对象实现的接口的类型)
动态代理也叫做:JDK代理,接口代理
DK实现代理只需要使用newProxyInstance方法,但是该方法需要接收三个参数,完整的写法是:
Proxy.newProxyInstance(ClassLoader loader, Class<>[] interfaces,InvocationHandler h )
newProxyInstance方法用来返回一个代理对象,这个方法总共有3个参数,ClassLoader loader用来指明生成代理对象使用哪个类装载器,
Class<?>[] interfaces用来指明生成哪个对象的代理对象,通过接口指定,
InvocationHandler h用来指明产生的这个代理对象要做什么事情。所以我们只需要调用newProxyInstance方法就可以得到某一个对象的代理对象了。
invocationHandler这个参数是个接口,所以该处是个匿名内部类,且必须要实现invoke方法
invoke(Object proxy, Method method, Object[] args)
第一个参数是指代理对象,要知道此方法中this并不是指代理对象,而是指invocationHandler的实现类对象,所以在这个方法中我们可以通过该参数调用代理对象
cglib动态代理
Cglib代理,也叫作子类代理,是基于asm框架,实现了无反射机制进行代理,利用空间来换取了时间,代理效率高于jdk,它是在内存中构建一个子类对象从而实现对目标对象功能的扩展.
它有如下特点:
JDK的动态代理有一个限制,就是使用动态代理的对象必须实现一个或多个接口,如果想代理没有实现接口的类,就可以使用Cglib实现.
Cglib是一个强大的高性能的代码生成包,它可以在运行期扩展java类与实现java接口.它广泛的被许多AOP的框架使用,例如Spring AOP和synaop,为他们提供方法的interception(拦截)
Cglib包的底层是通过使用一个小而块的字节码处理框架ASM来转换字节码并生成新的类.不鼓励直接使用ASM,因为它要求你必须对JVM内部结构包括class文件的格式和指令集都很熟悉.
注意
需要引入cglib的jar文件,但是Spring的核心包中已经包括了Cglib功能,所以直接引入pring-core-3.2.5.jar即可.
引入功能包后,就可以在内存中动态构建子类
代理的类不能为final,否则报错,因为要生成子类对象,如果为final,则无法构建子类
目标对象的方法如果为final/static,那么就不会被拦截,即不会执行目标对象额外的业务方法,因为子类对象要重写方法,而static修饰的方法无法形成多态,即使重写了该方法,父类引用指向子类对象调用该方法时依然是调用父类方法,而final修饰的方法不能被重写,所以这两种方法都无法被拦截,即父类引用调用该两种方法时其实依然是父类的方法,而不是代理的方法。
三个代理的优缺点
静态代理总结:
1.可以做到在不修改目标对象的功能前提下,对目标功能扩展.
2.缺点:
因为代理对象需要与目标对象实现一样的接口,所以会有很多代理类,类太多.同时,一旦接口增加方法,目标对象与代理对象都要维护.
JDK动态代理局限性
通过反射类Proxy和InvocationHandler回调接口实现的jdk动态代理,要求委托类必须实现一个接口,但事实上并不是所有类都有接口,对于没有实现接口的类,便无法使用该方方式实现动态代理。
cglib动态代理
使用CGLIB即使代理类没有实现任何接口也可以实现动态代理功能。CGLIB具有简单易用,它的运行速度要远远快于JDK的
https://blog.csdn.net/yangsnow_rain_wind/article/details/79291256