代理

来源:互联网 发布:csol游戏数据异常封号 编辑:程序博客网 时间:2024/06/08 03:20

1.代理模式的作用是:为其他对象提供一种代理以控制对这个对象的访问(在某些情况下,一个客户不想或者不能直接引用另一个对象,而代理对象可以在客户端和目标对象之间起到中介的作用).

2.代理模式一般涉及到的角色有:(1)抽象角色:声明真实对象和代理对象的共同接口; (2)真实角色:代理角色所代表的真实对象,是我们最终要引用的对象; (3)代理角色:代理对象角色内部含有对真实对象的引用,从而可以操作真实对象,同时代理对象提供与真实对象相同的接口以便在任何时刻都能代替真实对象。同时,代理对象可以在执行真实对象操作时,附加其他的操作,相当于对真实对象进行封装

例子1:实现一个静态代理

  1. package edu.hust.proxy;
  2. //抽象角色
  3. public interface RoleDAO {
  4.     public void doSomething();
  5. }
  1. package edu.hust.proxy;
  2. //真实角色
  3. public class RoleDAOImpl implements RoleDAO {
  4.     public void doSomething() {
  5.         System.out.println("我可以做一些事情, 比如:数据的增删改查.....");
  6.     }
  7.     
  8. }
  1. package edu.hust.proxy;
  2. //代理角色
  3. public class RoleProxy implements RoleDAO {
  4.     //将真实角色作为代理角色的属性。组合理论, 表示引用.
  5.     private RoleDAOImpl roleDAOImpl;
  6.     
  7.     //该方法封装了真实角色的doSomething()方法
  8.     public void doSomething() {
  9.         if (roleDAOImpl == null) {
  10.             roleDAOImpl = new RoleDAOImpl();
  11.         }
  12.         
  13.         this.before();
  14.         
  15.         roleDAOImpl.doSomething();
  16.         
  17.         this.after();
  18.         
  19.     }
  20.     public void before() {
  21.         System.out.println("做事情之前, 我想我可以做点其他事情:such as --> 判断一下权限..");
  22.     }
  23.     
  24.     public void after() {
  25.         System.out.println("做事情之后, 我想我还可以做点其他事情:such as --> 写个日志..");
  26.     }
  27.     
  28. }
  1. package edu.hust.proxy;
  2. //客户端
  3. public class Client {
  4.     public static void main(String[] args) {
  5.         new RoleProxy().doSomething();
  6.     }
  7. }

但是,如果要按照上述的方法使用代理模式,那么真实角色必须是事先已经存在的,并将其作为代理对象的内部属性。但是实际使用时,一个真实角色必须对应一个 代理角色,如果大量使用会导致类的急剧膨胀;此外,如果事先并不知道真实角色,该如何使用代理呢?这个问题可以通过Java的动态代理类来解决.

3.动态代理

所谓Dynamic Proxy是这样一种class:它是在运行时生成的class,在生成它时你必须提供一组interface给它,然后该class就宣称它实现了这些interface。你当然可以把该class的实例当作这些interface中的任何一个来用。当然,这个Dynamic Proxy其实就是一个Proxy,它不会替你作实质性的工作,在生成它的实例时你必须提供一个handler,由它接管实际的工作。Java动态代理类位于java.lang.reflect包下,一般主要涉及到以下两个类:

(1)Interface InvocationHandler:该接口中仅定义了一个方法

public object invoke(Object obj, Method method, Object[] args):在实际使用时,第一个参数obj一般是指代理类,method是被代理的方法,如上例中的doSomething(),args为该方法的参数数组。 这个抽象方法在代理类中动态实现。

(2)Proxy,该类即为动态代理类,作用类似于上例中的RoleProxy,其中主要包含以下内容

protected Proxy(InvocationHandler h):构造函数,用于给内部的h赋值。

static Class getProxyClass (ClassLoader loader, Class[] interfaces):获得一个代理类,其中loader是类装载器,interfaces是真实类所拥有的全部接口的数组。

static Object newProxyInstance(ClassLoader loader, Class[] interfaces, InvocationHandler h):返回代理类的一个实例。

