Spring 静态代理和动态代理
现在我们来模拟一下,某位学生去考试。
假设他(小明)正常的考试。
运行结果:
结果:
突然某一天,他睡过头了,来不急去考试,所有他打算叫另一个人(Cheater)去代替他考试。
运行结果:
结果:
上面的这些例子就是一个简单的代理行为。这个简单代理,耦合性太强了。作为演示就好了。
静态代理:
优点:
1、 实现松散耦合。
2、做到在不修改目标对象的功能前提下,对目标功能扩展。
还是拿上面的例子进行修改吧,小明还是一样起晚了,叫Cheater去代替他考试。
定义一个Exam接口 ,代表着考试的行为。
public interface Exam {//考试的接口 void exam(); }
去Student类中实现此接口,并实现方法。 (被代理)
public class Student implements Exam { public void exam(){ System.out.println("奋笔疾书,完成考试啦"); } }
Cheater也实现Exam接口,并实现方法 (代理)
public class Cheater implements Exam { //被代理的对象 private final Exam student; public Cheater(Exam student){ this.student = student; } public void exam() { System.out.println("考试的时候唱了一首凉凉,差点被劝退了。"); student.exam();//调用Student类的方法 } }
测试:
public class Main { public static void main(String[] args) { Exam xiaoMing = new Student(); xiaoMing.exam();//原来的行为
System.out.println("-----------下面是代理的行为------------");
Exam cheater = new Cheater(xiaoMing); cheater.exam();//代理的行为 } }
结果:
1 奋笔疾书,完成考试啦
2 -----------下面是代理的行为------------
3 考试的时候唱了一首凉凉,差点被劝退了。 4 奋笔疾书,完成考试啦
静态代理的缺点:
如果项目中有多个类,则需要编写多个代理类,工作量大,不好修改,不好维护,不能应对变化。
如果想解决上面的问题,可以使用动态代理。
动态代理:
1、使用JDK内置的Proxy实现
还是拿上面的例子进行修改吧,小明还是一样起晚了,叫人去代替他考试。(重新开始哈)
定义一个Exam接口。
public interface Exam { void exam(); }
Student 实现 Exam接口
public class Student implements Exam { public void exam() { System.out.println("奋笔疾书,完成考试啦"); } }
创建一个 JdkProxy 类 实现 InvocationHandler 接口
public class JdkProxy implements InvocationHandler { private Object object;//被代理的对象 public JdkProxy(){} public JdkProxy(Object object){//初始化的时候就赋值 this.object = object; }/** * 当用户调用对象中的每个方法时都通过下面的方法执行,方法必须在接口 * proxy 被代理后的对象 * method 将要被执行的方法信息(反射) * args 执行方法时需要的参数 */ public Object invoke(Object proxy, Method method, Object[] args) throws Throwable { System.out.println("考试的时候,吃颗糖压压惊,没有人知道,这次不是本人来考试的"); Object invoke = null; try{ invoke = method.invoke(object, args); }catch (Exception x){ System.out.println("异常信息:"+x.getMessage()); }return invoke;//调用被代理对象原来的方法(行为) } }
测试:
public class Main { public static void main(String[] args) { /* * 通过Proxy的newProxyInstance方法来创建我们的代理对象,我们来看看其三个参数 * 第一个参数 handler.getClass().getClassLoader() ,我们这里使用handler这个类的ClassLoader对象来加载我们的代理对象 * 第二个参数person1.getClass().getInterfaces(),我们这里为代理对象提供的接口是真实对象所实行的接口,表示我要代理的是该真实对象,这样我就能调用这组接口中的方法了 * 第三个参数handler, 我们这里将这个代理对象关联到了上方的 InvocationHandler 这个对象上 */ ClassLoader cl = Thread.currentThread().getContextClassLoader(); Exam o = (Exam) Proxy.newProxyInstance( cl,//类加载器 new Class[]{Exam.class},//获取被代理对象的所有接口 new JdkProxy(new Student())//InvocationHandler对象 ); o.exam();//代理后的行为 // 另一种写法 // Exam o1 = (Exam) Proxy.newProxyInstance( // Thread.currentThread().getContextClassLoader(), // Student.class.getInterfaces(), // new JdkProxy(new Student()) // ); // o1.exam(); } }
运行结果:
1 考试的时候,吃颗糖压压惊,没有人知道,这次不是本人来考试的 2 奋笔疾书,完成考试啦
使用内置的Proxy实现动态代理有一个问题:被代理的类必须实现接口,未实现接口则没办法完成动态代理。
2、动态代理,使用cglib实现
还是拿上面的例子来说吧。
这次我们不写接口了。不过 我们要实现MethodInterceptor接口,并实现方法
去maven 中心仓库 找到 CGlib 的依赖
Student类
public class Student { public void exam(){ System.out.println("奋笔疾书,完成考试啦"); } }
创建一个 CglibProxy 的类 并 实现 MethodInterceptor 接口 ,并实现方法
import net.sf.cglib.proxy.MethodInterceptor; import net.sf.cglib.proxy.MethodProxy; import java.lang.reflect.Method; public class CglibProxy implements MethodInterceptor { /* * 参数 * Object 为由CGLib动态生成的代理类实例 * method 为上文中实体类所调用的被代理的方法引用 * objects 为参数值列表 * methodProxy 为生成的代理类对方法的代理引用 * return 从代理实例的方法调用返回的值 * */ public Object intercept(Object o, Method method, Object[] objects, MethodProxy methodProxy) throws Throwable { System.out.println("考试的时候,唱了一首 我爱洗澡 ,差点被劝退 "); return methodProxy.invokeSuper(o,objects); } }
测试:
import net.sf.cglib.proxy.Enhancer; public class Main { public static void main(String[] args) { //增强器,动态代码生成器 Enhancer enhancer = new Enhancer(); //设置 生成类 的 父类 enhancer.setSuperclass(Student.class); //回调函数 enhancer.setCallback(new CglibProxy()); //动态生成字节码并返回代理对象 Student o = (Student) enhancer.create(); o.exam(); //这里是简化写法 //第一个参数 设置 生成类 的父类 ,第二参数 被代理类的所有接口 ,回调函数 Student student = (Student) new Enhancer().create(Student.class, null, new CglibProxy()); student.exam(); } }
运行结果:
1 考试的时候,唱了一首 我爱洗澡 ,差点被劝退 2 奋笔疾书,完成考试啦
这种代理的缺点:被代理的类必须不是final类。
小结:
使用cglib可以实现动态代理,即使被代理的类没有实现接口,但被代理的类必须不是final类。
示例代码下载地址: https://github.com/oukele/Spring-Proxy