最近我给女朋友买了一款可以更换外壳的手机现在的外壳是红色的如果我想用这款手机的时候会更换成银灰色的外壳但是我不能随意更换天线或者话筒因为这些功能模块在手机生产的时候就已经被固定了
软件中的修饰者(decorator)和手机的外壳一样封装了一些可以替换的功能例如下面是一段替换Swing中表模型的代码
TableSortDecorator sortDecorator = new TableSortDecorator(tablegetModel());
tablesetModel(sortDecorator);
在这段代码中程序首先将表模型包装在一个修饰对象中以后当表对它的模型进行操作的时候它实际上操作的是排序修饰对象(sortDecorator)该修饰对象在表模型中加入了排序功能而将其他基本的功能委托给缺省的表模型在修饰模型中这个缺省的表模型又被称为真实对象(real subject)
在Java的编程中基类和子类的继承关系在编译的时候就被固定了就像手机的天线和话筒一样由于继承关系是静态的开发人员无法在程序运行时改变对象的行为但是通过修饰者开发人员可以在运行时拼装对象因此修饰模式提供了一种比继承更灵活的功能扩充模式
修饰模式(Decorator Pattern)
在运行时将特定的功能绑定在对象上这就是修饰模式的核心修饰模式比继承更加灵活因为后者是在编译时就将特定的功能绑定到类上
下面然我们来看一个简单的I/O例子
FileReader frdr = new FileReader(filename);
LineNumberReader lrdr = new LineNumberReader(frdr);
这段代码中创建了一个Readerlrdr它从一个文件中读取数据并跟蹤文件的行号在第一行创建的frdr对象能够从文件中读取数据而第二行给lrdr增加了跟蹤行号的功能在运行时(runtime)修饰者将方法调用传递给它所修饰的真实对象在上面的例子中lrdr将方法调用传递给它修饰的真实对象frdr修饰者除了能够进行方法传递外还能够增加类的功能例如在上面的例子中lrdr能够跟蹤当前的文件流读入数据的行号
而下面的例子显示了如何在程序中使用修饰者lrdr程序将数据按行从文件中读出后加上行号输出到屏幕上
try {
LineNumberReader lrdr = new LineNumberReader(new FileReader(filename));
for(String line; (line = lrdrreadLine()) != null;)rticletxt {
Systemoutprint(lrdrgetLineNumber() + :\t + line);
}
}
catch(javaioFileNotFoundException fnfx) {
fnfxprintStackTrace();
}
catch(javaioIOException iox) {
ioxprintStackTrace();
}
修饰者的静态和动态特性
工程学上经常提到静态和动态的概念静态方法研究那些变化或位移相对较小的对象例如桥梁或建筑而动态方法研究那些变化和移动较快的对象例如发动机在软件工程中也有相应的概念静态方法研究在编译时类之间的关系而动态方法研究在运行时类参与的一些的事件在这一节中我将用UML类图来展示修饰者的静态特性用UML时序图来展示修饰者的动态特性
修饰者的静态特性
修饰者通过增加功能来修饰被修饰对象(Decorated也就是真实对象)下面的UML类图展示了修饰者和真实对象之间的关系
图 修饰者和被修饰者的关系
修饰者继承了被修饰者或者实现了被修饰者的接口同时修饰者还保存了对被修饰者实例的引用这个实例就是修饰者修饰的对象为了说明这些类在到底是如何关联的图中举了一个Java SDK的javaiopackage中的实际例子
图 一个真实的修饰模型例子
BufferedReader和FilterReader就是图中演示的抽象类他们都继承了抽象类Reader并且将方法调用传递给Reader对象由于继承了修饰者类因此LineNumberReader和PushbackReader也是修饰者类
修饰者的动态特性
在运行时修饰者将方法调用传递给被修饰者如图所示
图 修饰者的动态特性
修饰者通常将对被修饰者的调用包装起来图描述了这种特性图描述了上面的I/O例子中修饰者的动态特性
图 I/O例子中修饰者的动态特性
现在大家对修饰模式以及它的静态和动态特性有一个比较明确的认识了让我们通过一个完整的例子来说明如何在代码中实现修饰模式
排序和过滤修饰
修饰者主要是用于给被修饰者增加功能在下面的例子中我们会给Swing中的表增加排序和过滤的功能在介绍例子之前先简单介绍一下如何使用Swing中的JTable类
import javaxswing*;
import javaxswingtable*;
public class Test extends JFrame {
public static void main(String args[]) {
Test frame = new Test();
framesetTitle(Swing表的例子);
framesetBounds( );
framesetDefaultCloseOperation(JFrameDISPOSE_ON_CLOSE);
frameshow();
}
public Test() {
TableModel model = new TestModel();
getContentPane()add(new JScrollPane(new JTable(model)));
}
private static class TestModel extends AbstractTableModel {
final int rows = cols = ;
public intgetRowCount(){ return rows; }
public intgetColumnCount() { return cols; }
public Object getValueAt(int row int col) {
return ( + row + + col + );
}
}
}
该程序创建了一个×的表表对象由三个部分组成表模型视图和事件控制器表中的数据保存在表模型中视图控制数据的显示而事件控制器控制对事件的响应图是运行这个程序的结果
图 Swing表的例子
排序修饰者
图中的应用程序包含了一张两列的表一列是货物名称一列是价格通过单击列头可以根据货物的价格对表进行排序下面是这个程序的代码
图 排序修饰者的例子
//Testjava
import javaawt*;
import javaawtevent*;
import javautilLocale;
import javautilResourceBundle;
import javaxswing*;
import javaxswingtable*;
public class Test extends JFrame {
public static void main(String args[]) {
SwingApplaunch(new Test() 排序修饰者
);
}
public Test() {
// 生成修饰者的实例该实例用于修饰Swing Table原有的表模型
// 该实例必须是final的因为它会被内嵌类引用
final TableSortDecorator decorator =
new TableBubbleSortDecorator(tablegetModel());
// 将表的模型设定为修饰者因为修饰者实现了TableModel接口
// 因此Swing Table对象不知道修饰者和真实对象之间的差别
tablesetModel(decorator);
getContentPane()add(new JScrollPane(table)
BorderLayoutCENTER);
// 在界面中添加一个状态区
getContentPane()add(SwingAppgetStatusArea()
BorderLayoutSOUTH);
SwingAppshowStatus(进行排序前);
// 获得对表中列头的引用
JTableHeader hdr = (JTableHeader)tablegetTableHeader();
// 当单击鼠标单击列头时调用修饰者的sort()方法
hdraddMouseListener(new MouseAdapter() {
public void mouseClicked(MouseEvent e) {
TableColumnModel tcm = tablegetColumnModel();
int vc = tcmgetColumnIndexAtX(egetX());
int mc = nvertColumnIndexToModel(vc);
// 进行排序
decoratorsort(mc);
// 更新状态区
SwingAppshowStatus(headers[mc] + 排序中);
}
});
}
final String[] headers = { 品名 价格/每斤 };
JTable table = new JTable(new Object[][] {
{苹果}{芒果 }
{柠檬 }{香蕉}
{桔子 }