动态网站制作指南 [  QQ表情  ]
[ 投票调查 ]
[ 企业邮箱 ]
[ 网站空间 ]
网络编程 | 站长之家 | 网页制作 | 图形图象 | 操作系统 | 冲浪宝典 | 软件教学 | 网络办公 | 邮件系统 | 网络安全 | 认证考试 | 系统进程
ASP源码 | .Net源码 | PHP源码 | JSP源码 | JAVA源码 | CGI源码 | VB源码 | C++源码 | Delphi源码 | PB源码 | VF源码 | 汇编 | 服务器
Firefox | IE | Maxthon | 迅雷 | 电驴 | BitComet | FlashGet | QQ | QQ空间 | Vista | 输入法 | Ghost | Word | Excel | wps | Powerpoint
asp | .net | php | jsp | Sql | c# | Ajax | xml | Dreamweaver | FrontPages | Javascript | css | photoshop | fireworks | Flash | Cad | Discuz!
当前位置 > 网站建设学院 > 网络编程 > Java教程
Tag:注入,存储过程,分页,安全,优化,xmlhttp,fso,jmail,application,session,防盗链,stream,无组件,组件,md5,乱码,缓存,加密,验证码,算法,cookies,ubb,正则表达式,水印,索引,日志,压缩,base64,url重写,上传,控件,Web.config,JDBC,函数,内存,PDF,迁移,结构,破解,编译,配置,进程,分词,IIS,Apache,Tomcat,phpmyadmin,Gzip,触发器,socket
网络编程:ASP教程,ASP.NET教程,PHP教程,JSP教程,C#教程,数据库,XML教程,Ajax,Java,Perl,Shell,VB教程,Delphi,C/C++教程,软件工程,J2EE/J2ME,移动开发
文章搜索服务
邮件订阅
输入你的邮件地址,
你将不会错过任何关于:
[ Java教程 ]的信息

本月文章推荐
.Java项目资源布局方案.
.使用EJB 3.0简化企业级Java开发一.
.为什么要学习Java,成为Java程序.
.Mark Spritzler的SCJD学习指导!.
.非官方观点 Java vs .NET.
.台湾大学GigaPop和VLSI建设项目.
.java.applet.Applet的 .
.JavaBeans教程.
.Java语言中Timer类的简洁用法(一).
.Applet加载Java应用程序.
.对于java的打印问题.
.Javadoc利弊分析.
.Java 数据对象(JDO)介绍(1).
.PetStore 中EJB 的设计模式.
.简单的JDBC应用程序for Java DB.
.java image.
.Eclipse上的Tomcat插件安装和调试.
.Java:IT认证界的新贵.
.用例子来说明面向对象和面向过程.
.JBuilder 8新特性简介.

利用Observer模式解决组件间通信问题

