当前位置: 首页 > news >正文

设计模式-单一职责

单一职责

  • 前言
  • 1. Decorator
    • 1.1 模式介绍
    • 1.2 模式代码
      • 1.2.1 问题代码
      • 1.2.2 重构代码
    • 1.3 模式类图
    • 1.4 要点总结
  • 2. Bridge
    • 2.1 模式介绍
    • 2.2 模式代码
    • 2.2.1 问题代码
    • 2.2.2 重构代码
    • 2.3 模式类图
    • 2.4 模式总结

前言

在软件组件的设计中,如果责任划分的不清晰,使用继承得到的结果往往是随着需求的变化,子类急剧膨胀,同时充斥着重复代码,这时候的关键是划清责任。

典型模式:

  • Decorator 装饰模式
  • Bridge 桥模式

1. Decorator

1.1 模式介绍

在某些情况下我们可能会“过度地使用继承来扩展对象的功能”,由于继承为类型引入的静态特质,使得这种扩展方式缺乏灵活性;并且随着子类的增多(扩展功能的增多),各种子类的组合(扩展功能的组合)会导致更多子类的膨胀。

如何使“对象功能的扩展”能够根据需要来动态地实现?同时避免“扩展功能的增多”带来的子类膨胀问题?从而使得任何“功能扩展变化”所导致的影响将为最低?

动态(组合)地给一个对象增加一些额外的职责。就增加功能而言,Decorator模式比生成子类(继承)更为灵活(消除重复代码 & 减少子类个数)。

——《设计模式》GoF

1.2 模式代码

1.2.1 问题代码

//业务操作
class Stream{
publicvirtual char Read(int number)=0;
    virtual void Seek(int position)=0;
    virtual void Write(char data)=0;
    
    virtual ~Stream(){}
};

//主体类
class FileStream: public Stream{
public:
    virtual char Read(int number){
        //读文件流
    }
    virtual void Seek(int position){
        //定位文件流
    }
    virtual void Write(char data){
        //写文件流
    }

};

class NetworkStream :public Stream{
public:
    virtual char Read(int number){
        //读网络流
    }
    virtual void Seek(int position){
        //定位网络流
    }
    virtual void Write(char data){
        //写网络流
    }
    
};

class MemoryStream :public Stream{
public:
    virtual char Read(int number){
        //读内存流
    }
    virtual void Seek(int position){
        //定位内存流
    }
    virtual void Write(char data){
        //写内存流
    }
    
};

//扩展操作
class CryptoFileStream :public FileStream{
public:
    virtual char Read(int number){
       
        //额外的加密操作...
        FileStream::Read(number);//读文件流
        
    }
    virtual void Seek(int position){
        //额外的加密操作...
        FileStream::Seek(position);//定位文件流
        //额外的加密操作...
    }
    virtual void Write(byte data){
        //额外的加密操作...
        FileStream::Write(data);//写文件流
        //额外的加密操作...
    }
};

class CryptoNetworkStream : :public NetworkStream{
public:
    virtual char Read(int number){
        
        //额外的加密操作...
        NetworkStream::Read(number);//读网络流
    }
    virtual void Seek(int position){
        //额外的加密操作...
        NetworkStream::Seek(position);//定位网络流
        //额外的加密操作...
    }
    virtual void Write(byte data){
        //额外的加密操作...
        NetworkStream::Write(data);//写网络流
        //额外的加密操作...
    }
};

class CryptoMemoryStream : public MemoryStream{
public:
    virtual char Read(int number){
        
        //额外的加密操作...
        MemoryStream::Read(number);//读内存流
    }
    virtual void Seek(int position){
        //额外的加密操作...
        MemoryStream::Seek(position);//定位内存流
        //额外的加密操作...
    }
    virtual void Write(byte data){
        //额外的加密操作...
        MemoryStream::Write(data);//写内存流
        //额外的加密操作...
    }
};

class BufferedFileStream : public FileStream{
    //...
};

class BufferedNetworkStream : public NetworkStream{
    //...
};

class BufferedMemoryStream : public MemoryStream{
    //...
}




class CryptoBufferedFileStream :public FileStream{
public:
    virtual char Read(int number){
        
        //额外的加密操作...
        //额外的缓冲操作...
        FileStream::Read(number);//读文件流
    }
    virtual void Seek(int position){
        //额外的加密操作...
        //额外的缓冲操作...
        FileStream::Seek(position);//定位文件流
        //额外的加密操作...
        //额外的缓冲操作...
    }
    virtual void Write(byte data){
        //额外的加密操作...
        //额外的缓冲操作...
        FileStream::Write(data);//写文件流
        //额外的加密操作...
        //额外的缓冲操作...
    }
};



