代理模式

本文介绍了代理模式的概念及其在Java中的应用,包括静态代理和动态代理两种形式,并提供了详细的代码示例。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

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

    代理模式一般涉及到三个角色:

    抽象角色:声明真实对象和代理对象的共同接口;

    代理角色:代理对象角色内部含有对真实对象的引用,从而可以操作真实对象,同时代理对象提供与真实对象相同的接口以便在任何时刻都能代替真实对象。同时,代理对象可以在执行真实对象操作时,附加其他的操作,相当于对真实对象进行封装。
 
    真实角色:代理角色所代表的真实对象,是我们最终要引用的对象。<o:p></o:p>
以下以《Java与模式》中的示例为例:
        抽象角色
        abstract public class Subject { 
             abstract public void request(); 
        }

       真实角色:实现了Subject的request()方法。 
       public class RealSubject extends Subject {
            public RealSubject() {
 
           }

           public void request() { 
                 System.out.println("From real subject."); 
           }  
      }

      代理角色
       public class ProxySubject extends Subject { 

             private RealSubject realSubject; //以真实角色作为代理角色的属性 
             public ProxySubject() { 
             } 
             public void request()  { //该方法封装了真实对象的request方法
                   preRequest(); 
                   if( realSubject == null ) { 
                         realSubject = new RealSubject(); 
                   } 
                    realSubject.request(); //此处执行真实对象的request方法 
                    postRequest(); 
             }

             private void preRequest() { 
                    //something you want to do before requesting 
             } 

             private void postRequest() { 
                  //something you want to do after requesting 
             }
       } 
     客户端调用:     
     Subject sub = new  ProxySubject(); 
         sub.request(); 

2.动态代理
      Java动态代理类位于Java.lang.reflect包下,一般主要涉及到以下两个类:
      (1). Interface InvocationHandler:该接口中仅定义了一个方法Object:invoke(Object obj,Method method, Object[] args)。
      (2).Proxy:该类即为动态代理类,作用类似于上例中的ProxySubject,其中主要包含以下内容:
              Protected Proxy(InvocationHandler h):构造函数。
              Static Class getProxyClass (ClassLoader loader, Class[] interfaces):获得一个代理类,其中loader是类装载器,interfaces是真实类所拥有的全部接口的数组。
              Static Object newProxyInstance(ClassLoader loader, Class[] interfaces, InvocationHandler h)

    
      抽象角色(之前是抽象类,此处应改为接口):
      public interface Subject  { 
            public void request();
      } 

     具体角色RealSubject:实现了Subject接口的request()方法。 
     public class RealSubject implements Subject { 
                 public RealSubject() {
       }
       public void request() { 
          System.out.println("From real subject."); 
       } 
   }
 
   代理角色
   import java.lang.reflect.Method; 
   import java.lang.reflect.InvocationHandler; 

   public class DynamicSubject implements InvocationHandler { 
          private Object sub;
          public DynamicSubject(Object sub){
 
                 this.sub = sub;
          } 


          public Object invoke(Object proxy, Method method, Object[] args) throws Throwable { 
                 System.out.println("before calling " + method); 
                 method.invoke(sub,args); 
                 System.out.println("after calling " + method); 
                 return null; 
          }
    }

    该代理类的内部属性为Object类,实际使用时通过该类的构造函数DynamicSubject(Object obj)对其赋值;此外,在该类还实现了invoke方法,该方法中的
    method.invoke(sub,args);
    其实就是调用被代理对象的将要被执行的方法,方法参数sub是实际的被代理对象,args为执行被代理对象相应操作所需的参数。通过动态代理类,我们可以在调用之前或之后执行一些相关操作。

    客户端:
    import java.lang.reflect.InvocationHandler;
    import java.lang.reflect.Proxy;
    import java.lang.reflect.Constructor;
    import java.lang.reflect.Method; 
   
    public class Client {
          static public void main(String[] args) throws Throwable { 
                 RealSubject rs = new RealSubject(); //在这里指定被代理类 
                 InvocationHandler ds = new DynamicSubject(rs); //初始化代理类 
                 Class cls = rs.getClass(); 

                 //以下是分解步骤 
                 /* 
                 Class c = Proxy.getProxyClass(cls.getClassLoader(),cls.getInterfaces()) ; 
                 Constructor ct=c.getConstructor(new Class[]{InvocationHandler.class}); 
                 Subject subject =(Subject) ct.newInstance(new Object[]{ds}); 
                 */ 

                 //以下是一次性生成
 
                Subject subject = (Subject) Proxy.newProxyInstance(cls.getClassLoader(), cls.getInterfaces(), ds ); 
                subject.request(); 
        }

         通过这种方式,被代理的对象(RealSubject)可以在运行时动态改变,需要控制的接口(Subject接口)可以在运行时改变,控制的方式(DynamicSubject类)也可以动态改变,从而实现了非常灵活的动态代理关系。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值