【设计模式】02-SOLID 设计原则

面向对象编程(OOP)是一种广泛应用的编程范式,它鼓励开发者通过对象来模拟现实世界。为了提高面向对象设计(OOD)的质量和可维护性,Robert C. Martin提出了 SOLID 原则,这五个原则构成了编写良好、可扩展和可维护OOP代码的基础。

SOLID是一个缩写词,代表以下设计原则(及其缩写):

  • Single Responsibility Principle (SRP)
  • Open-Closed Principle (OCP)
  • Liskov Substitution Principle (LSP)
  • Interface Segregation Principle (ISP)
  • Dependency Inversion Principle (DIP)

这五个特定的主题贯穿了对模式和软件设计的一般讨论,所以在我们深入研究设计模式之前(我知道你们都很渴望),我们将简要回顾一下SOLID原则是什么。

2.1 S - 单一职责原则(Single Responsibility Principle)

每个类应该只有一个改变的理由,即一个类应该只负责一个功能领域中的相应职责。这有助于使类更加可维护,并减少在代码需要变更时出现的复杂性。

C++ 示例

cpp 复制代码
class Logger {
public:
    void logToFile(const std::string& message) {
        // 将消息记录到文件
    }
};

class Order {
    // Order类的其他职责...

    Logger logger;

public:
    void finalizeOrder() {
        // 完成订单处理
        logger.logToFile("Order finalized.");
    }
};

在上述示例中,Logger类负责日志记录的职责,而Order类处理订单相关的业务逻辑,并使用Logger来记录日志,而不是自己处理日志逻辑。

2.2 O - 开放/封闭原则(Open/Closed Principle)

软件实体(类、模块、函数等)应该对扩展开放,对修改封闭。这意味着应该能够在不修改现有代码的情况下增加新功能。

C++ 示例

cpp 复制代码
class Shape {
public:
    virtual double area() const = 0;
};

class Rectangle : public Shape {
    double width, height;

public:
    double area() const override {
        return width * height;
    }
};

class Circle : public Shape {
    double radius;

public:
    double area() const override {
        return 3.14159 * radius * radius;
    }
};

这里,Shape是一个抽象基类,它允许派生类通过实现area方法来扩展,而无需修改基类或其他派生类。

2.3 L - 里氏替换原则(Liskov Substitution Principle)

子类型必须能够替换掉它们的基类型。这意味着如果程序中使用了基类的对象,那么可以无缝地使用其子类的对象替换,而不会影响程序的正确性。

C++ 示例

cpp 复制代码
void printArea(const Shape& shape) {
    std::cout << shape.area() << std::endl;
}

// 由于Rectangle和Circle都是Shape的子类,
// 它们的对象都可以传递给printArea函数。
Rectangle rect;
Circle circle;
printArea(rect);
printArea(circle);

在这个例子中,不管是Rectangle还是Circle对象,都可以传递给printArea函数,因为它们都遵循Shape的约定。

2.4 I - 接口隔离原则(Interface Segregation Principle)

客户端不应该被迫依赖于它们不使用的方法。接口隔离原则鼓励我们创建细粒度的接口,这样客户端只需要了解它们真正使用的方法。

C++ 示例

cpp 复制代码
class Printer {
public:
    virtual void printDocument(const Document& doc) = 0;
};

class Scanner {
public:
    virtual void scanDocument(Document& doc) = 0;
};

// 不应该强迫一个只打印机实现扫描方法,
// 或者一个只扫描仪实现打印方法。
class OfficePrinter : public Printer, public Scanner {
    void printDocument(const Document& doc) override {
        // 实现打印
    }

    void scanDocument(Document& doc) override {
        // 实现扫描
    }
};

在这里,OfficePrinter实现了Printer

Scanner接口,但如果有一个只需要打印功能的类,它就只实现Printer接口。

2.5 D - 依赖倒置原则(Dependency Inversion Principle)

高层模块不应该依赖低层模块,两者都应该依赖抽象。抽象不应该依赖细节,细节应该依赖抽象。这意味着我们应该针对接口编程,而不是针对实现编程。

C++ 示例

cpp 复制代码
class DataAccess {
public:
    virtual Data getData() = 0;
};

class FileDataAccess : public DataAccess {
    Data getData() override {
        // 从文件系统获取数据
    }
};

class Application {
    DataAccess& dataAccess;

public:
    Application(DataAccess& da) : dataAccess(da) {}

    void doWork() {
        Data data = dataAccess.getData();
        // 使用数据
    }
};

FileDataAccess fileDataAccess;
Application app(fileDataAccess);
app.doWork();

这里的Application依赖于DataAccess抽象,而不是具体的FileDataAccess,使得数据访问的具体实现可以灵活更换。

SOLID原则为面向对象设计提供了强大的指导,遵循这些原则将帮助你构建更健壮、灵活且易于维护的软件系统。在C++编程中运用这些原则,能够使你的代码质量大幅提升。

相关推荐
Hunter_pcx14 分钟前
[C++技能提升]插件模式
开发语言·c++
左手の明天1 小时前
【C/C++】C++中使用vector存储并遍历数据
c语言·开发语言·c++
PaLu-LI1 小时前
ORB-SLAM2源码学习:Initializer.cc(13): Initializer::ReconstructF用F矩阵恢复R,t及三维点
c++·人工智能·学习·线性代数·ubuntu·计算机视觉·矩阵
呆呆珝1 小时前
RKNN_C++版本-YOLOV5
c++·人工智能·嵌入式硬件·yolo
笔触狂放1 小时前
第一章 语音识别概述
人工智能·python·机器学习·语音识别
小炫y1 小时前
IBM 后端开发(二)
python
晚秋贰拾伍1 小时前
设计模式的艺术-外观模式
服务器·设计模式·外观模式
c++初学者ABC2 小时前
蓝桥杯LQ1044 求完数
c++·算法·lq蓝桥杯
胡耀超2 小时前
13.快速构建领域知识库的完整指南:结合 ChatGPT 与 Python 提升效率
开发语言·python·chatgpt·知识图谱·知识库
_GR3 小时前
2013年蓝桥杯第四届C&C++大学B组真题及代码
c语言·数据结构·c++·算法·蓝桥杯