例子2:先实现一个最基本的动态代理

  1. package edu.hust.dynaProxy;
  2. //抽象角色
  3. public interface RoleDAO {
  4.     public void doSomething();
  5. }
  1. package edu.hust.dynaProxy;
  2. import edu.hust.dynaProxy.RoleDAO;
  3. public class RoleDAOImpl1 implements RoleDAO {
  4.     public void doSomething() {
  5.         System.out.println("吃牛肉。。。。。");
  6.     }
  7. }
  1. package edu.hust.dynaProxy;
  2. import edu.hust.dynaProxy.RoleDAO;
  3. public class RoleDAOImpl2 implements RoleDAO {
  4.     public void doSomething() {
  5.         System.out.println("吃蔬菜。。。");
  6.     }
  7. }
  1. package edu.hust.dynaProxy;
  2. import java.lang.reflect.InvocationHandler;
  3. import java.lang.reflect.Method;
  4. /*
  5.  * 代理角色
  6.  * 
  7.  * 通过构造函数RoleProxy(Object obj)将 被代理类对象 传递进来.
  8.  * 
  9.  * */
  10. public class RoleProxyBefore implements InvocationHandler {
  11.     private Object obj;
  12.     
  13.     public RoleProxyBefore() {}
  14.     
  15.     public RoleProxyBefore(Object obj) {
  16.         this.obj = obj;
  17.     }
  18.     
  19.     public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
  20.         this.before();
  21.         
  22.         Object object = method.invoke(obj, args);
  23.         
  24.         return object;
  25.     }
  26.     
  27.     public void before() {
  28.         System.out.println("做事情之前, 我想我可以做点其他事情:such as --> 判断一下权限..");
  29.     }
  30. }
  1. package edu.hust.dynaProxy;
  2. import java.lang.reflect.InvocationHandler;
  3. import java.lang.reflect.Method;
  4. /*
  5.  * 代理角色
  6.  * 
  7.  * 通过构造函数RoleProxy(Object obj)将 被代理类对象 传递进来.
  8.  * 
  9.  * */
  10. public class RoleProxyAfter implements InvocationHandler {
  11.     private Object obj;
  12.     
  13.     public RoleProxyAfter() {}
  14.     
  15.     public RoleProxyAfter(Object obj) {
  16.         this.obj = obj;
  17.     }
  18.     
  19.     public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
  20.         Object object = method.invoke(obj, args);
  21.         
  22.         this.after();
  23.         
  24.         return object;
  25.     }
  26.     
  27.     public void after() {
  28.         System.out.println("做事情之后, 我想我还可以做点其他事情:such as --> 写个日志..");
  29.     }
  30. }

