当前位置:网站首页>【HIT-SC-MEMO5】哈工大2022软件构造 复习笔记5

【HIT-SC-MEMO5】哈工大2022软件构造 复习笔记5

2022-08-04 05:32:00 XMeow

五、可维护性

5.1 可维护性的度量

可维护性的常见度量指标

指标

  • 可维护性
  • 可扩展性
  • 灵活性
  • 可适应性
  • 可管理性
  • 支持性

实际方法

  • 继承的层数
  • 类之间的耦合度
  • 单元测试的覆盖度

聚合度与耦合度

模块化编程应该做到 高内聚 & 低耦合

Rules of Modularity Design:

  • Direct Mapping 直接映射
  • Few Interfaces 尽可能少的接口
  • Small Interfaces 尽可能小的接口
  • Explicit Interfaces 显式接口
  • Information Hiding 信息隐藏

5.2 可维护性的构造原则 SOLID

  • (SRP) The Single Responsibility Principle 单一责任原则
    - 责任:变化的原因
    - 不应该有多于1个原因让你的ADT发生变化,否则就拆分开
  • (OCP) The Open-Closed Principle 开放-封闭原则
    • 对扩展性的开放
      • 模块的行为应是可扩展的
      • 从而该模块可表现出新的行为以满足需求的变化
    • 对修改的封闭
      • 但模块自身的代码是不应被修改的
      • 扩展模块行为的一般途径是修改模块的内部实现
      • 如果一个模块不能被修改,那么它通常被认为是具有固定的行为
    • 关键的解决方案:抽象技术
  • (LSP) The Liskov Substitution Principle 里氏替换原则
    • 子类型必须能够替换其基类型
    • 派生类必须能够通过其基类的接口使用,客户端无需了解二者之间的差异
  • (ISP) The Interface Segregation Principle 接口聚合原则
    • 不能强迫客户端依赖于它们不需要的接口:只提供必需的接口(聚合接口)
    • 客户端不应依赖于它们不需要的方法
  • (DIP) The Dependency Inversion Principle 依赖转置原则
    • 抽象的模块不应依赖于具体的模块
    • 具体应依赖于抽象

5.3 面向可维护性的设计模式和构造技术

Factory Method 工厂方法模式

虚拟构造器

  • 当client不知道要创建哪个具体类的实例,或者不想在client代码中指明要具体创建的实例时,用工厂方法。
  • 定义一个用于创建对象的接口,让其子类来决定实例化哪一个类,从而使一个类的实例化延迟到其子类。

Abstract Factory 抽象工厂模式
提供接口以创建一组相关/相互依赖的对象,但不需要指明其具体类。
创建的不是一个完整产品,而是“产品族”(遵循固定搭配规则的多类产品的实例),得到的结果是:多个不同产品的object,各产品创建过程对client可见,但“搭配”不能改变。
本质上,Abstract Factory是把多类产品的factory method组合在一起

例子

  • 一个UI,包含多个窗口控件,这些控件在不同的OS中实现不同
  • 一个仓库类,要控制多个设备,这些设备的制造商各有不同,控制接口有差异

Abstract Factory vs Factory Method

Abstract FactoryFactory Method
创建多个类型对象创建一个对象
多个factory方法一个factory方法
使用组合/委派使用继承/子类型

Proxy 代理模式

  • 某个对象比较“敏感”/“私密”/“贵重”,不希望被client直接访问到,故设置proxy,在二者之间建立防火墙
    在这里插入图片描述
public class ProxyImage implements Image {
    
	private Image realImage;
	private String fileName;
	public ProxyImage(String fileName){
    
		this.fileName = fileName; // 不需要在构造的时候从文件装载
	}
	@Override
	public void display() {
    
		if(realImage == null) {
     // 如果display的时候发现没有装载,则再委派
			realImage = new RealImage(fileName); // Delegate到原来的类来成具体装载
		}
		realImage.display();
	}
}
  • 客户端client
Image image = new ProxyImage("pic.jpg");
image.display();
image.display();

new ProxyImage("pic.jpg")时仅仅是将文件名保存下来,没有加载真正的图片RealImage
在第一次调用Image.display()时,image委派Rep中的realImage进行加载,并显示;
在第二次调用,因为已经加载过,因此直接委派realImage.display()显示

Adaptor和Proxy区别

  • Adaptor目的:消除不兼容,目的是B以客户端期望的统一的方式与A建立起联系。
  • Proxy目的:隔离对复杂对象的访问,降低难度/代价,定位在访问/使用行为

Observer 观察者模式

  • 本ADT随时获取另一ADT状态变化
  • 一对多广播
  • “偶像”对“粉丝”广播

在这里插入图片描述

  • 需要广播的ADT记录着所有观察自己的Observer,在自己状态变化时,对于每个Observer调用其Update()方法进行更新,即获取该ADT的状态变化

Java里已经实现了该模式,提供了Observable抽象类(直接派生子类即可,构造“偶像”)
Java提供了Observer接口,实现该接口,构造“粉丝”

在这里插入图片描述

//偶像类
public class Subject {
    
