linux社区爱心援助Linux认证系列教程业界动态站务新闻公司招聘网络学院网址大全LPI专题CISCO专题
设为首页
加入收藏
管理团队
JSP  
JAVA  
PERL  
 您的位置:首页 > 开发语言 > JAVA >
栏目导栏
  php
  JSP
  ASP
  asp.net
  JAVA
  c/c++/c#
  perl
  JavaScript
  Basic
  Delphi
资料搜索
热门文章
·Ajax实现在textbox中模糊查询显
·Ajax实现分页查询
·JAVA反编译工具jad的简单用法
·基于Spring+Hibernate+Eclip
·java定时执行的三种方法
·读取数码照片中的Exif信息
·JSP获取客户端的浏览器和操作系
·Java文件操作大全
·Java中利用通讯API编写短信软件
·JAVA技术:上传图片的缩放处理
·利用Java实现网页浏览器
·Java编程基础
·Java调用Oracle的过程和函数
·JAVA得到网卡物理地址(Windows
·NetBeans vs Eclipse之性能参数
最新文章
·实例解析:Linux操作系统下Jav
·JAVA得到网卡物理地址(Windows
·Ajax实现在textbox中模糊查询显
·Ajax简单示例之改变下拉框动态
·AJAX如何与后台交互
·Ajax实现分页查询
·Ajax核心:XMLHTTP组件相关技术
·面向Java程序员的Ajax:构建动态
·JSP获取客户端的浏览器和操作系
·提高Java技能的几种简单有效的
·敏捷开发中需掌握移除重复代码
·Java中利用通讯API编写短信软件
·关于String和StringBuffer
·用Java得到硬盘空间
·如何优化JAVA程序设计和编码,
Google
 
通过Java Swing看透MVC设计模式
[ 作者:  加入时间:2007-10-25 13:38:48  来自:51CTO ]

一个好的用户界面(GUI)的设计通常可以在现实世界找到相应的表现。例如,如果在您的面前摆放着一个类似于电脑键盘按键的一个简单的按钮,然而就是这么简单的一个按钮,我们就可以看出一个GUI设计的规则,它由两个主要的部分构成,一部分使得它具有了按钮应该具有的动作特性,例如可以被按下。另外一部分则负责它的表现,例如这个按钮是代表了A还是B。 85cLinux联盟
看清楚这两点你就发现了一个很强大的设计方法,这种方法鼓励重用reuse,而不是重新设计redesign。你发现按钮都有相同的机理,你只要在按钮的顶上喷上不同的字母便能制造出“不同”的按钮,而不用为了每个按钮而重新设计一份图纸。这大大减轻了设计工作的时间和难度。85cLinux联盟
如果您把上述设计思想应用到软件开发领域,那么取得相似的效果一点都不让人惊奇。一个在软件开发领域应用的非常广泛的技术Model/View/Controller(MVC)便是这种思想的一个实现。85cLinux联盟
这当然很不错,但是或许您又开始疑惑这和java基础类JFC(Java Foundation Class)中的用户界面设计部分(Swing)又有什么关系呢?好的,我来告诉你。85cLinux联盟
尽管MVC设计模式通常是用来设计整个用户界面(GUI)的,JFC的设计者们却独创性的把这种设计模式用来设计Swing中的单个的组件(Component),例如表格Jtable,树Jtree,组合下拉列表框JcomboBox等等等等。这些组件都有一个Model,一个View,一个Controller,而且,这些model,view,controller可以独立的改变,就是当组件正在被使用的时候也是如此。这种特性使得开发GUI界面的工具包显得非常的灵活。85cLinux联盟
MVC设计模式把一个软件组件区分为三个不同的部分,model,view,controller。 85cLinux联盟
Model是代表组件状态和低级行为的部分,它管理着自己的状态并且处理所有对状态的操作,model自己本身并不知道使用自己的view和controller是谁,系统维护着它和view之间的关系,当model发生了改变系统还负责通知相应的view。85cLinux联盟
View代表了管理model所含有的数据的一个视觉上的呈现。一个Model可以有一个以上的View,但是Swing中却很少有这样的情况。85cLinux联盟
Controller管理着model和用户之间的交互的控制。它提供了一些方法去处理当model的状态发生了变化时的情况。85cLinux联盟
使用键盘上的按钮的例子来说明一下:Model就是按钮的整个机械装置,View/Controller就是按钮的表面部分。85cLinux联盟
下面的图解释了如何把一个JFC开发的用户界面分为model,view,controller,注意,view/Controller被合并到了一起,这是MVC设计模式通常的用法,它们提供了组件的用户界面(UI)。85cLinux联盟
用Button的例子详细说明85cLinux联盟
为了更好的理解MVC设计模式和Swing用户界面组件之间的关系,让我们更加深入的进行分析。我将采用最常见的组件button来说明。85cLinux联盟
我们从model来开始。85cLinux联盟
Model85cLinux联盟
一个按钮的model所应该具备的行为由一个接口ButtonModel来完成。一个按钮model实例封装了其内部的状态,并且定义了按钮的行为。它的所有方法可以分为四类:85cLinux联盟
1、查询内部状态85cLinux联盟
2、操作内部状态85cLinux联盟
3、添加和删除事件监听器85cLinux联盟
4、发生事件85cLinux联盟
程序员通常并不会直接和model以及view/controller打交道,他们通常隐藏于那些继承自java.awt.Component的组件里面了,这些组件就像胶水一样把MVC三者合三为一。也正是由于这些继承的组件对象,一个程序员可以很方便的混合使用Swing组件和AWT组件,然后,我们知道,Swing组件有很多都是直接继承自相应的AWT组件,它能提供比AWT组件更加方便易用的功能,所以通常情况下,我们没有必要混合使用两者。85cLinux联盟
一个实例85cLinux联盟
现在我们已经明白了Java类与MVC各个部分的对应关系,我们可以更加深入一点去分析问题了。下面我们将要讲述一个小型的使用MVC模式开发的例子。因为JFC十分的复杂,我只能把我的例子局限于一个用户界面组件里面(如果你猜是一个按钮的例子,那么你对了!)85cLinux联盟
让我们来看看这个例子的所有部分吧。85cLinux联盟
Button类85cLinux联盟
最显而易见的开始的地方就是代表了按钮组件本省的代码,因为这个类是大部分程序员会接触的。85cLinux联盟
就像我前面提到的,按钮用户界面组件类实际上就是model和view/controller的之间的黏合剂。每个按钮组件都和一个model以及一个controller关联,model定义了按钮的行为,而view/controller定义了按钮的表现。而应用程序可以在任何事件改变这些关联。让我们看看得以实现此功能的代码。

public void setModel(ButtonModel buttonmodel)85cLinux联盟
{ if (this.buttonmodel != null)85cLinux联盟
 {85cLinux联盟
this.buttonmodel.removeChangeListener(buttonchangelistener);85cLinux联盟
this.buttonmodel.removeActionListener(buttonactionlistener);85cLinux联盟
buttonchangelistener = null;85cLinux联盟
buttonactionlistener = null;85cLinux联盟
 }85cLinux联盟
 this.buttonmodel = buttonmodel;85cLinux联盟
 if (this.buttonmodel != null)85cLinux联盟
 {85cLinux联盟
buttonchangelistener = new ButtonChangeListener();85cLinux联盟
buttonactionlistener = new ButtonActionListener();85cLinux联盟
this.buttonmodel.addChangeListener(buttonchangelistener);85cLinux联盟
this.buttonmodel.addActionListener(buttonactionlistener);85cLinux联盟
 }85cLinux联盟
 updateButton();85cLinux联盟
}85cLinux联盟
public void setUI(ButtonUI buttonui)85cLinux联盟
{85cLinux联盟
 if (this.buttonui != null)85cLinux联盟
 {85cLinux联盟
this.buttonui.uninstallUI(this);85cLinux联盟
 }85cLinux联盟
 this.buttonui = buttonui;85cLinux联盟
 if (this.buttonui != null)85cLinux联盟
 {85cLinux联盟
this.buttonui.installUI(this);85cLinux联盟
 }85cLinux联盟
 updateButton();85cLinux联盟
}85cLinux联盟
public void updateButton()85cLinux联盟
{85cLinux联盟
 invalidate();85cLinux联盟
}

在进入下一节之前,你应该多花一些时间来仔细阅读一下Button类的源代码。85cLinux联盟
ButtonModel类85cLinux联盟
ButtonModel维护着三种类型的状态信息:是否被按下(pressed),是否“武装上了”(armed),是否被选择(selected)。它们都是boolean类型的值。85cLinux联盟
一个按钮被按下(pressed)是指当鼠标在按钮上面的时候,按下鼠标但是还没有松开鼠标按钮的状态,及时用户此时把鼠标拖拽到按钮的外面也没有改变这种状态。85cLinux联盟
一个按钮是否“武装了”(armed)是指按钮被按下,并且鼠标还在按钮的上面。85cLinux联盟
一些按钮还可能被选择(selected),这种状态通过重复的点击按钮取得true或者false的值。85cLinux联盟
下面的代码是状态pressed的一个缺省的实现。状态armed以及selected实现的代码与之类似。ButtonModel类应该被继承,这样可以覆盖缺省的状态定义,实现有个性的按钮。

private boolean boolPressed = false;85cLinux联盟
public boolean isPressed()85cLinux联盟
{85cLinux联盟
 return boolPressed;85cLinux联盟
}85cLinux联盟
public void setPressed(boolean boolPressed)85cLinux联盟
{85cLinux联盟
 this.boolPressed = boolPressed;85cLinux联盟
 fireChangeEvent(new ChangeEvent(button));85cLinux联盟
}

在进入下一节之前,你应该多花一些时间来仔细阅读一下ButtonModel类的源代码。 85cLinux联盟
ButtonUI类85cLinux联盟
85cLinux联盟
按钮的view/controller是负责构建表示层的。缺省情况下它仅仅是用背景色画一个矩形而已,他们的子类继承了他们并且覆盖了绘制的方法,使得按钮可以有许多不同的表现,例如MOTIF,Windows 95,Java样式等等。

public void update(Button button, Graphics graphics)85cLinux联盟
{  }85cLinux联盟
public void paint(Button button, Graphics graphics)85cLinux联盟
{85cLinux联盟
 Dimension dimension = button.getSize();85cLinux联盟
 Color color = button.getBackground();85cLinux联盟
 graphics.setColor(color);85cLinux联盟
 graphics.fillRect(0, 0, dimension.width, dimension.height); 85cLinux联盟
}

ButtonUI类并不自己处理AWT事件,他们会使用一个定制的事件监听器把低级的AWT事件翻译为高级的Button模型期望的语义事件。下面就是安装/卸载事件监听器的代码。

private static ButtonUIListener buttonuilistener = null;85cLinux联盟
public void installUI(Button button)85cLinux联盟
{85cLinux联盟
 button.addMouseListener(buttonuilistener);85cLinux联盟
 button.addMouseMotionListener(buttonuilistener);85cLinux联盟
 button.addChangeListener(buttonuilistener);85cLinux联盟
}85cLinux联盟
public void uninstallUI(Button button)85cLinux联盟
{85cLinux联盟
 button.removeMouseListener(buttonuilistener);85cLinux联盟
 button.removeMouseMotionListener(buttonuilistener);85cLinux联盟
 button.removeChangeListener(buttonuilistener);85cLinux联盟
}

View/Controller实际上就是一些方法。他们不维护任何自己的状态信息。因此,许多按钮的实例可以共享一个ButtonUI实例。ButtonUI是通过在方面的参数列表里面加上按钮的引用来区分各个不同的按钮。85cLinux联盟
同样,希望你能多花一些时间来看看ButtonUI类,然后咱们进入下一节。85cLinux联盟
ButtonUIListener类85cLinux联盟
ButtonUIListener类可以帮助Button类去转变鼠标或者键盘的输入为对按钮模型的操作。这个监听器类实现了:MouseListener,MouseMotionListener,ChangeListener接口,并且处理一下事件:

public void mouseDragged(MouseEvent mouseevent)85cLinux联盟
{85cLinux联盟
 Button button = (Button)mouseevent.getSource();85cLinux联盟
 ButtonModel buttonmodel = button.getModel();85cLinux联盟
 if (buttonmodel.isPressed())85cLinux联盟
 {85cLinux联盟
if (button.getUI().contains(button, mouseevent.getPoint()))85cLinux联盟
{85cLinux联盟
 buttonmodel.setArmed(true);85cLinux联盟
}85cLinux联盟
else85cLinux联盟
{85cLinux联盟
 buttonmodel.setArmed(false);85cLinux联盟
}85cLinux联盟
 }85cLinux联盟
}85cLinux联盟
public void mousePressed(MouseEvent mouseevent)85cLinux联盟
{85cLinux联盟
 Button button = (Button)mouseevent.getSource();85cLinux联盟
 ButtonModel buttonmodel = button.getModel();85cLinux联盟
 buttonmodel.setPressed(true);85cLinux联盟
 buttonmodel.setArmed(true);85cLinux联盟
}85cLinux联盟
public void mouseReleased(MouseEvent mouseevent)85cLinux联盟
{85cLinux联盟
 Button button = (Button)mouseevent.getSource();85cLinux联盟
 ButtonModel buttonmodel = button.getModel();85cLinux联盟
 buttonmodel.setPressed(false);85cLinux联盟
 buttonmodel.setArmed(false);85cLinux联盟
}85cLinux联盟
public void stateChanged(ChangeEvent changeevent)85cLinux联盟
{85cLinux联盟
 Button button = (Button)changeevent.getSource();85cLinux联盟
 button.repaint();85cLinux联盟
}

85cLinux联盟
总结85cLinux联盟
85cLinux联盟
我希望你能按照上面讲述的方法去做。如果不能,那么所有的努力都将白费。这个例子以及Swing用户界面组件的好处在于你不用去花时间去弄明白他们底层是如何设计实现的就可以很方便的使用他们了。他们都提供了缺省的model以及view/controller,然后,当你自己做组件的时候,你会发现上面的思想的强大之处

Linux联盟收集整理 ,转贴请标明原始链接,如有任何疑问欢迎来本站Linux论坛讨论
评论】【加入收藏夹】【 】【打印】【关闭
※ 相关链接
 ·通过JAVAScript实现页面自适应  (2007-10-30 14:27:43)
 ·如何优化JavaScript脚本的性能  (2007-10-30 14:25:25)
 ·几个有用的Javascript验证脚本  (2007-10-30 14:24:47)
 ·Javascript+ASP打造无刷新新闻列表  (2007-10-30 14:23:38)
 ·如何用javascript防止双击  (2007-10-30 14:19:44)
 ·JavaScript 访问 JSF 组件的方法  (2007-10-30 14:18:21)
 ·JavaScript实现仿Windows关机效果  (2007-10-30 14:14:42)
 ·JavaScript去除空格的三种方法  (2007-10-30 14:07:32)
 ·用Javascript评估用户输入密码的强度  (2007-10-30 14:06:19)
 ·JavaScript处理事件:单击事件onClick  (2007-10-30 14:01:38)