《软件构造》 第五章 基于可复用性的软件构造处理

》》可复用性

高复用性的软件应具有如下特性:小、简单;与标准兼容;灵活可变;可扩展;泛型、参数化;模块化;变化的局部性;稳定;丰富的文档和帮助

白盒复用:源代码可见,可修改和扩展。复杂度高,需要对内部代码充分了解

黑盒复用:源代码不可见,无法修改代码;简单清晰,适应性差

可复用性外部观察:

——类型可变:能够复用的部分应该类型参数化,以适应不同的数据类型;复用的部分应该一般化;适应不同的类型,且满足LSP(子类一定可以替代父类)

——实现可变:ADT 有多种不同的实现,提供不同的representations 和abstract function ,但具有同样的specification (pre-condition, post-condition, invariants) ,从而可以适应不同的应用场景

——功能分组:提供完备的细粒度操作,保证功能的完整性,不同场景下复用不同的操作( 及其组合)

——表示独立:内部实现可能会经常变化,但客户端不应受到影响。

——共性抽取:将共同的行为(共性)抽象出来,形成可复用实体

LSP:更强的不变量;前置条件更弱后置条件更强

协变:父类型->子类型:越来越具体specific 返回值类型:不变或变得更具体;异常的类型:也是如此。

反协变、逆变:父类型->子类型:越来越具体specific 参数类型:要相反的变化,要不变或越来越抽象(在java中不允许)

数组的子类型化:

——数组是协变的:一个数组T[ ] ,可能包含了T类型的实例或者T的任何子类型的实例

——即子类型的数组可以赋予父类型的数组进行使用,但数组的类型实际为子类型。

——下面报错的原因是myNumber指向的还是一个Integer[] 而不是Number[]

Number[] numbers = new Number[2]; 
numbers[0] = new Integer(10); 
numbers[1] = new Double(3.14);
Integer[] myInts = {1,2,3,4}; 
Number[] myNumber = myInts;
myNumber[0] = 3.14; //run-time error!

泛型的子类型化:

泛型的类型是不变的:ArrayList is a subtype of List; List is not a subtype of List

通配符:? 表示所有满足条件的未知类型 List is a subtype of List 

List is a subtype of List 

List is a subtype of List

委派和组合:

比较器(Comparator):

《软件构造》 第五章 基于可复用性的软件构造处理_第1张图片

另一种方法:让你的ADT实现Comparable接口,然后override compareTo() 方法。与使用Comparator的区别:不需要构建新的Comparator类,比较代码放在ADT内部。下面为具体例子。

《软件构造》 第五章 基于可复用性的软件构造处理_第2张图片

委派(delegation):在一个类中调用另一个类的方法

——显性委派:将发送对象传递给接收对象。

——隐性委派:由语言的成员查找规则。

可以看到,想在B中调用A,需要先委派一个A

《软件构造》 第五章 基于可复用性的软件构造处理_第3张图片

CRP(合成复用原则):委派可以看做Object层面的复用机制,而继承可以看做是类的层面

在委派时多使用接口,实例化接口然后重写接口的方法

《软件构造》 第五章 基于可复用性的软件构造处理_第4张图片

委派的种类:

——临时性委派(Dependency):最简单的方法,调用类里的方法(use a),其中一个类使用另一个类而不实际地将其作为属性。

——永久性委派(Association):类之中有其它类的具体实例来作为一个变量(has a)。

——更强的委派,组合(Composition):更强的委派。将一些简单的对象组合成一个更为复杂的对象。(is part of)。里面其中一个对象损坏,则组合对象损坏

——聚合(Aggregation):对象是在类的外部生成的,然后作为一个参数传入到类的内部构造器。(has a)


可复用的库和框架:

白盒框架和黑盒框架:白盒一般是继承,黑盒一般是委派、组合

——白盒框架是基于面向对象的继承机制。之所以说是白盒框架,是因为在这种框架中,父类的方法对子类而言是可见的。子类可以通过继承或重写父类的方法来实现更具体的方法。虽然层次结构比较清晰,但是这种方式也有其局限性,父类中的方法子类一定拥有,要么继承,要么重写,不可能存在子类中不存在的方法而在父类中存在。