void Process(){

        //编译时装配
    CryptoFileStream *fs1 = new CryptoFileStream();

    BufferedFileStream *fs2 = new BufferedFileStream();

    CryptoBufferedFileStream *fs3 =new CryptoBufferedFileStream();

}

这段问题代码违反了单一职责原则:
最开始的文件流、网络流、内存流对继承流操作没问题,但是后来需要新增加密操作、缓冲操作不应该继承自上述流,因为他们只是在流操作的基础上新增了操作而已,这样依赖会使得代码急剧膨胀,达到n!

1.2.2 重构代码

//业务操作
class Stream{

publicvirtual char Read(int number)=0;
    virtual void Seek(int position)=0;
    virtual void Write(char data)=0;
    
    virtual ~Stream(){}
};

//主体类
class FileStream: public Stream{
public:
    virtual char Read(int number){
        //读文件流
    }
    virtual void Seek(int position){
        //定位文件流
    }
    virtual void Write(char data){
        //写文件流
    }

};

class NetworkStream :public Stream{
public:
    virtual char Read(int number){
        //读网络流
    }
    virtual void Seek(int position){
        //定位网络流
    }
    virtual void Write(char data){
        //写网络流
    }
    
};

class MemoryStream :public Stream{
public:
    virtual char Read(int number){
        //读内存流
    }
    virtual void Seek(int position){
        //定位内存流
    }
    virtual void Write(char data){
        //写内存流
    }
    
};

//扩展操作

DecoratorStream: public Stream{
protected:
    Stream* stream;//...
    
    DecoratorStream(Stream * stm):stream(stm){
    
    }
    
};

class CryptoStream: public DecoratorStream {
 

public:
    CryptoStream(Stream* stm):DecoratorStream(stm){
    
    }
    
    
    virtual char Read(int number){
       
        //额外的加密操作...
        stream->Read(number);//读文件流
    }
    virtual void Seek(int position){
        //额外的加密操作...
        stream::Seek(position);//定位文件流
        //额外的加密操作...
    }
    virtual void Write(byte data){
        //额外的加密操作...
        stream::Write(data);//写文件流
        //额外的加密操作...
    }
};



class BufferedStream : public DecoratorStream{
    
    Stream* stream;//...
    
public:
    BufferedStream(Stream* stm):DecoratorStream(stm){
        
    }
    //...
};




void Process(){

    //运行时装配
    FileStream* s1=new FileStream();
    
    CryptoStream* s2=new CryptoStream(s1);
    
    BufferedStream* s3=new BufferedStream(s1);
    
    BufferedStream* s4=new BufferedStream(s2);
}

这样设计的好处在于分隔了扩展功能和原功能,每个类实现其对应功能即可

1.3 模式类图

在这里插入图片描述

1.4 要点总结

  • 通过采用组合而非继承的手法, Decorator模式实现了在运行时动态扩展对象功能的能力,而且可以根据需要扩展多个功能。避免了使用继承带来的“灵活性差”和“多子类衍生问题”。
  • Decorator类在接口上表现为is-a Component的继承关系,即Decorator类继承了Component类所具有的接口。但在实现上又表现为has-a Component的组合关系,即Decorator类又使用了另外一个Component类。
  • Decorator模式的目的并非解决“多子类衍生的多继承”问题,Decorator模式应用的要点在于解决“主体类在多个方向上的扩展功能”——是为“装饰”的含义

2. Bridge

2.1 模式介绍

动机:由于某些类型的固有的实现逻辑,使得它们具有两个变化的维度,乃至多个纬度的变化。

如何应对这种“多维度的变化”?如何利用面向对象技术来使得类型可以轻松地沿着两个乃至多个方向变化,而不引入额外的复杂度?

将抽象部分(业务功能)与实现部分(平台实现)分离,使它们都可以独立地变化。

——《设计模式》GoF

2.2 模式代码

2.2.1 问题代码

假设您有一个几何Shape类,它有一对子类:CircleSquare。您想扩展此类层次结构以包含颜色,因此您计划创建RedBlue塑造子类。但是,由于您已经有两个子类,因此您需要创建四个类组合,例如BlueCircleRedSquare
在这里插入图片描述

#include <iostream>

class Shape{
public:
	virtual ~Shape() {}
	virtual void ShowColor() = 0;
	virtual void ShowShape() = 0;
};

class Circle : public Shape
{
public:
	virtual void ShowShape()
	{
		std::cout << "Circle" << std::endl;
	}
};

class Square : public Shape{
public:
	virtual void ShowShape()
	{
		std::cout << "Circle" << std::endl;
	}
};