发表日期:2008-1-5 |



  1. 问题的提出
  以前做一个界面的时候经常会碰到这样的尴尬情况:希望保留各个独立的组件(类),但又希望它们之间能够相互通信。譬如Windows中的EXPlorer,我们希望鼠标点击左边是树型目录的一个节点,右边的文件浏览能及时列出该节点目录下的文件和子目录,类似这样一个简单的应用,假如只有一个类继续JFrame,而树型组件和浏览文件的面板作为成员,就像:
  public class MainFrame extends JFrame
  {
  JPanel treePanel;
  JTree tree;
  JPanel filePanel;
  ...
  }
  
  这样当然轻易在两者之间传递消息,但是可扩展性较差。通常轻易想到的是两种办法:在一个组件里保留另一个组件类型的成员,初始化时作为参数传入引用,比如:
  class TreePanel extends JPanel
  {
  JTree tree;
  ...
  }
  
  class FilePanel extends JPanel
  {
  public FilePanel(JTree tree){...}
  ...
  }
  
  
  或者将一个组件线程化,不停地监听另一个组件的变化,然后作出相应的反映,比如:
  class TreePanel extends JPanel
  {
  JTree tree;
  ...
  }
  
  class FilePanel extends JPanel implements Runnable
  {
  public void run()
  {
  while (true)
  {
  //监听tree的变化
  }
  ...
  }
  ...
  }
  
  这样确实可以达到我们的目的,但是第一种方案显然不利于松散耦合,第二种方案比较占用系统资源。通过学习设计模式,我们发现可以用Observer模式来解决这个问题。
  
  2. Observer模式
  设计模式分为创建型、结构型和行为型,其中行为型模式专门处理对象间通信,指定交互方式等,Observer模式就是属于行为型的一种设计模式。按照“四人帮”(Gang of Four)在“Design Patterns”里的定义,Observer模式“定义对象间的一种一对多的依靠关系,当一个对象的状态发生改变时, 所有依靠于它的对象都得到通知并被自动更新”,这个描述正好符合我们对“组件通信”问题的需求。让我们先看看Observer模式的结构:
  
  其中各元素的含义如下:
  Subject:被观察的目标的抽象接口,它提供对观察者(Observer)的注册、注销服务,Notify方法通知Observer目标发生改变;
  Object:观察者的抽象接口,Update方法是当得到Subject状态变化的通知后所要采取的动作;
  ConcreteSubject:Subject的具体实现;
  ConcreteObserver:Observer的具体实现
  Observer模式在实现MVC结构时非常有用,为数据和数据表示解耦合。
  
  3. Java中的Observer模式:Observer和Observable
  在大致了解了Observer模式的描述之后,现在我们更为关心的是它在Java中是如何应用的。幸运的是,自从JDK 1.0起,就有了专门处理这种应用的API,这就是Observer接口和Observable类,它们是属于java.util包的一部分。看来Java的开发者们真是深谙设计模式的精髓,而Java的确是为了真正的面向对象而生的,呵呵!
  这里的Observer和Observable分别对应设计模式中的Observer和Subject,对比一下它们定义的方法,痕迹还是相当明显的:
  Observer的方法:
  update(Observable subject, Object arg) 监控subject,当subject对象状态发生变化时Observer会有什么响应,arg是传递给Observable的notifyObservers方法的参数;
  Observable的方法:
  addObserver(Observer observer) observer向该subject注册自己
  hasChanged() 检查该subject状态是否发生变化
  setChanged() 设置该subject的状态为“已变化”
  notifyObservers() 通知observer该subject状态发生变化
  
  4. Observer模式在Java GUI事件模型中应用
  其实在AWT/Swing事件模型中用到了好几种设计模式,以前的JDK 1.0 AWT使用的是“基于继续的事件模型”,在该模型Component类中定义了一系列事件处理方法,如:handleEvent,mouseDown,mouseUp等等,我们对事件的响应是通过对组件类继续并覆盖相应的事件处理方法的手段来实现,这种模型有很多缺点,事件的处理不应当由事件产生者负责,而且根据“设计模式”一书中的原则,“继续”通常被认为是“对封装性的破坏”,父子类之间的紧密耦合关系降低了灵活性,同时继续轻易导致家族树规模的庞大,这些都不利于组件可重用。
  JDK 1.1以后新的事件模型是被成为“基于授权的事件模型”,也就是我们现在所熟悉的Listener模型,事件的处理不再由产生事件的对象负责,而由Listener负责。尤其在Swing组件中设计MVC结构时用到了Observer模式,众所周知,MVC表示“模型-视图-控制器”,即“数据-表示逻辑-操作”,其中数据可以对应多种表示,这样视图就处在了observer的地位,而model则是subject。
  
  5. 简单的例子
  回到本文一开始的那个Explorer的例子,我们考虑做一个简单的图片浏览器,使树型选择组件和图片浏览面板在两个不同的类中,其中图片浏览面板根据所选择的树的节点显示相应的图片,所以图片浏览面板是一个observer,树是subject。由于Java单根继续的原因,我们不能同时继续JPanel和Observable,但可以用对象的组合把一个subject放到我们的类当中,并通过TreeSelectionListener触发subject的setChanged方法,并通过notifyObservers方法通知observer。
  例子代码如下:
  //LeFTPanel.java
  package com.jungleford.test;
  import java.awt.BorderLayout;
  import javax.swing.*;
  import javax.swing.event.TreeSelectionListener;
  import javax.swing.event.TreeSelectionEvent;
  import javax.swing.tree.DefaultMutableTreeNode;
  import java.util.Observable;
  import java.util.Observer;
  
  public final class LeftPanel extends JPanel
  {// 把树型选择视图布局在左边
  private JTree tree;// 树型选择视图
  private JScrollPane scroll;// 让视图可滚动
  private DefaultMutableTreeNode root, node1, node2;// 根节点及两个叶子
  private Sensor sensor;// sensor是一个Observable,由于只能单根继续,所以作为组合成员
  private String file;// 图片文件名,与RightPanel通信的内容
  
  public LeftPanel(Observer observer)
  {
  file = "";
  sensor = new Sensor();
  sensor.addObserver(observer);// 向Observable注册Observer
  root = new DefaultMutableTreeNode("Images");
  tree = new JTree(root);
  node1 = new DefaultMutableTreeNode("Rabbit");
  node2 = new DefaultMutableTreeNode("Devastator");
  root.add(node1);
  root.add(node2);
  tree.addTreeSelectionListener(new TreeSelectionListener()
  {// 树节点选择动作
  public void valueChanged(TreeSelectionEvent e)
  {
  Object obj = e.getPath().getLastPathComponent();
  if (obj instanceof DefaultMutableTreeNode)
  {
  DefaultMutableTreeNode node = (DefaultMutableTreeNode)obj;
  if (node == root)
  file = "";// 选择根
  if (node == node1)
  file = "rabbit.jpg";// 选择node1
  if (node == node2)
  file = "devastator.gif";// 选择node2
  sensor.setData(file);// 改变Observable
  sensor.notifyObservers();// 通知observer,对象已改变
  }
  }
  });
  scroll = new JScrollPane(tree);
  add(scroll, BorderLayout.CENTER);
  }
  
  public Observable getSensor()
  {// 返回Observable对象,使Observer可以获取
  return sensor;
  }
  }
  
  class Sensor extends Observable
  {// 定义自己的Observable
  private Object data;
  
  public void setData(Object newData)
  {
  data = newData;
  setChanged();// 改变Observable
  System.out.println("Data changed!");
  }
  
  public Object getData()
  {
  return data;
  }
  }
  
  //RightPanel.java
  package com.jungleford.test;
  import java.awt.*;
  import javax.swing.JPanel;
  import java.util.Observer;
  import java.util.Observable;
  
  public class RightPanel extends JPanel implements Observer
  {// 把图片浏览视图布局在右边
  private Image image;
  
  public void update(Observable subject, Object obj)
  {// 定义接收到Observable变化后的响应动作
  String file = (String)((Sensor)subject).getData();
  if (!file.equals(""))
  {
  image = Toolkit.getDefaultToolkit().getImage(file);
  MediaTracker tracker = new MediaTracker(this);// 定义图像跟踪
  tracker.addImage(image, 0);
  try
  {
  tracker.waitForID(0);// 等待图像的完全加载
  }
  catch (InterruptedException e)
  {
  e.printStackTrace();
  }
  }
  else
  image = null;
  repaint();// 重绘组件
  }
  
  public void paintComponent(Graphic
上一篇:Java设计模式例子 FactoryMethod Pattern 人气:517
下一篇:使用Decorator模式 翻译者:Disneytiger 人气:432
浏览全部Java的内容 Dreamweaver插件下载 常用网页广告代码全集
  最新网站源码 最新软件下载
2008-10-13 爬爬思特新闻管理系统 v2.0 Beta1
2008-10-13 Pligg v9.9.5 Beta
2008-10-13 广优邮件发送系统 v2.1
2008-10-13 缤纷互动视频交友 v3.1 RC
2008-10-13 MyShop网络商城 build 081005
2008-10-13 Chyrp 超轻量级开源博客引擎 v2.
2008-10-13 162100静态(论坛/文章)系统 v2.4
2008-10-13 金博人才招聘求职网黄金版 v4.2
2008-10-13 愚人笔记 v4.0
2008-10-11 联系人分组工具 v1.1 中文破解版
2008-10-11 FaceMelter变脸 v2.0 汉化破解版
2008-10-11 PathTracker道路跟踪仪 v1.2 破解
2008-10-11 Rooms手机聊天室 v0.6.7 破解版
2008-10-11 RemoteDesktop远程桌面 v1.0 破解
2008-10-11 ProRemote远程调音台 v1.0.1 破解
2008-10-11 PicShare照片共享 v1.0.0 破解版
2008-10-11 Photogene照片编辑器 v1.5 汉化破
2008-10-11 WriteRoom共享文档 v1.0 破解版
  发表评论
姓 名: 验证码:
内 容:
站长工具:网站收录查询 | Google PR查询 | ALEXA排名查询 | CSS在线编辑器 | 广告代码 | js/vbs加密 | md5加密 | 进制转换 | UTF-8 转换工具 | Html转换js | Html转换asp | Html转换php | Html转换perl
实用工具:汉字翻译拼音 | 拼音字典 | 符号对照表 | 个税计算 | 实时汇率查询换算 | 经典小工具 | 汉字简繁转换 | 普通单位换算 | 公制单位换算 | 生辰老黄历 | 国内电话区号 | 国家代码与域名缩写 | 文字加密解密 | 健康查询 | 万年历 | 汉字横竖排版 | 手机号码查询 | 计算器 | ip搜索
业务联系 | 广告刊登 | 频道合作 | 投稿荐稿 | 联系方式 | 加入收藏 | RSS订阅
Copyright © 2000-2008 www.knowsky.com All rights reserved | 网络实名:动态网站制作指南 | 沪ICP备05001343号
ホームページ制作 不動産検索システム 求人情報
防水工事·改修工事 フットサル大会 探偵
SEO対策 中国語教室 ホームページ作成