	private List<Observer> observers = new ArrayList<Observer>();//维护一个粉丝列表
	private int state;
	public int getState() {
     return state; }
   	public void attach(Observer observer){
     observers.add(observer); } //粉丝关注偶像
	public void setState(int state) {
    
		this.state = state;
		notifyAllObservers(); //状态有变化的时候广播给粉丝
	}
	private void notifyAllObservers(){
    
		for (Observer observer : observers) 
			observer.update();
	}
}
//粉丝类
public abstract class Observer {
     //粉丝的抽象接口
	protected Subject subject;
	public abstract void update();
}
public class BinaryObserver extends Observer{
     //粉丝的具体类
	public BinaryObserver(Subject subject){
    
		this.subject = subject; // 关注偶像
		this.subject.attach(this); // 告知偶像自己关注了他
	}
	@Override
	public void update() {
     //被偶像回调,通知自己有新消息
        //可能有不同的行为
		System.out.println("Binary String: "+Integer.toBinaryString(subject.getState()));
	}
}
//
public class ObserverPatternDemo {
    
	public static void main(String[] args) {
    
		Subject subject = new Subject(); //一个偶像
		new HexaObserver(subject); //三个粉丝
		new OctalObserver(subject);
		new BinaryObserver(subject);
		System.out.println("First state change: 15");
		subject.setState(15); //偶像状态变化,虽然没有直接调用粉丝行为的代码,但确实有对粉丝的delegation
		System.out.println("Second state change: 10");
		subject.setState(10);
	}
}

Visitor 访问者模式

  • 对特定类型的object的特定操作(visit),在运行时将二者动态绑定到一起,该操作可以灵活更改,无需更改被visit的类
  • 本质上:将数据和作用于数据上的某种/些特定操作分离开来
  • 为ADT预留一个将来可扩展功能的“接入点”,外部实现的功能代码可以在不改变ADT本身的情况下通过delegation接入ADT
    在这里插入图片描述

“我”(源ADT)允许(调用this.accept())“你”(visitor)来访问我的数据(在accept()方法内委派visitor.visit())——数据源主动允许访问
使得访问方法可以变化
可以为源ADT预留功能

/* Abstract element interface (visitable) */
public interface ItemElement {
    
	public int accept(ShoppingCartVisitor visitor); //埋下一个槽
}
/* Concrete element */
public class Book implements ItemElement{
    
	private double price;
	...
	int accept(ShoppingCartVisitor visitor) {
    
		visitor.visit(this); //把自己通过这个槽传过去
	}
}
public class Fruit implements ItemElement{
    
	private double weight;
	...
	int accept(ShoppingCartVisitor visitor) {
    
		visitor.visit(this); //所有的子类都会实现这个槽
	}
}
/* Abstract visitor interface */
public interface ShoppingCartVisitor {
    
	int visit(Book book);
	int visit(Fruit fruit);
}
public class ShoppingCartVisitorImpl implements ShoppingCartVisitor {
     //一种实现
	public int visit(Book book) {
    
		int cost=0;
		if(book.getPrice() > 50) cost = book.getPrice()-5;
		else cost = book.getPrice();
		System.out.println("Book ISBN::"+book.getIsbnNumber() + " cost ="+cost);
		return cost;
	}
	public int visit(Fruit fruit) {
    
		int cost = fruit.getPricePerKg()*fruit.getWeight();
		System.out.println(fruit.getName() + " cost = "+cost);
		return cost;
	}
}
/* Client */
public class ShoppingCartClient {
    
	public static void main(String[] args) {
    
		ItemElement[] items = new ItemElement[]{
    new Book(20, "1234"),
            new Book(100, "5678"), new Fruit(10, 2, "Banana"), new Fruit(5, 5, "Apple")};
		int total = calculatePrice(items);
		System.out.println("Total Cost = " + total);
	}
	private static int calculatePrice(ItemElement[] items) {
    
		ShoppingCartVisitor visitor = new ShoppingCartVisitorImpl();
		int sum=0;
		for(ItemElement item : items)
			sum = sum + item.accept(visitor);
		return sum;
	}
}

Visitor vs Iterator

  • Iterator:以遍历的方式访问集合数据而无需暴露其内部表示,将“遍历”这项功能delegate到外部的iterator对象。
  • Visitor在特定ADT上执行某种特定操作,但该操作不在ADT内部实现,而是delegate到独立的visitor对象,客户端可灵活扩展/改变visitor的操作算法,而不影响ADT

Strategy vs Visitor

  • visitor是站在外部client的角度,灵活增加对ADT的各种不同操作(哪怕ADT没实现该操作)
  • strategy则是站在内部ADT的角度,灵活变化对其内部功能的不同配置。

State 状态模式

  • 最好不要使用if/else结构在ADT内部实现状态转换(考虑将来的扩展和修改)
  • 使用delegation,将状态转换的行为委派到独立的state对象去完成
    在这里插入图片描述

Memento 备忘录模式

  • 记住对象的历史状态,以便于“回滚”
  • defines three distinct roles
    • 需要“备忘”的类
    • 添加originator的备忘记录和恢复
    • 备忘录,记录originator对象的历史状态
      在这里插入图片描述
  • 需要“备份”的ADT的rep中只记录当前状态
  • 每次“备份”都生成一个外部的Memento对象
  • Caretaker负责掌控全部的状态备份,客户端通过它来操纵ADT的状态备份与恢复

5.4 语法驱动的构造

  • Concatenation连接:x ::= y z x matches y followed by z
  • Repetition重复:x ::= y* x matches zero or more y
  • Union选择:x ::= y | z x matches either y or z

例:解析URL:http://didit.csail.mit.edu:4949/

url ::= 'http://' hostname (':' port)? '/'
hostname ::= word '.' hostname | word '.' word
port ::= [0-9]+
word ::= [a-z]+

语法解析树:
在这里插入图片描述

原网站

版权声明
本文为[XMeow]所创,转载请带上原文链接,感谢
https://blog.csdn.net/qq_53603164/article/details/125221593