什么是动态代理?
动态代理就是,在程序运行期,创建目标对象的代理对象,并对目标对象中的方法进行功能性增强的一种技术。在生成代理对象的过程中,目标对象不变,代理对象中的方法是目标对象方法的增强方法。可以理解为运行期间,对象中方法的动态拦截,在拦截方法的前后执行功能操作(也可以对原方法的参数进行操作)。
代理类在程序运行期间,创建的代理对象称之为动态代理对象。这种情况下,创建的代理对象,并不是事先在Java代码中定义好的。而是在运行期间,根据我们在动态代理对象中的“指示”,动态生成的 。也就是说,你想获取哪个对象的代理,动态代理就会为你动态的生成这个对象的代理对象。动态代理可以对被代理对象的方法进行功能增强。有了动态代理的技术,那么就可以在不修改方法源码 的情况下,增强被代理对象的方法的功能,在方法执行前后做任何你想做的事情。
特点:字节码随用随创建,随用随加载 作用:不修改源码的基础上对方法增强
正常类创建对象的过程: 动态代理创建代理对象的过程:
动态代理的常用两种方式:
-
基于接口的动态代理 提供者:JDK 使用JDK官方的Proxy类创建代理对象 注意:代理的目标对象必须实现接口(至少一个) -
基于类的动态代理 提供者:第三方 CGLib 使用CGLib的Enhancer类创建代理对象 注意:被代理类不能用 final 修饰的类(最终类)。如果报 asmxxxx 异常,需要导入 asm.jar包
目录
基于接口的动态代理
Proxy.newProxyInstance(三个参数);
ClassLoader:类加载器
它是用于加载代理对象字节码的。和被代理对象使用相同的类加载器。(固定写法)
Class[]:字节码数组
它是用于让代理对象和被代理对象有相同方法。(固定写法)
InvocationHandler:用于提供增强的代码
它是让我们写如何代理。我们一般都是写一个该接口的实现类,通常情况下都是匿名内部类,但不是必须
InvocationHandler该接口的实现类是谁用谁写,此时我们用就需要我们自己写
此处以一个演员的例子为例: 在很久以前,演员和剧组都是直接见面联系的。没有中间人环节。 而随着时间的推移,产生了一个新兴职业:经纪人(中间人),这个时候剧组再想找演员就需要通过经纪人来找了。下面我们就用代码演示出来。
package com.haust.service;
public interface IActor {
public void basicAct(float money);
public void dangerAct(float money);
}
package com.haust.serviceImpl;
import com.haust.service.IActor;
public class Actor implements IActor {
@Override
public void basicAct(float money) {
System.out.println("拿到钱,开始基本的表演:"+money);
}
@Override
public void dangerAct(float money) {
System.out.println("拿到钱,开始危险的表演:"+money);
}
}
package com.haust.test;
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;
import com.haust.service.IActor;
import com.haust.serviceImpl.Actor;
public class Client {
public static void main(String[] args) {
final Actor actor = new Actor();
IActor proxyActor = (IActor)Proxy.newProxyInstance(actor.getClass().getClassLoader(),
actor.getClass().getInterfaces(),
new InvocationHandler() {
@Override
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
String name = method.getName();
Float money = (Float) args[0];
Object rtValue = null;
if("basicAct".equals(name)){
if(money > 2000){
rtValue = method.invoke(actor, money/2);
} }
if("dangerAct".equals(name)){
if(money > 5000){
rtValue = method.invoke(actor, money/2);
} }
return rtValue;
}
});
proxyActor.basicAct(2000f);
proxyActor.dangerAct(50000f);
}
}
总结: 首先需要创建一个interface然后一个class实现这个interface,然后对这个class进行代理,这个class必须实现至少一个接口
基于子类的动态代理
设计的类:Enhancer
提供者:第三方cglib库
如何创建代理对象:
使用Enhancer类中的create方法
创建代理对象的要求:
被代理对象不是最终类(最终类没有子类)
create方法的参数:
Class方法的参数:
Class:字节码
它是用于指定被代理对象的字节码
callback:用于提供增强的代码
它是让我们写如何代理。我们一般都是些一个该接口的实现类,通常情况下都是匿名内部类,但不是必须的。此接口的实现类都是谁用谁写。
我们一般写的都是该接口的子接口实现类:MethodInterceptor
Enhancer.create(两个参数);
代码如下:
package com.haust.serviceImpl;
public class Actor{
public void basicAct(float money) {
System.out.println("拿到钱,开始基本的表演:"+money);
}
public void dangerAct(float money) {
System.out.println("拿到钱,开始危险的表演:"+money);
}
}
package com.haust.test;
import java.lang.reflect.Method;
import org.springframework.cglib.proxy.Enhancer;
import org.springframework.cglib.proxy.MethodInterceptor;
import org.springframework.cglib.proxy.MethodProxy;
import com.haust.serviceImpl.Actor;
public class test {
public static void main(String[] args) {
Actor actor = new Actor();
Actor cglibActor = (Actor)Enhancer.create(actor.getClass(),new MethodInterceptor() {
@Override
public Object intercept(Object proxy, Method method, Object[] args, MethodProxy methodProxy) throws Throwable {
String name = method.getName();
Float money = (Float) args[0];
Object rtValue = null;
if("basicAct".equals(name)){
if(money > 2000){
rtValue = method.invoke(actor, money/2);
} }
if("dangerAct".equals(name)){
if(money > 5000){
rtValue = method.invoke(actor, money/2);
}
}
return rtValue;
}
});
cglibActor.basicAct(10000);
cglibActor.dangerAct(100000);
}
}
总结:
-
无论哪种代理方式,都需要创建一个被代理的类(实例)。 -
不管是基于接口的代理,还是基于子类的代理,均拦截被代理对象的所有方法,然后我们可以对这些方法进行增强或者其他一些操作。
|