看了tom老师讲的深刻分析spring源码,讲的挺好,作个小总结java
代理模式的定义:spring
为其余对象提供一种代理以控制对这个对象的访问。在某些状况下,一个对象不适合或者不能直接引用另外一个对象,而代理对象能够在客户端和目标对象之间起到中介的做用。ide
好比:测试
租房中介、火车票黄牛、媒人、经纪人、快递 this
这些人和你之间的关系能够算做为代理模式,在我须要租房、买票、拿快递时我能够找一个这个中间人去替我办这件事代理
代理模式须要知足的特色:code
一、执行者、被代理人对象
二、对于被代理人来讲,这件事情是必定要作的,可是我本身又不想作或者没有时间作,找代理。blog
三、须要获取到被代理的人我的资料。接口
代理模式和装饰模式区别:
代理模式关心的不是结果 是过程;装饰模式关心的是最终结果
下面是一个小demo,建立了一个媒婆代理帮你找对象功能:
文件结构
Person类:
package spring; public interface Person { void findLove(); }
Zhansan类:
package spring; public class Zhansan implements Person{ private String sex = "女"; private String name = "Zhansan"; @Override public void findLove() { System.out.println("我叫:"+this.name+" 性别:"+this.sex); System.out.println("我要找高富帅"); } }
Meipo类:
package spring; import java.lang.annotation.Target; import java.lang.reflect.InvocationHandler; import java.lang.reflect.Method; import java.lang.reflect.Proxy; public class Meipo implements InvocationHandler{ private Person target; //获取被代理人的我的资料 public Object getInstance(Person target) { this.target = target; Class clazz = target.getClass(); //生成代理对象:3个参数:ClassLoader类加载器 实现接口,代理人 System.out.println("被代理对象是:"+clazz); return Proxy.newProxyInstance(clazz.getClassLoader(), clazz.getInterfaces(), this); } @Override public Object invoke(Object proxy, Method method, Object[] args) throws Throwable { System.out.println("我是媒婆,准备介绍"); System.out.println("------------"); this.target.findLove(); System.out.println("------------"); return null; } }
测试类:
package spring; public class TestFindLove { public static void main(String[] args) { Person obj = (Person) new Meipo().getInstance(new Zhansan()); System.out.println(obj.getClass()); obj.findLove(); /*原理: * 1 拿到带代理对象的引用,而后获取它的接口 * 2 jdk代理从新生成一个类,同时实现咱们给的代理对象所实现的接口 * 3 把被代理对象的引用也拿到了 * 4 从新动态生成一个class字节码 * 5 编译 */ } }
输出结果:
能够看到,测试类中虽然new的是一个zhangsan类 可是obj.getClass()的输出是代理类com.sun.proxy.$Proxy0