Java设计模式之代理模式篇(1)

原创 2003年06月24日 14:25:00
<?xml:namespace prefix = o ns = "urn:schemas-microsoft-com:office:office" />

kong.gif
Java设计模式之代理模式篇(1)
作者:冯睿    本文选自:赛迪网  2003年03月11日

在软件工程中,代理模式(Proxy Pattern)在很多情况下都非常有用。例如在Java XML保重,开发人员可以利用代理来访问Web服务。例1中演示了经典的Hello World Web服务的例子:

例1 一个SOAP代理的例子

public class HelloClient {
    public static void main(String[] args) {
        try {
            HelloIF_Stub proxy = (HelloIF_Stub)(new HelloWorldImpl().getHelloIF());
            proxy._setTargetEndpoint(args[0]);
            System.out.println(proxy.sayHello("Hello World!"));
        } catch (Exception ex) {
            ex.printStackTrace();
        }
    }
}


在例一中,客户端首先获得对代理的引用,然后利用命令行参数设定代理的端点(即Web服务的URL地址),接下来调用代理的sayHello()方法,代理再将方法调用传递给相应的Web服务。

代理模式和修饰模式(Decorator Pattern)有一定的相似之处。两个模式又使用了代理将方法调用传递给另一个对象,该对象被称为真实对象(Real Subject)。代理模式和修饰模式的不同之处在于:在代理模式中,代理和真实对象之间的关系在程序被编译的时候就确定下来了,而修饰模式则是在运行时递归地创建。

本文首先提供一个ImageIcon的例子来说明代理模式,然后会探讨一下JDK是如何支持代理模式的。

代理模式


代理模式通过使用代理来替代实际的对象,使程序能够控制对该对象的访问。下面是一个ImageIcon的例子。

例2 ImageIcon的例子

import java.awt.*;
import java.awt.event.*;
import javax.swing.*;
public class IconTest extends JFrame {
   private static String IMAGE_NAME = "hands.jpg";
   private static int FRAME_X = 150,      FRAME_Y = 200, 
                  FRAME_WIDTH = 430, FRAME_HEIGHT = 392;
   private Icon imageIcon = null, imageIconProxy = null;
   static public void main(String args[]) {
      IconTest app = new IconTest();
      app.show();
   }
   public IconTest() {
      super("ImageIcon测试");
      imageIcon = new ImageIcon(IMAGE_NAME);
      setBounds(FRAME_X, FRAME_Y, FRAME_WIDTH, FRAME_HEIGHT);
      setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
   }
   public void paint(Graphics g) {
      super.paint(g);
      Insets insets = getInsets();
      imageIcon.paintIcon(this, g, insets.left, insets.top);
   }
}


111375.jpg


图1 ImageIcon测试


在上面的例子中程序创建了一个javax.swing.ImageIcon对象,并且重载了paint()方法来显示图标。例二中的代码有一些缺陷,在程序中开发人员只能够使用比较小的图片。因为创建一个图形会耗费很多系统资源,而ImageIcon的实例是在初始化时就创建包含在其中的图形对象。如果程序需要在较短的时间内显示很多比较大的图形对象,系统就有可能处理不过来。同时如果应用程序没有使用到这些图形,在前台创建这些图形对于系统资源来说也是一种浪费。

一个更好的解决方案是在需要显示图形的时候再加载图形。为了达到这个目的,可以通过利用代理来实现。当代理的paintIcon()方法第一次被调用时,程序才创建图形。图二中显示了一个既包含ImageIcon(左)又包含ImageIcon代理(右)的例子。在图二中上面一幅图中显示了程序刚加载时的情形。由于ImageIcon对象在初始化的时候就需要加载图形,因此当窗口出现时图片就显示在窗口的左边。而ImageIcon代理中的图片要到第一次被绘制时才会被调用。图二中下面一幅图显示了两幅图片都被加载后的情景。

111376.jpg


111377.jpg


图2 ImageIcon和ImageIcon代理


例3 ImageIcon对象和ImageIcon代理

import java.awt.*;
import java.awt.event.*;
import javax.swing.*;
public class ProxyTest extends JFrame {
   private static String IMAGE_NAME = "hands.jpg";
   private static int IMAGE_WIDTH = 430, IMAGE_HEIGHT = 390,
                          SPACING = 5,        FRAME_X = 150,
                          FRAME_Y = 200, FRAME_WIDTH = 880, 
                     FRAME_HEIGHT = 394;
   private Icon imageIcon = null, imageIconProxy = null;
   static public void main(String args[]) {
      ProxyTest app = new ProxyTest();
      app.show();
   }
   public ProxyTest() {
      super("ImageIcon代理测试");
     // 生成ImageIcon和ImageIcon代理的实例
      imageIcon = new ImageIcon(IMAGE_NAME);
      imageIconProxy = new ImageIconProxy(IMAGE_NAME,
                                          IMAGE_WIDTH, 
                                          IMAGE_HEIGHT);
      // 设定边框和缺省的退出操作
      setBounds(FRAME_X, FRAME_Y, FRAME_WIDTH, FRAME_HEIGHT);
      setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
   }
   public void paint(Graphics g) {
      super.paint(g);
      Insets insets = getInsets();
      imageIcon.paintIcon(this, g, insets.left, insets.top);
      imageIconProxy.paintIcon(this, g, 
               insets.left + IMAGE_WIDTH + SPACING, // 宽
               insets.top); // 高
   }
}


从上面的代码我们可以注意到在ProxyTest的构造函数中创建了一个ImageIcon对象和一个ImageIconProxy对象,并且重写了基类的paint()方法。在讨论代理类的实现代码之前,让我们先来看一下ImageIcon的类结构图:

111378.jpg


图3 ImageIcon的类结构图


