电话:0731-83595998
导航

利用Observer模式实现组件间通信

来源: 2017-12-23 13:53

 百度广告

  1. 问题的提出

  以前做一个界面的时候常常会遇到这样的尴尬情况:希望保留各个独立的组件(类),但又希望它们之间能够相互通信。譬如Windows中的Explorer,我们希望鼠标点击左边是树型目录的一个节点,右边的文件浏览能及时列出该节点目录下的文件和子目录,类似这样一个简单的应用,如果只有一个类继承JFrame,而树型组件和浏览文件的面板作为成员,就像:

  这样当然容易在两者之间传递消息,但是可扩展性较差。通常容易想到的是两种办法:在一个组件里保留另一个组件类型的成员,初始化时作为参数传入引用,比如:

 

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模式

  设计模式分为创建型、结构型和行为型,其中行为型模式专门处理对象间通信,指定交互方式等,Observer模式就是属于行为型的一种设计模式。按照"四人帮"(Gang of Four)在"Design Patterns"里的定义,Observer模式"定义对象间的一种一对多的依赖关系,当一个对象的状态发生改变时, 所有依赖于它的对象都得到通知并被自动更新",这个描述正好符合我们对"组件通信"问题的需求。让我们先看看Observer模式的结构:

 

 


  Subject:被观察的目标的抽象接口,它提供对观察者(Observer)的注册、注销服务,Notify方法通知Observer目标发生改变; 
ConcreteSubject:Subject的具体实现; 
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状态发生变化

  其实在AWT/Swing事件模型中用到了好几种设计模式,以前的JDK 1.0 AWT使用的是"基于继承的事件模型",在该模型Component类中定义了一系列事件处理方法,如:handleEvent,mouseDown,mouseUp等等,我们对事件的响应是通过对组件类继承并覆盖相应的事件处理方法的手段来实现,这种模型有很多缺点,事件的处理不应当由事件产生者负责,而且根据"设计模式"一书中的原则,"继承"通常被认为是"对封装性的破坏",父子类之间的紧密耦合关系降低了灵活性,同时继承容易导致家族树规模的庞大,这些都不利于组件可重用。

  JDK 1.1以后新的事件模型是被成为"基于授权的事件模型",也就是我们现在所熟悉的Listener模型,事件的处理不再由产生事件的对象负责,而由Listener负责。尤其在Swing组件中设计MVC结构时用到了Observer模式,众所周知,MVC表示"模型-视图-控制器",即"数据-表示逻辑-操作",其中数据可以对应多种表示,这样视图就处在了observer的地位,而model则是subject。 简单的例子 

  回到一开始的那个Explorer的例子,我们考虑做一个简单的图片浏览器,使树型选择组件和图片浏览面板在两个不同的类中,其中图片浏览面板根据所选择的树的节点显示相应的图片,所以图片浏览面板是一个observer,树是subject。由于Java单根继承的原因,我们不能同时继承JPanel和Observable,但可以用对象的组合把一个subject放到我们的类当中,并通过TreeSelectionListener触发subject的setChanged方法,并通过notifyObservers方法通知observer。

  例子代码

 

//LeftPanel.Java 
import Java.awt.BorderLayout;
import Javax.swing.event.TreeSelectionListener;
import Javax.swing.tree.DefaultMutableTreeNode;
import Java.util.Observer;

public final class LeftPanel extends JPanel
private JTree tree;// 树型选择视图 
private DefaultMutableTreeNode root, node1, node2;// 根节点及两个叶子 
private String file;// 图片文件名,与RightPanel通信的内容 

  public LeftPanel(Observer observer)
file = "";
sensor.addObserver(observer);// 向Observable注册Observer 
tree = new JTree(root);
node2 = new DefaultMutableTreeNode("Devastator");
root.add(node2);
{// 树节点选择动作 
{
if (obj instanceof DefaultMutableTreeNode)
DefaultMutableTreeNode node = (DefaultMutableTreeNode)obj;
file = "";// 选择根 
file = "rabbit.jpg";// 选择node1 
file = "devastator.gif";// 选择node2 
sensor.notifyObservers();// 通知observer,对象已改变 
}
scroll = new JScrollPane(tree);
}

  public Observable getSensor()
return sensor;
}

class Sensor extends Observable
private Object data;

  public void setData(Object newData)
data = newData;
System.out.println("Data changed!");

  public Object getData()
return data;
}

//RightPanel.Java 
import Java.awt.*;
import Java.util.Observer;

public class RightPanel extends JPanel implements Observer
private Image image;

  public void update(Observable subject, Object obj)
String file = (String)((Sensor)subject).getData();
{
MediaTracker tracker = new MediaTracker(this);// 定义图像跟踪 
try
tracker.waitForID(0);// 等待图像的完全加载 
catch (InterruptedException e)
e.printStackTrace();
}
image = null;
}

  public void paintComponent(Graphics g)
g.setColor(Color.LIGHT_GRAY);
if (image != null)
}

//MainFrame.Java 
import Java.awt.*;

public class MainFrame extends JFrame
public static void main(String args)
MainFrame frame = new MainFrame();
LeftPanel left = new LeftPanel(right);// 注册Observer 
frame.getContentPane().add(right, BorderLayout.CENTER);
frame.setSize(400, 300);
frame.setVisible(true);
}

||| 
  启动界面

 

 

 

 

编辑推荐:

下载Word文档

温馨提示:因考试政策、内容不断变化与调整,长理培训网站提供的以上信息仅供参考,如有异议,请考生以权威部门公布的内容为准! (责任编辑:长理培训)

网络课程 新人注册送三重礼

已有 22658 名学员学习以下课程通过考试

网友评论(共0条评论)

请自觉遵守互联网相关政策法规,评论内容只代表网友观点!

最新评论

点击加载更多评论>>

精品课程

更多
10781人学习

免费试听更多

相关推荐
图书更多+
  • 电网书籍
  • 财会书籍
  • 其它工学书籍
拼团课程更多+
  • 电气拼团课程
  • 财会拼团课程
  • 其它工学拼团
热门排行

长理培训客户端 资讯,试题,视频一手掌握

去 App Store 免费下载 iOS 客户端