客户端类有两个版本:Client1.java是最原始的,Client2.java是用工厂重构过的

  1. package edu.hust.dynaProxy;
  2. import java.lang.reflect.InvocationHandler;
  3. import java.lang.reflect.Proxy;
  4. public class Client1 {
  5.     public static void main(String[] args) {
  6.         //获得 被代理类 的对象
  7.         RoleDAO roleDAO1 = new RoleDAOImpl1();
  8.         RoleDAO roleDAO2 = new RoleDAOImpl2();
  9.         
  10.         //获得 代理类 的对象
  11.         //(1) 取得Class实例
  12.         Class<?> cls_roleDAO1 = roleDAO1.getClass();
  13.         Class<?> cls_roleDAO2 = roleDAO2.getClass();
  14.         //(2) 针对每个 被代理类对象, 初始化其对应的动态代理类
  15.         InvocationHandler h1_before = new RoleProxyBefore(roleDAO1);
  16.         InvocationHandler h2_before = new RoleProxyBefore(roleDAO2);
  17.         InvocationHandler h1_after = new RoleProxyAfter(roleDAO1);
  18.         InvocationHandler h2_after = new RoleProxyAfter(roleDAO2);
  19.         //(3) 获得 代理类 的对象
  20.         RoleDAO roleDAO1_before = (RoleDAO) Proxy.newProxyInstance(cls_roleDAO1.getClassLoader(), cls_roleDAO1.getInterfaces(), h1_before);
  21.         RoleDAO roleDAO2_before = (RoleDAO) Proxy.newProxyInstance(cls_roleDAO2.getClassLoader(), cls_roleDAO2.getInterfaces(), h2_before);
  22.         RoleDAO roleDAO1_after = (RoleDAO) Proxy.newProxyInstance(cls_roleDAO1.getClassLoader(), cls_roleDAO1.getInterfaces(), h1_after);
  23.         RoleDAO roleDAO2_after = (RoleDAO) Proxy.newProxyInstance(cls_roleDAO2.getClassLoader(), cls_roleDAO2.getInterfaces(), h2_after);
  24.         
  25.         //直接获得 代理类 的对象:相比上面的6行代码(只是获得代理类对象的分解动作), 我更喜欢用一下被注释的方式获得代理类对象。
  26.         //RoleDAO roleDAO1_before = (RoleDAO) Proxy.newProxyInstance(roleDAO1.getClass().getClassLoader(), roleDAO1.getClass().getInterfaces(), new RoleProxyBefore(roleDAO1));
  27.         //RoleDAO roleDAO2_before = (RoleDAO) Proxy.newProxyInstance(roleDAO2.getClass().getClassLoader(), roleDAO2.getClass().getInterfaces(), new RoleProxyBefore(roleDAO2));
  28.         //RoleDAO roleDAO1_after = (RoleDAO) Proxy.newProxyInstance(roleDAO1.getClass().getClassLoader(), roleDAO1.getClass().getInterfaces(), new RoleProxyAfter(roleDAO1));
  29.         //RoleDAO roleDAO2_after = (RoleDAO) Proxy.newProxyInstance(roleDAO2.getClass().getClassLoader(), roleDAO2.getClass().getInterfaces(), new RoleProxyAfter(roleDAO2));
  30.         
  31.         roleDAO1_before.doSomething();
  32.         System.out.println("===========================================================");
  33.         roleDAO2_before.doSomething();
  34.         System.out.println("===========================================================");
  35.         roleDAO1_after.doSomething();
  36.         System.out.println("============================================================");
  37.         roleDAO2_after.doSomething();
  38.     }
  39.     /*
  40.      * 输出结果:
  41.      * 
  42.      * 做事情之前, 我想我可以做点其他事情:such as --> 判断一下权限..
  43.      * 吃牛肉。。。。。
  44.      * ===========================================================
  45.      * 做事情之前, 我想我可以做点其他事情:such as --> 判断一下权限..
  46.      * 吃蔬菜。。。
  47.      * ===========================================================
  48.      * 吃牛肉。。。。。
  49.      * 做事情之后, 我想我还可以做点其他事情:such as --> 写个日志..
  50.      * ============================================================
  51.      * 吃蔬菜。。。
  52.      * 做事情之后, 我想我还可以做点其他事情:such as --> 写个日志..
  53.      * 
  54.      * */
  55. }
  1. package edu.hust.dynaProxy;
  2. import java.lang.reflect.InvocationHandler;
  3. import java.lang.reflect.Proxy;
  4. public class Client2 {
  5.     
  6.     /* 
  7.      * 改进:建立代理工厂, 进一步简化代码
  8.      * param1:被代理类  的对象
  9.      * param2:代理类       的名字
  10.      * 
  11.      * 最后:这个重构方法常放与代理类(代理类也算是一种工具类)中, 这样的代码更加优雅..
  12.      */
  13.     public static Object proxyFactory(Object obj, String name_classOfProxy) {       
  14.         InvocationHandler h = null;
  15.         if ("RoleProxyBefore".equals(name_classOfProxy)) h = new RoleProxyBefore(obj);
  16.         else if ("RoleProxyAfter".equals(name_classOfProxy)) h = new RoleProxyAfter(obj);
  17.         
  18.         return Proxy.newProxyInstance(obj.getClass().getClassLoader(), obj.getClass().getInterfaces(), h);
  19.     }
  20.     
  21.     public static void main(String[] args) {
  22.         //获得 被代理类 的对象
  23.         RoleDAO roleDAO1 = new RoleDAOImpl1();
  24.         RoleDAO roleDAO2 = new RoleDAOImpl2();
  25.         
  26.         //获得 代理类 的对象
  27.         RoleDAO roleDAO1_before = (RoleDAO) Client2.proxyFactory(roleDAO1, "RoleProxyBefore");
  28.         RoleDAO roleDAO2_before = (RoleDAO) Client2.proxyFactory(roleDAO2, "RoleProxyBefore");
  29.         RoleDAO roleDAO1_after = (RoleDAO) Client2.proxyFactory(roleDAO1, "RoleProxyAfter");
  30.         RoleDAO roleDAO2_after = (RoleDAO) Client2.proxyFactory(roleDAO2, "RoleProxyAfter");
  31.         
  32.         roleDAO1_before.doSomething();
  33.         System.out.println("===========================================================");
  34.         roleDAO2_before.doSomething();
  35.         System.out.println("===========================================================");
  36.         roleDAO1_after.doSomething();
  37.         System.out.println("============================================================");
  38.         roleDAO2_after.doSomething();
  39.     }
  40.     /*
  41.      * 输出结果:
  42.      * 
  43.      * 做事情之前, 我想我可以做点其他事情:such as --> 判断一下权限..
  44.      * 吃牛肉。。。。。
  45.      * ===========================================================
  46.      * 做事情之前, 我想我可以做点其他事情:such as --> 判断一下权限..
  47.      * 吃蔬菜。。。
  48.      * ===========================================================
  49.      * 吃牛肉。。。。。
  50.      * 做事情之后, 我想我还可以做点其他事情:such as --> 写个日志..
  51.      * ============================================================
  52.      * 吃蔬菜。。。
  53.      * 做事情之后, 我想我还可以做点其他事情:such as --> 写个日志..
  54.      * 
  55.      * */
  56. }