从类结构图中可以看到,javax.swing.Icon接口中定义了三个最基本的方法:paintIcon(),getIconWidth()和getIconHeight()。ImageIcon类实现了Icon接口并且增加了一些方法。同时ImageIcon中也保存了对包含在其中的图形对象的引用以及描述。

ImageIcon代理类也实现了Icon接口,同时保存了对真实对象――ImageIcon的引用。图四显示了ImageIconProxy的类结构图。

111379.jpg


图4 ImageIconProxy的类结构图


下面是ImageIconProxy的实现代码:

例4 ImageIcon代理

// ImageIconProxy是ImageIcon对象的代理,它将图形的显示延迟到图形第一次被
// 绘制的时候。当图形还没有被绘制以前,该代理在界面上显示"加载图片…"的信息
class ImageIconProxy implements javax.swing.Icon {
   private Icon realIcon = null;
   boolean isIconCreated = false;
   private String  imageName;
   private int     width, height;
   public ImageIconProxy(String imageName, int width, int height){
      this.imageName = imageName;
      this.width = width;
      this.height = height;
   }
   public int getIconHeight() {
      return isIconCreated ? height : realIcon.getIconHeight(); 
   }
   public int getIconWidth() {
      return isIconCreated realIcon == null ? width : realIcon.getIconWidth(); 
   }
   // 代理的paint()方法覆盖了积累中的该方法。注意代理直到在需要显示图形时才加
   // 载了图形。
   public void paintIcon(final Component c, 
                               Graphics g, int x, int y) {
      if(isIconCreated) {
         realIcon.paintIcon(c, g, x, y);
      }
      else {
         g.drawRect(x, y, width-1, height-1);
         g.drawString("加载图片...", x+20, y+20);
         // ImageIcon对象实在另一个线程中被创建的
         synchronized(this) {
            SwingUtilities.invokeLater(new Runnable() {
                 public void run() {
                  try {
                     // 为了使ImageIcon对象和ImageIcon代理之间的差别
                     // 更加显著,该线程休眠2秒
                     Thread.currentThread().sleep(2000);
                     realIcon = new ImageIcon(imageName);
                     isIconCreated = true;
                  }
                  catch(InterruptedException ex) {
                       ex.printStackTrace();
                  }
                  // 当创建了ImageIcon对象后调用repaint()方法重绘图形
                  c.repaint();
               }
            });
         }
      }
   }
}


ImageIconProxy通过realIcon保存了对一个对图形的引用。当第一次对代理进行绘制时,ImageIcon对象在一个独立的线程中被创建,然后图形被加载,并通过repaint()方法绘制。图五通过时序图说明了这些事件之间的关系。

111380.jpg


图5 ImageIcon代理的时序图

[置顶] 国内首部Java多线程设计模式原创作品《Java多线程编程实战指南(设计模式篇)》已出版

国内首部Java多线程设计模式原创作品《Java多线程编程实战指南(设计模式篇)》已由电子工业出版社出版。本书从理论到实战,用生活化的实例和通俗易懂的语言全面介绍Java多线程编程的"三十六计"——多...
  • viscent_huang
  • viscent_huang
  • 2015年11月18日 11:21
  • 4338

JAVA设计模式之代理模式

【代理模式应用场景举例】 比如在玩“极品飞车”这款游戏,如果游戏者手中的金钱达到了一定的数量就可以到车店买一部性能更高的赛车,那么这个卖车的“车店”就是一个典型的“汽车厂家”的“代理”,他为汽车厂家...
  • jason0539
  • jason0539
  • 2014年04月05日 07:07
  • 22945

Java设计模式——代理模式实现及原理

简介 Java编程的目标是实现现实不能完成的,优化现实能够完成的,是一种虚拟技术。生活中的方方面面都可以虚拟到代码中。代理模式所讲的就是现实生活中的这么一个概念:中介。 代理模式的定义:给某一个对象提...
  • Goskalrie
  • Goskalrie
  • 2016年09月07日 13:43
  • 20124

设计模式之代理模式 java (1)

近期再看有关于设计模式的的东西,感觉在代码架构方面尤其有用,在上学的时候没好好学这些东西,感觉自己写的代码乱成一锅粥!呵呵--- 今天把看到的代理模式给大家分享分享: 中心思想:(自己总结,不妥之...
  • zhangxiweicaochen
  • zhangxiweicaochen
  • 2012年11月30日 11:18
  • 518

Java设计模式-代理模式

  • 2015年01月25日 01:27
  • 10KB
  • 下载

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

  • 2017年07月09日 12:54
  • 26KB
  • 下载

Java设计模式之代理模式

  • 2014年01月22日 11:09
  • 51KB
  • 下载

设计模式之代理模式Java实现和类设计图

  • 2013年04月20日 16:35
  • 60KB
  • 下载

详解java设计模式(五)之代理模式上篇(结构型)

一.认识代理模式 1.定义: 给某一个对象提供一个代理,并由代理对象控制对原对象的引用。这种角色类似与我们平时生活中见到的中介,首先中介没有房子,只要房主有房子,房主把房子委托给中介。中介替房主找...
  • xqhadoop
  • xqhadoop
  • 2017年03月11日 14:03
  • 136

Java设计模式之代理模式(基础篇)

代理模式就是为其他对象提供一种代理以控制对这个对象的访问的模式。在一些情况下,一个客户不想或者不能够直接引用一个对象,而代理对象可以在客户端和目标对象之间起到中介的作用。换言之,就是在一些情况下,一个...
  • wchycsdn
  • wchycsdn
  • 2016年09月03日 19:18
  • 315
内容举报
返回顶部
收藏助手
不良信息举报
您举报文章:Java设计模式之代理模式篇(1)
举报原因:
原因补充:

(最多只允许输入30个字)