代理模式
在代理模式里面,使用代理类控制对目标类的访问,通过代理对象控制对原对象的访问。这里的代理对象其实就是相当于是生活中的中介。
举个例子来说明:假如说我现在想买一辆二手车,虽然我可以自己去找车源,做质量检测等一系列的车辆过户流程,但是这确实太浪费我得时间和精力了。我只是想买一辆车而已为什么我还要额外做这么多事呢?于是我就通过中介公司来买车,他们来给我找车源,帮我办理车辆过户流程,我只是负责选择自己喜欢的车,然后付钱就可以了。
代理对象的存在隔离了用户和目标对象,并且可以通过给代理对象添加额外的功能来扩展目标类的功能,并且不用修改目标类的代码。总而言之,代理对象解决了直接访问对象而带来的问题。比如,要访问的对象在远程机器上面。在面向对象系统中,有些对象由于某些原因(比如对象创建开销很大,或者某些操作需要安全控制,或者需要进程外的访问),直接访问会给使用者或者系统结构带来很多麻烦,我们可以在访问此对象时加上一个对此对象的访问层。
用户只会访问代理对象,代理对象是目标对象的扩展,代理对象会调用目标对象,用户不会直接访问目标对象。
静态代理
静态代理是由程序员或是特定工具生成源代码,再对其进行编译,在程序员运行之前代理类的.class文件就已经被创建了。
BuyHouse接口:
public interface BuyHouse { void buyHouse(); }
BuyHouse接口实现:
public class BuyHouseImpl implements BuyHouse{ @Override public void buyHouse() { System.out.println("I want to buy a house!"); } }
代理对象的实现:
public class BuyHouseProxy implements BuyHouse{ private BuyHouse buyHouse; public BuyHouseProxy(BuyHouse buyHouse) { this.buyHouse=buyHouse; } @Override public void buyHouse() { System.out.println("before buy house"); buyHouse.buyHouse(); System.out.println("after buy house"); } }
测试类实现:
public class ProxyTest { public static void main(String[] args) { BuyHouse buyHouse = new BuyHouseImpl(); buyHouse.buyHosue(); BuyHouseProxy buyHouseProxy = new BuyHouseProxy(buyHouse); buyHouseProxy.buyHosue(); } }
JDK动态代理
动态代理的代理对象是是在程序运行时通过java的反射机制生成的。在动态代理里面,我们并不需要为每一个服务都创建代理类,我们只需要实现一个代理处理器,真正的对象由JDK动态的为我们创建。
动态处理器:
public class DynamicProxyHandler implements InvocationHandler{ private Object object; public DynamicProxyHandler(Object object) { // TODO Auto-generated constructor stub this.object=object; } @Override public Object invoke(Object proxy, Method method, Object[] args) throws Throwable { System.out.println("before method"); Object res=method.invoke(object, args); System.out.println("after method"); return res; } }
测试代码:
public class DynamicProxyTest { public static void main(String[] args) { BuyHouse buyHouse=new BuyHouseImpl(); BuyHouse proxyBuyHouse=(BuyHouse) Proxy.newProxyInstance(BuyHouse.class.getClassLoader(), new Class[]{BuyHouse.class}, new DynamicProxyHandler(buyHouse)); proxyBuyHouse.buyHouse(); } }
也可以直接将动态处理器写在Proxy.newProxyInstance方法中,这样代码比较简洁。
public class DynamicProxyTest { public static void main(String[] args) { final BuyHouse buyHouse=new BuyHouseImpl(); BuyHouse proxyBuyHouse=(BuyHouse) Proxy.newProxyInstance(BuyHouse.class.getClassLoader(), new Class[]{BuyHouse.class}, new InvocationHandler(){@Override public Object invoke(Object proxy, Method method, Object[] args) throws Throwable { // TODO Auto-generated method stub System.out.println("before buy house"); Object resObject=method.invoke(buyHouse, args); System.out.println("after buy house"); return resObject; }}); proxyBuyHouse.buyHouse(); } }
JDK实现代理,只需要实现newProxyInstance方法,该方法需要接受三个参数。
public static Object newProxyInstance(ClassLoader loader, Class<?>[] interfaces, InvocationHandler h)
loader:目标类的类加载器。
interfaces:目标类实现的接口集合,可以是一个,可以是多个。
invocationHandler:动态处理器,当调用目标类方法时会调用动态处理器的invoke方法,这时invoke(Object proxy, Method method,Object[] args)的参数proxy是代理类,method是调用的代理类对象方法,args则是方法的参数。
JDK动态代理的目标类一定要实现接口,否则不能用JDK的动态代理。
CGlib动态代理
创建CGlib代理类:
public class CglibProxy implements MethodInterceptor { private Object target; public Object getInstance(final Object target) { this.target = target; Enhancer enhancer = new Enhancer(); enhancer.setSuperclass(this.target.getClass()); enhancer.setCallback(this); return enhancer.create(); } public Object intercept(Object object, Method method, Object[] args, MethodProxy methodProxy) throws Throwable { System.out.println("买房前准备"); Object result = methodProxy.invoke(object, args); System.out.println("买房后装修"); return result; } }
创建测试类:
public class CglibProxyTest { public static void main(String[] args){ BuyHouse buyHouse = new BuyHouseImpl(); CglibProxy cglibProxy = new CglibProxy(); BuyHouseImpl buyHouseCglibProxy = (BuyHouseImpl) cglibProxy.getInstance(buyHouse); buyHouseCglibProxy.buyHosue(); } }
这里CGLib和JDK动态代理实现代理模式的方法不同,JDK动态代理实现是通过代理类和目标类实现相同的接口实现的,而CGLib是通过让代理类继承目标类来实现的。
CGLIB创建的动态代理对象比JDK创建的动态代理对象的性能更高,但是CGLIB创建代理对象时所花费的时间却比JDK多得多。所以对于单例的对象,因为无需频繁创建对象,用CGLIB合适,反之使用JDK方式要更为合适一些。同时由于CGLib由于是采用动态创建子类的方法,对于final修饰的方法无法进行代理。
原文:https://www.cnblogs.com/xiaobaituyun/p/10779611.html