例子3:对例子2进行重构,其中RoleDAO.java、RoleDAOImpl1.java、RoleDAOImpl2.java无需改动。只要将两个代理类和Client类重构就ok了。

  1. package edu.hust.dynaProxy2;
  2. import java.lang.reflect.InvocationHandler;
  3. import java.lang.reflect.Method;
  4. import java.lang.reflect.Proxy;
  5. /*
  6.  * 代理角色
  7.  * 
  8.  * 通过构造函数RoleProxy(Object obj)将 被代理类对象 传递进来.
  9.  * 
  10.  * */
  11. public class RoleProxyBefore implements InvocationHandler {
  12.     private Object obj;
  13.     
  14.     public Object bind(Object obj) {
  15.         this.obj = obj;
  16.         return Proxy.newProxyInstance(obj.getClass().getClassLoader(), obj.getClass().getInterfaces(), this);
  17.     }
  18.     
  19.     public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
  20.         this.before();
  21.         
  22.         Object object = method.invoke(obj, args);
  23.         
  24.         return object;
  25.     }
  26.     
  27.     public void before() {
  28.         System.out.println("做事情之前, 我想我可以做点其他事情:such as --> 判断一下权限..");
  29.     }
  30. }
  1. package edu.hust.dynaProxy2;
  2. import java.lang.reflect.InvocationHandler;
  3. import java.lang.reflect.Method;
  4. import java.lang.reflect.Proxy;
  5. /*
  6.  * 代理角色
  7.  * 
  8.  * 通过构造函数RoleProxy(Object obj)将 被代理类对象 传递进来.
  9.  * 
  10.  * */
  11. public class RoleProxyAfter implements InvocationHandler {
  12.     private Object obj;
  13.     
  14.     public Object bind(Object obj) {
  15.         this.obj = obj;
  16.         return Proxy.newProxyInstance(obj.getClass().getClassLoader(), obj.getClass().getInterfaces(), this);
  17.     }
  18.     
  19.     public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
  20.         Object object = method.invoke(obj, args);
  21.         
  22.         this.after();
  23.         
  24.         return object;
  25.     }
  26.     
  27.     public void after() {
  28.         System.out.println("做事情之后, 我想我还可以做点其他事情:such as --> 写个日志..");
  29.     }
  30. }
  1. package edu.hust.dynaProxy2;
  2. public class Client {
  3.     public static void main(String[] args) {
  4.         //获得 代理类 的对象
  5.         RoleProxyBefore roleProxyBefore = new RoleProxyBefore();
  6.         RoleProxyAfter roleProxyAfter = new RoleProxyAfter();
  7.         
  8.         //获得 代理
  9.         RoleDAO roleDAO1_before = (RoleDAO) roleProxyBefore.bind(new RoleDAOImpl1());
  10.         RoleDAO roleDAO2_before = (RoleDAO) roleProxyBefore.bind(new RoleDAOImpl2());
  11.         RoleDAO roleDAO1_after = (RoleDAO) roleProxyAfter.bind(new RoleDAOImpl1());
  12.         RoleDAO roleDAO2_after = (RoleDAO) roleProxyAfter.bind(new RoleDAOImpl2());
  13.         
  14.         roleDAO1_before.doSomething();
  15.         System.out.println("===========================================================");
  16.         roleDAO2_before.doSomething();
  17.         System.out.println("===========================================================");
  18.         roleDAO1_after.doSomething();
  19.         System.out.println("============================================================");
  20.         roleDAO2_after.doSomething();
  21.     }
  22.     /*
  23.      * 输出结果:
  24.      * 
  25.      * 做事情之前, 我想我可以做点其他事情:such as --> 判断一下权限..
  26.      * 吃牛肉。。。。。
  27.      * ===========================================================
  28.      * 做事情之前, 我想我可以做点其他事情:such as --> 判断一下权限..
  29.      * 吃蔬菜。。。
  30.      * ===========================================================
  31.      * 吃牛肉。。。。。
  32.      * 做事情之后, 我想我还可以做点其他事情:such as --> 写个日志..
  33.      * ============================================================
  34.      * 吃蔬菜。。。
  35.      * 做事情之后, 我想我还可以做点其他事情:such as --> 写个日志..
  36.      * 
  37.      * */
  38. }

总结:通过这种方式,(1)被代理的对象(RoleDAOImpl1或RoleDAOImpl2)可以在运行时动态改变;(2)需要控制的接口(RoleDAO)可以在运行时改变;(3)控制的方式(RoleProxyBefore或RoleProxyAfter)也可以动态改变,从而实现了非常灵活的动态代理关系。