`
wlhok
  • 浏览: 56878 次
  • 性别: Icon_minigender_1
  • 来自: 杭州
社区版块
存档分类
最新评论

代理模式[转][整理]

    博客分类:
  • Java
阅读更多
代理模式

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

代理模式一般涉及到的角色有:

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

以下以《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  
    } 
}

// 使用场景
public class ProxyTest { 
    public static void main(String[] args) { 
        Subject sub=new ProxySubject(); 
        Sub.request();
    }
}

以上的代码,创建一个新的类,实现抽象接口,并对真实角色进行封装,从而达到代理的效果。
但是,这样的实现,1、真实角色必须已经存在,并将其作为代理角色的内部属性。2、在使用时,必须一个真实角色对应一个代理角色。如果大量使用会导致类的急剧膨胀。
此外,如果实现不知道真实对象,也无法使用这样的代理方式。

动态代理

使用java.lang.reflect.Porxy 和 java.lang.reflect.InvocationHandler接口实现代理模式。

    * Interface InvocationHandler :是代理角色实现的接口。提供一个方法

      Object invoke(Object proxy, Method method, Object[] args)  throws Throwable


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

    * java.lang.reflect.Proxy :该类即为动态代理类,作用类似于上例中的ProxySubject,提供用于创建动态代理类和实例的静态方法,它还是由这些方法创建的所有动态代理类的超类。
          o Protected Proxy(InvocationHandler h):构造函数,估计用于给内部的h赋值。
          o Static Class getProxyClass (ClassLoader loader, Class[] interfaces):获得一个代理类,其中loader是类装载器,interfaces是真实类所拥有的全部接口的数组。它的主要方法有:
          o Static Object newProxyInstance(ClassLoader loader, Class[] interfaces, InvocationHandler h):返回代理类的一个实例,返回后的代理类可以当作被代理类使用(可使用被代理类的在Subject接口中声明过的方法)。

所谓Dynamic Proxy是这样一种class:它是在运行时生成的class,在生成它时你必须提供一组interface给它,然后该class就宣称它实现了这些 interface。你当然可以把该class的实例当作这些interface中的任何一个来用。当然啦,这个Dynamic Proxy其实就是一个Proxy,它不会替你作实质性的工作,在生成它的实例时你必须提供一个handler,由它接管实际的工作。

将上面的例子,修改为被代理的类如果实现接口,那么使用JDK的代理模式,设计到Proxy,InvocationHandler。
如果被代理的类没有实现接口,那么使用CGLIB来实现代理。

public interface Subject{       
    public void request(); 
}

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

public class ProxySubject implements InvocationHandler {

    private Object target;

    public ProxySubject(Object obj) {
        this.target = obj;
    }

    public Object invoke(Object proxy, Method method, Object[] args)
            throws Throwable {
        System.out.println("before calling " + method);
        Object result = method.invoke(this.target, args);
        System.out.println("after calling " + method);
        return result;proxyRs
    }

    /**
     * 提供一个静态方法,获取动态代理实例
     */
    public static Object createProxyInstance(Object obj) {
        Assert.notNull(obj);
        Class<?> clz = obj.getClass();
        /*
          以下是分解步骤
          Class c = Proxy.getProxyClass(clz.getClassLoader(),clz.getInterfaces());
          Constructor ct=c.getConstructor(new Class[]{InvocationHandler.class});
          Subject subject =(Subject) ct.newInstance(new Object[]{new MyInvocationHandler(obj)});
         */

         //以下是一次性生成
        return Proxy.newProxyInstance(clz.getClassLoader(), clz.getInterfaces(), new DynamicSubject(obj));
    }
}


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

// 使用场景
public class ProxyTest {
    public static void main(String[] args) throws Throwable {
        RealSubject rs = new RealSubject(); // 在这里指定被代理类
        Subject subject = (Subject) DynamicSubject.createProxyInstance(rs);// 获取代理类
        subject.request();
    }
}


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

Cglib实现代理模式

被代理的类如果实现接口,那么使用JDK的代理模式,设计到Proxy,InvocationHandler。
如果被代理的类没有实现接口,那么使用CGLIB来实现代理。

还是用上面的例子,去掉接口类。

//真实角色:实现了Subject的request()方法。 
public class RealSubject { 
    public RealSubject(){  
    } 
    public void request(){ 
        System.out.println("From real subject."); 
    } 
}

public class CGLIBProxyFactory implements MethodInterceptor {

    private Object target;

    public CGLIBProxyFactory(Object obj) {
        this.target = obj;
    }

    public static Object creatNewProxyInstance(Object obj) {
        CGLIBProxyFactory proxy = new CGLIBProxyFactory(obj);

        Enhancer enhancer = new Enhancer();
        enhancer.setSuperclass(proxy.getTarget().getClass()); ////代理类会生产一个被代理类的子类,final方法除外
        enhancer.setCallback(proxy);//将CGLIBProxyFactory实例设为回调对象,必须实现接口 MehtoInterceptor
        return enhancer.create();
    }

    //代理对象被回调的时候调用此方法
    @Override
    public Object intercept(Object proxy, Method method, Object[] args,
            MethodProxy methodProxy) throws Throwable {

        System.out.println("before calling " + method);
        Object result = methodProxy.invoke(this.target, args);
        System.out.println("after calling " + method);

        return result;
    }

    public Object getTarget() {
        return target;
    }

    public void setTarget(Object target) {
        this.target = target;
    }
}

// 使用场景
public class ProxyTest {
    public static void main(String[] args) throws Throwable {
        RealSubject rs = new RealSubject();
        RealSubject proxyRs = (RealSubject) CGLIBProxyFactory.creatNewProxyInstance(rs);
        proxyRs.request();
    }
}


可以看到,cglib在使用场景里的代码,与使用反射的代码,基本上是一样的。只是cglib可以直接实现对一个实例的代理,而用反射方式,必须有声明一个接口。

CgLib 就可以不用接口,它底层调用 asm 动态生成一个代理类去覆盖父类中非 final 的方法,然后实现 MethodInterceptor 接口的 intercept 方法,这样以后直接调用重写的方法,比 JDK 要快。

加载 cglib 消耗时间比直接 jdk 反射时间长,开发的过程中,如果是反复动态生成新的代理类推荐用 jdk 自身的反射,反之用 cglib.

简单的做了一个测试,使用上面的代码,循环10000次,使用反射方式创建代理类需要402毫秒;使用cglib创建代理类需要1072毫秒。
分享到:
评论

相关推荐

    根据《JAVA与设计模式》整理的笔记及示例代码

    1. 代理模式:为其他对象提供一种代理以控制对这个对象的访问。 2. 组合模式:将对象组合成树形结构以表示“部分-整体”的层次结构。 3. 外观模式:提供一个统一的接口,用来访问子系统中的多个接口。 4. 桥接模式:...

    OO中对于23种设计模式的整理

    OO 中的 23 种设计模式的整理 ...代理模式可以并不知道真正的被代理对象,而仅仅持有一个被代理对象的接口,这时候代理对象不能够创建被代理对象,被代理对象必须有系统的其他角色代为创建并传入。

    C#设计模式-整理收藏(吕震宇 设计模式速成经典示例)

    结构型模式包括适配器模式、装饰器模式、代理模式、桥接模式、组合模式、外观模式和享元模式,它们关注如何组合和构建对象,以实现更复杂的结构。行为型模式如观察者模式、命令模式、策略模式、模板方法模式、访问者...

    Java中23种设计模式(个人整理实用).doc

    代理模式是一种结构型设计模式,它提供了一种将对象的访问控制的方式,解决了对象访问控制的问题。 这些设计模式都是 Java 编程中常用的解决方案,了解和掌握这些模式可以帮助开发者编写更加灵活、可维护的代码。

    windows下apache+tomcat反向代理模式配置负载均衡(091124整理).rar

    windows下apache+tomcat反向代理模式配置负载均衡

    Java设计模式整理

    - 代理模式:为其他对象提供一种代理以控制对这个对象的访问。 - 模板方法模式:定义一个操作中的算法骨架,而将一些步骤延迟到子类中。 3. **行为型模式**:行为型模式关注对象之间的责任分配和交互方式。包括:...

    设计模式 资料整理.rar

    7. **代理模式(Proxy)**:为其他对象提供一种代理以控制对这个对象的访问,可以用来控制权限、缓存、事务处理等。 8. **适配器模式(Adapter)**:将两个不兼容的接口连接起来,使得原本不能一起工作的类可以协同...

    w3school 设计模式教程 飞龙整理 20141001

    整体的层次结构)、装饰器模式(动态地给对象添加职责)、外观模式(提供一个统一的接口来访问子系统)、享元模式(减少大量相似对象的创建)、代理模式(在对象访问控制或远程访问中常用)。 3. 行为型模式:关注...

    Java开发中的23种设计模式代码整理

    - 代理模式:为其他对象提供一种代理以控制对这个对象的访问。 - 享元模式:运用共享技术有效地支持大量细粒度的对象。 3. 行为型设计模式: - 责任链模式:将请求沿着处理者链传递,直到有处理者处理请求。 - ...

    设计模式之五种PPT整理

    代理模式是一种结构型设计模式,它创建了一个代理对象来控制对原对象的访问。代理对象通常在原对象基础上添加额外功能,如日志记录、权限检查、缓存等。这种模式使得我们可以在不修改原有对象的基础上,增加新功能...

    C#设计模式资料整理

    10. 代理模式(Proxy Pattern):为其他对象提供一种代理以控制对这个对象的访问。在C#中,可以使用虚拟代理(延迟加载)或远程代理(控制对远程对象的访问)等不同形式。 这些设计模式都是软件工程中的重要工具,...

    java设计模式整理版

    以下是对"java设计模式整理版"中可能包含的知识点的详细解释: 1. **单例模式**:保证一个类只有一个实例,并提供一个全局访问点。这种模式常用于配置管理、日志服务等场景。 2. **工厂模式**:提供一个创建对象的...

    TerryLee的设计模式,手工整理成的word文档,方便学习存档

    13. 代理模式(Proxy Pattern):为其他对象提供一种代理以控制对该对象的访问。代理模式可以在不影响目标对象的情况下,增加额外的功能或者控制访问。 这些设计模式的掌握对于C#和.NET开发者来说至关重要,它们...

    js设计模式之代理模式及订阅发布模式实例详解

    本文实例讲述了js设计模式之代理模式及订阅发布模式。分享给大家供大家参考,具体如下: 为啥将两种模式放在一起呢?因为这样文章比较长啊。 写博客的目的我觉得首要目的是整理自己的知识点,进而优化个人所得知识...

    JAVA设计模式整理(有部分参考代码)

    7. **代理模式**:为其他对象提供一种代理以控制对这个对象的访问,通常用于远程代理、虚拟代理和保护代理等场景。 8. **桥接模式**:将抽象部分与实现部分分离,使它们可以独立变化。 9. **组合模式**:表示部分-...

    关于自己学习设计模式的笔记整理

    该文档是自己在学习设计模式时整理的常用设计模式pdf文档,包括源码,包括装饰模式,代理模式,责任链模式,命令模式,解释器模式,迭代器模式,备忘录模式,观察者模式,工厂模式,建造者模式,适配器模式,桥梁...

    整理 - 设计模式(Design Patterns)- 整理 Word版本

    2. 结构型模式:处理类与对象的组合和封装,如适配器模式(Adapter)、装饰器模式(Decorator)、桥接模式(Bridge)、组合模式(Composite)、外观模式(Facade)、享元模式(Flyweight)和代理模式(Proxy)。...

Global site tag (gtag.js) - Google Analytics