`
tommy14101
  • 浏览: 4600 次
  • 性别: Icon_minigender_1
  • 来自: 大连
最近访客 更多访客>>
社区版块
存档分类
最新评论

代理模式

 
阅读更多

代理模式是常用的Java 设计模式,它的特征是代理类与委托类有同样的接口,如图1所示。代理类主要负责为委托类预处理消息、过滤消息、把消息转发给委托类,以及事后处理消息等。 代理类与委托类之间通常会存在关联关系,一个代理类的对象与一个委托类的对象关联,代理类的对象本身并不真正实现服务,而是通过调用委托类的对象的相关方 法,来提供特定的服务。


图1:Proxy模式

 

按照代理类的创建时期,代理类可分为两种。

  • 静态代理类:由程序员创建或由特定工具自动生成源代码,再对其编译。在程序运行前,代理类的.class文件就已经存在了。
  • 动态代理类:在程序运行时,运用反射机制动态创建而成。

 

静态代理类的实现:

 

1. 定义一个IService接口

Java代码  收藏代码
  1. package  cn.lettoo.proxy;  
  2.   
  3. public   interface  IService {  
  4.   
  5.     void  execute();  
  6. }  
package cn.lettoo.proxy;

public interface IService {

    void execute();
}

 

2. 具体的实现类:

Java代码  收藏代码
  1. package  cn.lettoo.proxy;  
  2.   
  3. public   class  PrintService  implements  IService {  
  4.   
  5.     public   void  execute() {  
  6.         System.out.println("The Print Service works." );  
  7.     }  
  8. }  
package cn.lettoo.proxy;

public class PrintService implements IService {

    public void execute() {
        System.out.println("The Print Service works.");
    }
}

 

3. 代理类:

Java代码  收藏代码
  1. package  cn.lettoo.proxy;  
  2.   
  3. public   class  PrintServiceProxy  implements  IService {  
  4.   
  5.     IService printSerivce;  
  6.   
  7.     public  PrintServiceProxy(IService service) {  
  8.         this .printSerivce = service;  
  9.     }  
  10.   
  11.     public   void  setPrintSerivce(IService printSerivce) {  
  12.         this .printSerivce = printSerivce;  
  13.     }  
  14.   
  15.     public   void  execute() {  
  16.         this .beforePrint();  
  17.         this .printSerivce.execute();  
  18.         this .afterPrint();  
  19.     }  
  20.   
  21.     private   void  beforePrint() {  
  22.         System.out.println("Before print." );  
  23.     }  
  24.   
  25.     private   void  afterPrint() {  
  26.         System.out.println("Before print." );  
  27.     }  
  28.   
  29. }  
package cn.lettoo.proxy;

public class PrintServiceProxy implements IService {

    IService printSerivce;

    public PrintServiceProxy(IService service) {
        this.printSerivce = service;
    }

    public void setPrintSerivce(IService printSerivce) {
        this.printSerivce = printSerivce;
    }

    public void execute() {
        this.beforePrint();
        this.printSerivce.execute();
        this.afterPrint();
    }

    private void beforePrint() {
        System.out.println("Before print.");
    }

    private void afterPrint() {
        System.out.println("Before print.");
    }

}

    代理类的execute()方法只是调用了被代理的Service的execute方法,被代理的Service通过构造函数或者set的方式被注入到代 理对象中。同时,代理对象也有一些自己的代理方法,如本例中在被代理类的execute()方法调用前后加上自己的方法。

 

4. 客户调用代码:

Java代码  收藏代码
  1. package  cn.lettoo.proxy;  
  2.   
  3. public   class  Client {  
  4.   
  5.     public   static   void  main(String[] args) {  
  6.         IService service = new  PrintService();  
  7.         IService proxy = new  PrintServiceProxy(service);  
  8.   
  9.         proxy.execute();  
  10.     }  
  11. }  
package cn.lettoo.proxy;

public class Client {

    public static void main(String[] args) {
        IService service = new PrintService();
        IService proxy = new PrintServiceProxy(service);

        proxy.execute();
    }
}

 

执行结果
Before print.
The Print Service works.
Before print.

 

 动态代理的实现:

 

与静态代理类对照的是动态代理类,动态代理类的字节码在程序运行时由Java反射机制动态生成,无需程序员手工编写它的源代码。动态代理类不仅简化 了编程工作,而且提高了软件系统的可扩展性,因为Java 反射机制可以生成任意类型的动态代理类。java.lang.reflect 包中的Proxy类和InvocationHandler 接口提供了生成动态代理类的能力。

 

Proxy类提供了创建动态代理类及其实例的静态方法。

Porxy有两种方式来生成代理对象:

方法1:

Java代码  收藏代码
  1. //创建InvocationHandler对象   
  2. InvocationHandler handler = newMyInvocationHandler(...);  
  3. //创建动态代理类,IService是被代理的接口   
  4. Class proxyClass = Proxy.getProxyClass(IService.class .getClassLoader(),  new  Class[] { IService. class  });  
  5. //创建动态代理类的实例   
  6. IService proxyService = (IService) proxyClass.getConstructor(new  Class[] { invocationHandler. class  }).newInstance( new  Object[] { handler });  
//创建InvocationHandler对象
InvocationHandler handler = newMyInvocationHandler(...);
//创建动态代理类,IService是被代理的接口
Class proxyClass = Proxy.getProxyClass(IService.class.getClassLoader(), new Class[] { IService.class });
//创建动态代理类的实例
IService proxyService = (IService) proxyClass.getConstructor(new Class[] { invocationHandler.class }).newInstance(new Object[] { handler });

方法2:

Java代码  收藏代码
  1. //创建InvocationHandler对象   
  2. InvocationHandler handler = newMyInvocationHandler(...);  
  3. //直接创建动态代理类的实例   
  4. IService serviceProxy = (IService) Proxy.newProxyInstance(IService.class .getClassLoader(),  new  Class[] { IService. class  },  
  5. handler);  
//创建InvocationHandler对象
InvocationHandler handler = newMyInvocationHandler(...);
//直接创建动态代理类的实例
IService serviceProxy = (IService) Proxy.newProxyInstance(IService.class.getClassLoader(), new Class[] { IService.class },
handler);

 

InvocationHandler 接口为方法调用接口,它声明了负责调用任意一个方法的invoke()方法:

Java代码  收藏代码
  1. Object invoke(Object proxy,Method method,Object[] args)  throws  Throwable  
Object invoke(Object proxy,Method method,Object[] args) throws Throwable

 

代码实现:

1. 写一个ServiceFactory,用于生成代理对象

Java代码  收藏代码
  1. package  cn.lettoo.proxy;  
  2.   
  3. import  java.lang.reflect.InvocationHandler;  
  4. import  java.lang.reflect.Method;  
  5. import  java.lang.reflect.Proxy;  
  6.   
  7. public   class  ServiceFactory {  
  8.   
  9.     public   static  IService getServiceProxy( final  IService service) {  
  10.         InvocationHandler handler = new  InvocationHandler() {  
  11.   
  12.             public  Object invoke(Object proxy, Method method, Object[] args)  throws  Throwable {  
  13.                 beforePrint();  
  14.                 Object result = method.invoke(service, args);  
  15.                 afterPrint();  
  16.                 return  result;  
  17.             }  
  18.         };  
  19.   
  20.         return  (IService) Proxy.newProxyInstance(IService. class .getClassLoader(),  new  Class[] { IService. class  },  
  21.                                                  handler);  
  22.     }  
  23.       
  24.     private   static   void  beforePrint() {  
  25.         System.out.println("Before print." );  
  26.     }  
  27.   
  28.     private   static   void  afterPrint() {  
  29.         System.out.println("Before print." );  
  30.     }  
  31.       
  32. }  
package cn.lettoo.proxy;

import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;

public class ServiceFactory {

    public static IService getServiceProxy(final IService service) {
        InvocationHandler handler = new InvocationHandler() {

            public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
                beforePrint();
                Object result = method.invoke(service, args);
                afterPrint();
                return result;
            }
        };

        return (IService) Proxy.newProxyInstance(IService.class.getClassLoader(), new Class[] { IService.class },
                                                 handler);
    }
    
    private static void beforePrint() {
        System.out.println("Before print.");
    }

    private static void afterPrint() {
        System.out.println("Before print.");
    }
    
}

 

2. 客户端代码:

Java代码  收藏代码
  1. package  cn.lettoo.proxy;  
  2.   
  3. public   class  Client {  
  4.   
  5.     public   static   void  main(String[] args) {  
  6.         IService service = new  PrintService();          
  7.         IService proxy = ServiceFactory.getServiceProxy(service);          
  8.         proxy.execute();  
  9.     }  
  10.   
  11. }  
package cn.lettoo.proxy;

public class Client {

    public static void main(String[] args) {
        IService service = new PrintService();        
        IService proxy = ServiceFactory.getServiceProxy(service);        
        proxy.execute();
    }

}

 

运行结果
Before print.
The Print Service works.
Before print.
 

 

    也许有人要问,采用动态代理有什么好处?那么我告诉你,你不需要为每个IService的实现类都去写一个Proxy类了(前提是代理方法是一样的),假如,我现在有另外一个IService的实现类如下:

Java代码  收藏代码
  1. package  cn.lettoo.proxy;  
  2.   
  3. public   class  AnotherService  implements  IService {  
  4.   
  5.     public   void  execute() {  
  6.         System.out.println("I am another service." );  
  7.     }  
  8.   
  9. }  
package cn.lettoo.proxy;

public class AnotherService implements IService {

    public void execute() {
        System.out.println("I am another service.");
    }

}

 

    现在客户端只要这样写:

Java代码  收藏代码
  1. public   static   void  main(String[] args) {  
  2.     IService service = new  AnotherService();          
  3.     IService proxy = ServiceFactory.getServiceProxy(service);          
  4.     proxy.execute();  
  5. }  
    public static void main(String[] args) {
        IService service = new AnotherService();        
        IService proxy = ServiceFactory.getServiceProxy(service);        
        proxy.execute();
    }

 

运行结果
Before print.
I am another service.
Before print.
 

    可以看到,不需要单独再为AnotherService写一个代理类了,动态代理类会帮助我们去实现的。

分享到:
评论

相关推荐

    代理模式的使用示例程序

    代理模式是一种设计模式,它在软件工程中扮演着重要的角色,允许我们为其他对象提供一个替代接口,以控制对原始对象的访问。这种模式的主要目的是为了增加灵活性、安全性或者在不修改原有对象的情况下,增强或扩展其...

    JAVA设计模式之代理模式实例

    代理模式是设计模式的一种,它提供了一种对目标对象进行增强或者控制访问的方式。在本实例中,我们将深入探讨Java中的代理模式及其应用。 代理模式的核心思想是为一个对象创建一个代理对象,这个代理对象在客户端和...

    设计模式之代理模式Proxy

    代理模式是设计模式中的一种结构型模式,它在对象交互中起到了中介的作用,允许通过代理对象来控制对原对象的访问。代理模式的核心思想是为一个对象提供一个替身,以便增加新的功能或者控制对原对象的访问。这种模式...

    Android设计模式之代理模式(Proxy Pattern)

    代理模式是设计模式的一种,它的主要目的是在不改变原有对象的基础上,为一个对象提供额外的功能或者控制对这个对象的访问。在Android开发中,代理模式的应用尤为常见,尤其在处理复杂的业务逻辑、网络请求、界面...

    android使用signalr代理模式和非代理模式

    SignalR提供了两种主要的工作模式:代理模式和非代理模式。这两种模式在实现上有所不同,各自具有优缺点,适用于不同的场景。 **1. 代理模式(Proxy Mode)** 在代理模式下,SignalR为每个Hub(服务端的业务逻辑...

    设计模式-代理模式

    代理模式是一种常用的设计模式,它在软件开发中扮演着重要的角色,特别是在iOS平台的应用程序设计中。代理模式的核心思想是为一个对象提供一个替身或代理,以控制对这个对象的访问。这种模式允许我们通过代理来间接...

    java中的三种代理模式

    在Java编程中,代理模式是一种常用的面向对象设计模式,它允许我们为一个对象提供一个代理以控制对该对象的访问。代理模式通常用于增加额外的功能,如日志、权限检查等,或者为了创建虚拟代理以提高性能。以下是Java...

    JAVA设计模式(代理模式)

    **Java设计模式——代理模式详解** 代理模式是软件设计模式中的一个重要组成部分,它在Java编程中扮演着举足轻重的角色。代理模式的核心思想是为一个对象提供一个替身,这个替身即代理对象,代理对象可以控制对原...

    代理模式小例子

    代理模式是一种设计模式,它在软件工程中扮演着重要的角色,允许我们为其他对象提供一个替代接口,以控制对原对象的访问。这种模式的主要目的是为了增加灵活性、安全性或者为对象提供额外的功能,同时保持客户端代码...

    设计模式之代理模式proxy

    **设计模式之代理模式(Proxy Pattern)** 设计模式是软件工程中的一种最佳实践,它是在特定情境下解决常见问题的模板。代理模式是其中一种行为设计模式,它的核心思想是为一个对象提供一个替身或者代理,以控制对...

    设计模式实现——代理模式

    **设计模式实现——代理模式** 在软件工程中,设计模式是一种通用可重用的解决方案,它描述了在特定上下文中经常出现的问题以及该问题的解决方案。代理模式是设计模式的一种,它提供了一种对目标对象的间接访问方式...

    结构型模式之代理模式(Proxy)

    代理模式是一种设计模式,属于结构型模式之一,其主要目的是为其他对象提供一个代理,以控制对该对象的访问。在实际应用中,代理模式能够帮助我们实现如下的功能: 1. 远程代理:代理对象可以代表一个位于远程系统...

    Java设计模式-代理模式例子

    在这个“Java设计模式-代理模式例子”中,我们将深入探讨代理模式的概念、实现方式以及它在实际开发中的应用。 代理模式的核心思想是为一个对象提供一个替身,这个替身即代理对象,代理对象控制对原对象的访问。在...

    android设计模式之代理模式

    代理模式在软件设计中是一种常用的设计模式,尤其在Android开发中,它可以帮助我们实现复杂的控制逻辑,隔离复杂性,以及提供额外的功能。在Android上下文中,代理模式常常用于数据加载、权限控制、事件处理等方面。...

    设计模式C++学习之代理模式(Proxy)

    代理模式是一种设计模式,它是结构型模式之一,主要用于在客户端和目标对象之间建立一个代理对象,以便控制对目标对象的访问。在C++中,代理模式可以用来为其他对象提供一种代理以控制对这个对象的访问,或者增加...

    Java代理模式Java动态代理

    ### Java代理模式与Java动态代理详解 #### 一、代理模式概述 代理模式是一种软件设计模式,它在客户端和目标对象之间提供了一种间接层。这种模式的主要目的是控制客户端对目标对象的访问,并且可以在不修改原有...

    设计模式--代理模式

    代理模式是一种常用的设计模式,它在软件开发中扮演着重要角色,允许我们通过一个代理类来控制对原对象的访问。在《设计模式:可复用面向对象软件的基础》(通常称为GoF设计模式)中,代理模式被定义为“为其他对象...

    代理模式 C++实现

    代理模式(Proxy) 定义: 为其他对象提供一种代理以控制对这个对象的访问 结构: 由三部分组成 1.RealSubject(真实对象): 真正会调用到的对象 2.Proxy(代理对象): 代理真实对象的地方 3.Subject(共同点): 代理对象...

    cas代理模式代码示例

    在IT行业中,代理模式是一种常见的设计模式,它允许我们在不修改原有对象的基础上,为对象添加新的功能或控制访问。在本示例中,我们将重点讨论如何在Java环境下使用代理模式来实现代理逻辑,特别是在CAS(Central ...

Global site tag (gtag.js) - Google Analytics