public abstract class PrintOnScreen {
    public void print() { 
        JFrame frame = new JFrame(); 
        JOptionPane.showMessageDialog(frame, textToShow());
        frame.dispose();
    } 
    protected abstract String textToShow(); 
}
public class MyApplication extends PrintOnScreen {
@Override protected String textToShow() {
        return "printing this text on " + "screen using PrintOnScreen " + "white Box Framework"; 
    }
}

——黑盒框架时基于委派的组合方式,是不同对象之间的组合。之所以是黑盒,是因为不用去管对象中的方法是如何实现的,只需关心对象上拥有的方法。这种方式较白盒框架更为灵活,因为可以在运行时动态地传入不同对象,实现不同对象间的动态组合;而继承机制在静态编译时就已经确定好。黑盒框架和白盒框架可以互相转化。

public interface TextToShow { 
    String text(); 
}

public class MyTextToShow implements TextToShow {
    @Override 
    public String text() { 
        return "Printing"; 
    }
}

public final class PrintOnScreen {
    TextToShow textToShow;   
    public PrintOnScreen(TextToShow tx) { 
        this.textToShow = tx; 
    }
    public void print() { 
        JFrame frame = new JFrame(); 
        JOptionPane.showMessageDialog(frame, textToShow.text());
        frame.dispose(); 
    }
}

》》复用的设计模式

——结构型模式:Structural patterns

适配器模式(Adapter)

装饰器模式(Decorator )

外观模式(Facade)

——行为类模式:Behavioral patterns 

策略模式(Strategy)

模板方法模式(Template method)

迭代器模式( Iterator)

适配器模式:将某个类或接口转换为用户期望的其他形式

实例:问题描述:其中LegacyRectangle是已有的类(需要传入矩形的一个顶点、长和宽),但是与client要求的接口不一致(需要给出对角线两个顶点坐标),我们此时建立一个新的接口Shape以供客户端调用,用户通过Shape接口传入两个点的坐标。Rectangle作为Adapter,实现该抽象接口,通过具体的方法实现适配。

  在不适用适配器时:会发生委派不相容。

《软件构造》 第五章 基于可复用性的软件构造处理_第5张图片

装饰器模式:为对象增加不同侧面的特性,对每一个特性构造子类,通过委派增加到对象上

《软件构造》 第五章 基于可复用性的软件构造处理_第6张图片

《软件构造》 第五章 基于可复用性的软件构造处理_第7张图片

Stack s = new ArrayStack(); //构建一个普通的堆栈
UndoStack s = new UndoStack(new ArrayStack()); //构建撤消堆栈
SecureStack s = new SecureStack( new SynchronizedStack( new UndoStack(s)))//构建安全的同步撤消堆栈


外观模式:客户通过一个简化的接口来访问复杂系统内的功能。把多种相似类型的类及其功能用一个类封装起来,通过传参在内部完成选择

《软件构造》 第五章 基于可复用性的软件构造处理_第8张图片

《软件构造》 第五章 基于可复用性的软件构造处理_第9张图片

客户端代码

《软件构造》 第五章 基于可复用性的软件构造处理_第10张图片

策略模式:一个类的行为或算法可以在运行时更改。针对特定任务存在不同的算法,但客户端可以根据动态上下文在运行时切换算法。就是在一个接口下设计多个子类,每个子类有自己的功能算法。在运行时调用接口的不同子类实现不同选择。

模板模式:模板模式中,一个抽象类公开定义了执行它的方法的方式/模式,它的子类可以按照需要重写实现方法,但调用将以抽象类中定义的方法进行。

例子,对于边的定义时,无向边,有向边只需要重写边的方法,或者再增加一些方法。

迭代器模式:这种模式用于顺序访问集合对象的元素,而又无需暴露该对象的内部表示。用途:解决客户需要统一的策略来访问容器中的所有元素,与容器类型无关。实现方法:这种模式让自己的集合类实现Iterator接口,并实现自己的独特Iterator迭代器(hasNext, next, remove),允许客户端利用这 个迭代器进行显式或隐式的迭代遍历

例子:

《软件构造》 第五章 基于可复用性的软件构造处理_第11张图片

《软件构造》 第五章 基于可复用性的软件构造处理_第12张图片


你可能感兴趣的:(《软件构造》 第五章 基于可复用性的软件构造处理)