1、CENTRAL SOUTH UNIVERSITY软件体系结构实验报告实验名称 设计模式实验一 学生姓名 学生学号 XXX 专业班级 软件工程1007班 指导教师 刘伟 完成时间 2012年12月15日 实验二 设计模式实验一一、实验目的熟练使用面向对象设计原则对系统进行重构;熟练使用PowerDesigner和任意一种面向对象编程语言实现几种常见的设计模式,包括简单工厂模式、工厂方法模式、抽象工厂模式、单例模式、适配器模式、桥接模式和组合模式,理解每一种设计模式的模式动机,掌握模式结构,学习如何使用代码实现这些模式,并学会分析这些模式的使用效果。二、实验内容使用面向对象设计原则对系统进行重构;
2、使用PowerDesigner和任意一种面向对象编程语言实现简单工厂模式、工厂方法模式、抽象工厂模式、单例模式、适配器模式、桥接模式和组合模式,包括根据实例绘制模式结构图、编写模式实例模拟实现代码,运行并测试所编写代码。(1) 面向对象设计原则在某图形库API中提供了多种矢量图模板,用户可以基于这些矢量图创建不同的显示图形,图形库设计人员设计的初始类图如下所示:在该图形库中,每个图形类(如Circle、Triangle等)的init()方法用于初始化所创建的图形, setColor()方法用于给图形设置边框颜色,fill()方法用于给图形设置填充颜色,setSize()方法用于设置图形的大小,
3、display()方法用于显示图形。客户类(Client)在使用该图形库时发现存在如下问题: 由于在创建窗口时每次只需要使用图形库中的一种图形,因此在更换图形时需要修改客户类源代码; 在图形库中增加并使用新的图形时需要修改客户类源代码; 客户类在每次使用图形对象之前需要先创建图形对象,有些图形的创建过程较为复杂,导致客户类代码冗长且难以维护。现需要根据面向对象设计原则对该系统进行重构,要求如下: 隔离图形的创建和使用,将图形的创建过程封装在专门的类中,客户类在使用图形时无须直接创建图形对象,甚至不需要关心具体图形类类名; 客户类能够方便地更换图形或使用新增图形,无须针对具体图形类编程,符合开闭
4、原则。绘制重构之后的类图并说明在重构过程中所运用的面向对象设计原则。(2) 简单工厂模式使用简单工厂模式设计一个可以创建不同几何形状(Shape),如圆形(Circle)、矩形(Rectangle)和三角形(Triangle)等的绘图工具类,每个几何图形均具有绘制Draw()和擦除Erase()两个方法,要求在绘制不支持的几何图形时,抛出一个UnsupportedShapeException异常,绘制类图并编程模拟实现。(3) 工厂方法模式现需要设计一个程序来读取多种不同类型的图片格式,针对每一种图片格式都设计一个图片读取器(ImageReader),如GIF图片读取器(GifReader)用
5、于读取GIF格式的图片、JPG图片读取器(JpgReader)用于读取JPG格式的图片。图片读取器对象通过图片读取器工厂ImageReaderFactory来创建,ImageReaderFactory是一个抽象类,用于定义创建图片读取器的工厂方法,其子类GifReaderFactory和JpgReaderFactory用于创建具体的图片读取器对象。试使用工厂方法模式设计该程序,绘制类图并编程模拟实现。(4) 抽象工厂模式某软件公司欲开发一套界面皮肤库,可以对桌面软件进行界面美化。不同的皮肤将提供视觉效果不同的按钮、文本框、组合框等界面元素,其结构如下图所示:该皮肤库需要具备良好的灵活性和可扩展
6、性,用户可以自由选择不同的皮肤,开发人员可以在不修改既有代码的基础上增加新的皮肤。试使用抽象工厂模式设计该皮肤库,绘制类图并编程模拟实现。(5) 单例模式使用单例模式的思想实现多例模式,确保系统中某个类的对象只能存在有限个,例如两个或三个,设计并编写代码实现一个多例类。(6) 单例模式使用单例模式设计一个多文档窗口(注:在Java AWT/Swing开发中可使用JDesktopPane和JInternalFrame来实现),要求在主窗体中某个内部子窗体只能实例化一次,即只能弹出一个相同的子窗体,如下图所示,编程实现该功能。(注:用C#或C+实现类似功能也可以)(7) 适配器模式现有一个接口Da
7、taOperation定义了排序方法sort(int) 和查找方法search(int, int),已知类QuickSort的quickSort(int)方法实现了快速排序算法,类BinarySearch 的binarySearch (int, int)方法实现了二分查找算法。现使用适配器模式设计一个系统,在不修改源代码的情况下将类QuickSort和类BinarySearch的方法适配到DataOperation接口中。绘制类图并编程实现。(要求实现快速排序和二分查找)(8) 适配器模式Windows Media Player和RealPlayer是两种常用的媒体播放器,它们的API结构和调
8、用方法存在区别。现在你的应用程序需要支持这两种播放器API,而且在将来可能还需要支持新的媒体播放器,请问如何设计该应用程序?选择一种合适的设计模式,绘制类图并编程模拟实现。(9) 桥接模式某软件公司欲开发一个跨平台图像浏览系统,要求该系统能够显示BMP、JPG、GIF、PNG等多种格式的文件,并且能够在Windows、Linux、Unix等多个操作系统上运行。系统首先将各种格式的文件解析为像素矩阵(Matrix),然后将像素矩阵显示在屏幕上,在不同的操作系统中可以调用不同的绘制函数来绘制像素矩阵。系统需具有较好的扩展性,以便在将来支持新的文件格式和操作系统。试使用桥接模式设计该跨平台图像浏览系
9、统,绘制类图并编程模拟实现。(10) 组合模式某教育机构组织结构如下图所示:在该教育机构的OA系统中可以给各级办公室下发公文,现采用组合模式设计该机构的组织结构,绘制相应的类图并编程模拟实现,在客户端代码中模拟下发公文。三、实验要求1. 选择合适的面向对象设计原则对系统进行重构,正确无误绘制重构之后的类图;2. 正确无误绘制简单工厂模式、工厂方法模式、抽象工厂模式、单例模式、适配器模式、桥接模式和组合模式实例的模式结构图;3. 使用任意一种面向对象编程语言实现简单工厂模式、工厂方法模式、抽象工厂模式、单例模式、适配器模式、桥接模式和组合模式实例,代码运行正确无误。四、实验步骤1. 选择合适的面
10、向对象设计原则对系统进行重构,使用PowerDesigner绘制重构之后的类图;2. 使用PowerDesigner绘制简单工厂模式实例结构图并用面向对象编程语言实现该模式实例;3. 使用PowerDesigner绘制工厂方法模式实例结构图并用面向对象编程语言实现该模式实例;4. 使用PowerDesigner绘制抽象工厂模式实例结构图并用面向对象编程语言实现该模式实例;5. 使用PowerDesigner绘制单例模式实例结构图并用面向对象编程语言实现该模式实例;6. 使用PowerDesigner绘制适配器模式结构图并用面向对象编程语言实现该模式;7. 使用PowerDesigner绘制桥接
11、模式结构图并用面向对象编程语言实现该模式;8. 使用PowerDesigner绘制组合模式结构图并用面向对象编程语言实现该模式。五、实验报告要求1. 提供重构之后的类图,简要说明重构过程中所运用的面向对象设计原则;2. 提供简单工厂模式实例结构图及实现代码;3. 提供工厂方法模式实例结构图及实现代码;4. 提供抽象工厂模式实例结构图及实现代码;5. 提供单例模式实例结构图及实现代码;6. 提供适配器模式实例结构图及实现代码;7. 提供桥接模式实例结构图及实现代码;8. 提供组合模式实例结构图及实现代码;9. 注意调整类图中文字的字体及字号,注意图的整体布局和图的大小,避免使用交叉线,注意代码字
12、体和颜色,代码字体需一致,颜色需统一;10. 认真写好实验总结;11. 提交电子版实验报告,格式为“学号-姓名-实验2.doc”,如“060102-张三-实验2.doc”,所有材料(包括类图和源代码)均放在一个Word文件中,格式为doc或docx,不需要提交PowerDeigner所保存的oom模型,也无须单独提交源代码。六、实验结果1. 重构之后的类图:重构过程中所使用的面向对象设计原则及简要说明:根据题意重构后使用了在开闭原则,软件实体可以指一个软件模块、一个由多个类组成的局部结构或一个独立的类。抽象化是开闭原则的关键。 开闭原则还可以通过一个更加具体的“对可变性封装原则”来描述,对可变
13、性封装原则(Principle of Encapsulation of Variation, EVP)要求找到系统的可变因素并将其封装起来。 2. 类图:实现代码:Client.javapackage dm.simplefactory;public class Client public static void main(String args) try Shape shape;String shapName = XMLUtil.getShape();shape = ShapeFactory.createShape(shapName);shape.draw();shape.erase(); c
14、atch (Exception e) e.printStackTrace();XMLUtil.javapackage dm.simplefactory;import javax.xml.parsers.DocumentBuilder;import javax.xml.parsers.DocumentBuilderFactory;import org.w3c.dom.Document;import org.w3c.dom.Node;import org.w3c.dom.NodeList;public class XMLUtil public static String getShape() tr
15、y DocumentBuilderFactory dFactory = DocumentBuilderFactory.newInstance();DocumentBuilder builder = dFactory.newDocumentBuilder();Document doc;doc = builder.parse(configShape.xml);NodeList nl = doc.getElementsByTagName(shapeName);Node classNode = nl.item(0).getFirstChild();String shapeName = classNod
16、e.getNodeValue().trim();return shapeName; catch (Exception e) e.printStackTrace();return null;ShapeFactory.javapackage dm.simplefactory;public class ShapeFactory public static Shape createShape(String shape) throws Exception if (shape.equalsIgnoreCase(Circle) System.out.println(圆形);return new Circle
17、(); else if (shape.equalsIgnoreCase(Rectangle) System.out.println(长方形);return new Rectangle(); else if (shape.equalsIgnoreCase(Triangle) System.out.println(三角形);return new Rectangle(); else throw new Exception(UnsupportedShapeException);Shape.javapackage dm.simplefactory;public interface Shape void
18、draw();void erase();Triangle.javapackage dm.simplefactory;public class Triangle implements Shape public void draw() System.out.println(画三角形);public void erase() System.out.println(擦除三角形);Rectangle.javapackage dm.simplefactory;public class Rectangle implements Shape public void draw() System.out.prin
19、tln(画长方形);public void erase() System.out.println(擦除长方形);Circle.javapackage dm.simplefactory;public class Circle implements Shape public void draw() System.out.println(画圆形);public void erase() System.out.println(擦除圆形);3. 类图:实现代码:Client.javapackage dm.factory;public class Client public static void mai
20、n(String args) try ImageReader ir;ImageReaderFactory factory;factory = (ImageReaderFactory)XMLUtil.getBean();ir = factory.createReader();ir.readImage(); catch (Exception e) e.printStackTrace();XMLUtil.javapackage dm.simplefactory;import javax.xml.parsers.DocumentBuilder;import javax.xml.parsers.Docu
21、mentBuilderFactory;import org.w3c.dom.Document;import org.w3c.dom.Node;import org.w3c.dom.NodeList;public class XMLUtil public static String getShape() try DocumentBuilderFactory dFactory = DocumentBuilderFactory.newInstance();DocumentBuilder builder = dFactory.newDocumentBuilder();Document doc;doc
22、= builder.parse(configShape.xml);NodeList nl = doc.getElementsByTagName(shapeName);Node classNode = nl.item(0).getFirstChild();String shapeName = classNode.getNodeValue().trim();return shapeName; catch (Exception e) e.printStackTrace();return null;ImageReaderFactory.javapackage dm.factory;public abs
23、tract class ImageReaderFactory public abstract ImageReader createReader();ImageReader.javapackage dm.factory;public interface ImageReader void readImage();JpgReaderFactory.javapackage dm.factory;public class JpgReaderFactory extends ImageReaderFactory public ImageReader createReader() System.out.pri
24、ntln(Jpg创建中);return new JpgReader();GifReaderFactory.javapackage dm.factory;public class GifReaderFactory extends ImageReaderFactory public ImageReader createReader() System.out.println(Gif创建中);return new GifReader();JpgReader.javapackage dm.factory;public class JpgReader implements ImageReader publ
25、ic void readImage() System.out.println(读取Jpg图片);GifReader.javapackage dm.factory;public class GifReader implements ImageReader public void readImage() System.out.println(读取Gif图片);4. 类图:实现代码:Client.javapackage dm.abstractfactory;public class Client public static void main(String args) try Factory fac
26、tory;Button bt;Combobox cb;Textbox tb;factory = (Factory) XMLUtil.getBean();bt = factory.createButton();bt.show();tb = factory.createTextbox();tb.show();cb = factory.createCombobox();cb.show(); catch (Exception e) System.out.println(e.getMessage();XMLUtil.javapackage dm.abstractfactory;import java.i
27、o.File;import javax.xml.parsers.DocumentBuilder;import javax.xml.parsers.DocumentBuilderFactory;import org.w3c.dom.Document;import org.w3c.dom.Node;import org.w3c.dom.NodeList;public class XMLUtil / 该方法用于从XML配置文件中提取具体类类名,并返回一个实例对象public static Object getBean() try / 创建文档对象DocumentBuilderFactory dFac
28、tory = DocumentBuilderFactory.newInstance();DocumentBuilder builder = dFactory.newDocumentBuilder();Document doc;doc = builder.parse(new File(configStyle.xml);/ 获取包含类名的文本节点NodeList nl = doc.getElementsByTagName(className);Node classNode = nl.item(0).getFirstChild();String cName = classNode.getNodeVa
29、lue();/ 通过类名生成实例对象并将其返回SuppressWarnings(rawtypes)Class c = Class.forName(cName);Object obj = c.newInstance();return obj; catch (Exception e) e.printStackTrace();return null;Factory.javapackage dm.abstractfactory;public interface Factory Button createButton();Textbox createTextbox();Combobox createCo
30、mbobox();SummerStyleFactory.javapackage dm.abstractfactory;public class SummerStyleFactory implements Factory public Button createButton() return new ShallowBlueButton();public Textbox createTextbox() return new BlueBorderTextbox();public Combobox createCombobox() return new BlueBorderCombobox();Spr
31、ingStyleFactory.javapackage dm.abstractfactory;public class SpringStyleFactory implements Factory public Button createButton() return new LightGreenButton();public Textbox createTextbox() return new GreenBorderTextbox();public Combobox createCombobox() return new GreenBorderCombobox();Textbox.javapa
32、ckage dm.abstractfactory;public interface Textbox void show();Combobox.javapackage dm.abstractfactory;public interface Combobox void show();Button.javapackage dm.abstractfactory;public interface Button void show();LightGreenButton.javapackage dm.abstractfactory;public class LightGreenButton implemen
33、ts Button public void show() System.out.println(Spring风格-浅绿色按钮);GreenBorderTextbox.javapackage dm.abstractfactory;public class GreenBorderTextbox implements Textbox public void show() System.out.println(Spring风格-绿色边框文本);GreenBorderCombobox.javapackage dm.abstractfactory;public class GreenBorderCombo
34、box implements Combobox public void show() System.out.println(Spring风格-绿色边框组合框);ShallowBlueButton.javapackage dm.abstractfactory;public class ShallowBlueButton implements Button public void show() System.out.println(Summer风格-浅蓝色按钮);BlueBorderTextbox.javapackage dm.abstractfactory;public class BlueBo
35、rderTextbox implements Textbox public void show() System.out.println(Summer风格-蓝色边框文本框);BlueBorderCombobox.javapackage dm.abstractfactory;public class BlueBorderCombobox implements Combobox public void show() System.out.println(Summer风格-蓝色边框组合框);5. 类图:实现代码:Client.javapackage dm.singleton1;public clas
36、s Client public static void main(String args) MySingleton s1 = MySingleton.getInstance();MySingleton s2 = MySingleton.getInstance();System.out.println(由单例生成多例1 + s1);System.out.println(由单例生成多例2 + s2);System.out.println(生成的两个对象不是一样 + (s1 = s2);MySingleton.javapackage dm.singleton1;public class MySing
37、leton private static MySingleton instance = new MySingleton2;private MySingleton() public static MySingleton getInstance() if (instance = null)instance = new MySingleton1;elseinstance = new MySingleton2;return instance;6. 类图:实现代码:Client.javapackage dm.singleton2;public class Client public static voi
38、d main(String args) try MainFrame mf1 = null;MainFrame mf2 = null;mf1 = MainFrame.getInstance();mf2 = MainFrame.getInstance();System.out.println(mf1=mf2?);System.out.println(mf1 = mf2); catch (Exception e) e.printStackTrace();MainFrame.javapackage dm.singleton2;import java.awt.*;import java.awt.even
39、t.*;import javax.swing.*;public class MainFrame extends JFrame implements ActionListener private static final long serialVersionUID = 1L;private static MainFrame subFrame = null;static JDesktopPane desktopPane;static JInternalFrame internalFrame;private MainFrame() super(MainFrame);this.setLocation(
40、500, 250);Container contentPane = this.getContentPane();contentPane.setLayout(new BorderLayout();JButton button = new JButton(Crate New Internal Frames);button.addActionListener(this);contentPane.add(button, BorderLayout.SOUTH);desktopPane = new JDesktopPane();contentPane.add(desktopPane);/ 將虛擬桌面加入到
41、content pane中this.setSize(new Dimension(300, 300);this.setVisible(true);this.addWindowListener(new WindowAdapter() Overridepublic void windowClosing(WindowEvent e) System.exit(0););Overridepublic void actionPerformed(ActionEvent e) internalFrame = new JInternalFrame(Internale Frame, true, true, true
42、,true);internalFrame.setSize(new Dimension(200, 200);internalFrame.setVisible(true);Container icontentPane = internalFrame.getContentPane();JTextArea textArea = new JTextArea();icontentPane.add(textArea, Center);desktopPane.add(internalFrame);/ 將internalFrame加入到desktopPane,通過虛擬桌面來管理internalFrametry internalFrame.setSelected(true);/ 獲得焦點 catch (java.beans.PropertyVetoException ex) System.out.println(Exception while selecting);public static MainFrame getInstance() if (subFrame = null) subFrame = new MainFrame(); else JOptionPane.showMessageDialog(null, SubFrame已经在运行!);return subFrame;