class RedCircle : public Circle{
public:
	virtual void ShowColor()
	{
		std::cout << "Red" << std::endl;
	}
};


class BlueCircle : public Circle{
public:
	virtual void ShowColor()
	{
		std::cout << "Blue" << std::endl;
	}
};

class RedSquare: public Square{
//....
};

class BlueSquare: public Square{
//....
};

在层次结构中添加新的形状类型和颜色将使其呈指数级增长。例如,要添加三角形,您需要引入两个子类,每个颜色一个。之后,添加新颜色将需要创建三个子类,每个形状类型一个。我们越往后走,情况就越糟。

2.2.2 重构代码

class IColor{
public:
	virtual ~IColor() {}
	virtual	void ShowColor() = 0;
};

class Red : public IColor{
public:
	virtual void ShowColor()
	{
		std::cout << "Red" << std::endl;
	}
};
//....

class IShape{
public:
	virtual ~IShape() {}
	virtual void ShowShape() = 0;
	IShape(IColor* color) :_color(color)
	{}

protected:
	IColor* _color;
};

class Square :public IShape{
public:
	Square(IColor* color) :IShape(color)
	{}
	virtual void ShowShape() 
	{
		std::cout << "Square" << std::endl;
	}
};

//....

出现此问题的原因是,我们试图在两个独立维度上扩展形状类:按形状和按颜色。这是类继承中非常常见的问题。

Bridge 模式试图通过从继承切换到对象组合来解决此问题。这意味着您将其中一个维度提取到单独的类层次结构中,以便原始类将引用新层次结构的对象,而不是将其所有状态和行为都放在一个类中。

Bridge 模式建议的解决方案
您可以通过将类层次结构转换为几个相关的层次结构来防止其爆炸式增长。

按照这种方法,我们可以将与颜色相关的代码提取到它自己的类中,该类有两个子类:Red和Blue。然后,该类获得指向其中一个颜色对象的引用字段。现在,形状可以将任何与颜色相关的工作委托给链接的颜色对象。该引用将充当和类Shape之间的桥梁。从现在开始,添加新颜色不需要更改形状层次结构,反之亦然.

在这里插入图片描述

2.3 模式类图

在这里插入图片描述

2.4 模式总结

  • Bridge模式使用“对象间的组合关系”解耦了抽象和实现之间固有的绑定关系,使得抽象和实现可以沿着各自的维度来变化。所谓抽象和实现沿着各自纬度的变化,即“子类化”它们。
  • Bridge模式有时候类似于多继承方案,但是多继承方案往往违背单一职责原则(即一个类只有一个变化的原因),复用性比较差。
  • Bridge模式是比多继承方案更好的解决方法。
  • Bridge模式的应用一般在“两个非常强的变化维度”,有时一个类也有多于两个的变化维度,这时可以使用Bridge的扩展模式

相关文章:

  • 最新!Ubuntu Docker 安装教程
  • 11 Collection集合、Map集合:分类、功能、遍历、底层原理,Stream流:获取、中间方法、终结方法 (黑马Java视频笔记)
  • 电信大带宽服务器的应用场景都有哪些?
  • 21.多态
  • 【JDK17】开源应用服务器大比对
  • redis zset基本介绍以及底层实现
  • Unity音乐内存优化
  • jmeter吞吐量控制器-Throughput Controller
  • 计算机四级 - 数据库原理 - 第9章「数据库应用及安全性」
  • WebLogic XMLDecoder反序列化漏洞(CVE-2017-10271)深度解析与实战复现
  • C/C++蓝桥杯算法真题打卡(Day6)
  • 在群晖DS923+手动安装我Wordpress最新版
  • 小科普《php、jsp、asp和aspx的区别》
  • 使用JSON存储数据的场景
  • 第七章:SELinux
  • DeepSeek R1在医院后勤故障报修工单自动化处理中的路径设计
  • API调用大模型推理与第三方API实现业务整合
  • leetcode974. 和可被 K 整除的子数组
  • 现代操作系统阅读笔记(一)
  • C#语言的事务管理
  • 绵阳一村民在外务工家中老宅被拆,镇政府回应:系施工方误拆
  • 破局之路,阳光保险何以向“新”而行
  • 上海婚登人聂晶:见证爱情故事开启,也向长久婚姻致敬
  • 雀巢中国回应“巴黎水”丑闻报告:在中国销售的产品均符合相关法律法规要求
  • 金爵奖主竞赛单元评委名单公布,中国评委有黄渤、咏梅等人
  • 上海电视节评委会名单公布,陈宝国担任电视剧评委会主席