本文主要是介绍面向对象设计的五大原则:SOLID原则(聚合和耦合)_v0.1.0,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!
文章目录
- 聚合、组合与耦合
- 1. **聚合(Aggregation)**
- 2. **组合(Composition)**
- 3. **耦合(Coupling)**
- 比较总结
- 总结
- 面向对象设计五大原则:SOLID原则
- 1. 单一职责原则(Single Responsibility Principle,SRP)
- 原则解释
- 举例说明
- 初始设计(不符合单一职责原则)
- 改进后
- 2. 开闭原则(Open/Closed Principle,OCP)
- 原则解释
- 举例说明
- 初始设计(不符合开闭原则)
- 改进设计(符合开闭原则)
- 3. 里氏替换原则(Liskov Substitution Principle,LSP)
- 原则解释
- 举例说明
- 初始设计(违反里氏替换原则)
- 改进设计(符合里氏替换原则)
- 遵循 LSP 的几个要点
- 里氏替换原则的好处
- 4. 接口隔离原则(Interface Segregation Principle,ISP)
- 原则解释
- 举例说明
- 初始设计(违反接口隔离原则)
- 改进设计(符合接口隔离原则)
- 遵循 ISP 的几个要点
- 接口隔离原则的好处
- 5. 依赖倒置原则(Dependency Inversion Principle,DIP)
- 原则解释
- 举例说明
- 初始设计(违反依赖倒置原则)
- 改进设计(符合依赖倒置原则)
- 遵循 DIP 的几个要点
- 依赖倒置原则的好处
更新记录
时间 | 版本 | 内容 | 修订者 | 备注 |
---|---|---|---|---|
2024/08/22 | 0.1.0 | 创建 | henry.xu |
聚合、组合与耦合
这三个概念虽然都与对象之间的关系有关,但它们在设计意图和应用场景上有显著不同。以下是对聚合、组合和耦合的整理和比较:
1. 聚合(Aggregation)
-
定义: 聚合是一种“整体-部分”关系,其中整体对象包含或拥有部分对象,但部分对象的生命周期独立于整体对象。换句话说,部分对象可以在没有整体对象的情况下存在。
-
关系: 整体和部分之间的关系是松散的。整体对象只是引用部分对象,而不强制要求部分对象依赖于整体。
-
示例: 部门与员工的关系。部门(整体)可以拥有多个员工(部分),但员工可以独立于部门存在。如果一个员工离开部门,他仍然是一个独立的个体。
-
应用场景: 适用于当部分对象需要在多个整体对象之间共享或复用,或部分对象的生命周期与整体不完全绑定的情况。
2. 组合(Composition)
-
定义: 组合也是一种“整体-部分”关系,但它比聚合更紧密。组合中,部分对象的生命周期依赖于整体对象。如果整体对象被销毁,部分对象也会随之销毁。
-
关系: 整体和部分之间的关系是紧密绑定的。部分对象无法独立于整体对象存在。
-
示例: 汽车与引擎的关系。汽车(整体)包含引擎(部分),如果汽车被销毁,引擎也随之销毁,因为引擎依赖于汽车存在。
-
应用场景: 适用于当部分对象必须严格依赖于整体对象存在,且不希望部分对象被其他整体对象共享的情况。
3. 耦合(Coupling)
-
定义: 耦合描述的是两个或多个模块或类之间的依赖程度。高耦合意味着模块或类之间的依赖性强,低耦合则意味着它们之间的依赖性弱。耦合程度通常反映了系统的模块化和可维护性。
-
关系: 耦合不一定是“整体-部分”关系,而是更广泛的模块或类之间的关联程度。耦合可以是由于依赖、继承、接口实现等多种原因引起的。
-
示例: 假设有两个类
A
和B
,如果A
的实现严重依赖于B
,那么A
和B
之间是高耦合的。如果A
和B
之间仅通过接口通信且相对独立,则它们是低耦合的。 -
应用场景: 在系统设计中,通常追求低耦合,以提高系统的模块化和灵活性,使得修改一个模块不会对其他模块产生过多影响。
比较总结
-
聚合与组合: 这两者都描述了“整体-部分”关系,但聚合关系较为松散,部分对象可以独立于整体对象存在;而组合关系紧密,部分对象的生命周期完全依赖于整体对象。生命周期是描述一个对象或实体在系统中从创建到销毁所经历的各个阶段的概念。理解生命周期有助于有效管理资源,避免内存泄漏,控制系统复杂性,并编写更健壮的代码。
-
聚合/组合与耦合: 聚合和组合主要描述类或对象之间的结构性关系,而耦合则描述类或模块之间的依赖程度。高耦合系统往往难以维护,而聚合和组合的合理使用可以帮助降低耦合度,使系统更易于扩展和维护。
总结
- 聚合: 松散的“整体-部分”关系,部分对象可以独立存在。
- 组合: 紧密的“整体-部分”关系,部分对象的生命周期依赖于整体。
- 耦合: 类或模块之间的依赖程度,追求低耦合以提高系统的可维护性和灵活性。
理解这三者之间的关系和差异有助于设计更加模块化、灵活和可维护的软件系统。
面向对象设计五大原则:SOLID原则
1. 单一职责原则(Single Responsibility Principle,SRP)
单一职责原则(Single Responsibility Principle,SRP)是面向对象设计中的一个重要原则。它的核心思想是:一个类应该只有一个引起它变化的原因。换句话说,一个类应当只负责一项职责。这样设计的好处是提高了代码的可读性、可维护性和可扩展性。
原则解释
在软件开发中,“职责”可以理解为类的功能或用途。如果一个类负责太多的职责,当某一职责发生变化时,类可能需要修改。而修改类可能会影响它的其他职责,从而增加了代码出错的风险。
SRP 强调,每个类只应专注于一项职责。如果有多项职责,应该考虑将这些职责分离到不同的类中。这样,每当某一职责发生变化时,只需修改对应的类,而不必担心影响其他职责。
- 作用对象:类
- 作用: 降低耦合,聚合关系的清晰化;尽管单一职责原则有助于减少类内部的复杂性,但它并不能完全避免类之间的耦合。
举例说明
初始设计(不符合单一职责原则)
假设我们有一个用户管理系统的类 UserManager
,它有以下职责:
- 处理用户数据(如创建用户、删除用户)。
- 发送电子邮件通知。
class UserManager:def create_user(self, username, email):# 处理创建用户的逻辑print(f"User {username} created.")self.send_email(email, "Welcome!", "Thanks for joining us!")def delete_user(self, username):# 处理删除用户的逻辑print(f"User {username} deleted.")def send_email(self, email_address, subject, body):# 发送电子邮件的逻辑print(f"Sending email to {email_address} with subject: {subject}")
在这个例子中,UserManager
类同时承担了用户管理和发送电子邮件的职责。如果将来发送电子邮件的逻辑发生变化(例如引入新的电子邮件服务),我们必须修改 UserManager
类,这就违背了单一职责原则。
改进后
我们可以将发送电子邮件的职责分离到另一个类 EmailService
中。
class EmailService:def send_email(self, email_address, subject, body):# 发送电子邮件的逻辑print(f"Sending email to {email_address} with subject: {subject}")class UserManager:def __init__(self, email_service):self.email_service = email_servicedef create_user(self, username, email):# 处理创建用户的逻辑print(f"User {username} created.")self.email_service.send_email(email, "Welcome!", "Thanks for joining us!")def delete_user(self, username):# 处理删除用户的逻辑print(f"User {username} deleted.")
现在,UserManager
类只负责用户管理,而 EmailService
类负责发送电子邮件。如果将来电子邮件服务发生变化,只需修改 EmailService
类即可,不需要改动 UserManager
。
通过将不同的职责分离到独立的类中,不仅可以提升代码的可读性,还能使代码更易于测试和维护。如果一个类负责的职责太多,当其中的某个职责发生变化时,整个类可能需要进行大规模的修改,这不利于代码的扩展性。SRP 帮助开发者创建更简洁、功能单一的类,从而提高代码的质量。
2. 开闭原则(Open/Closed Principle,OCP)
开闭原则(Open/Closed Principle, OCP)是面向对象设计的五大原则之一(即SOLID原则中的“O”)。它的核心思想是:软件实体(如类、模块、函数等)应该对扩展开放,对修改关闭。
原则解释
- 对扩展开放:意味着当需求变化或增加时,我们应该能够通过增加新代码来扩展软件实体的功能,而不是修改已有的代码。
- 对修改关闭:意味着在实现新功能或满足新需求时,不应该改变已经存在的代码。现有的代码一旦通过测试并投入使用,就应该被视为“封闭的”。
- 作用对象:类,模块,函数等;
- 作用:降低耦合的同时允许扩展,聚合关系的可扩展性;这个原则的主要目的是提高代码的可维护性和灵活性,避免因为修改已有代码而引入新的错误。
举例说明
假设我们要开发一个图形绘制程序,可以绘制不同的形状,例如圆形和矩形。最开始,我们可能会有一个简单的类来处理这些形状的绘制。
初始设计(不符合开闭原则)
def draw_circle():print("draw circle")def draw_rectangle():print("draw rectangle")def draw_item(item):if item == "circle":draw_circle()elif item == "rectangle":draw_rectangle()# 使用:
draw_item("circle")
draw_item("rectangle")
在这个设计中,通过 draw_item
方法来绘制不同的形状。但如果我们需要增加新的形状,比如三角形或五边形,我们就需要修改 draw_item
方法,添加新的条件判断。这违背了开闭原则,因为每次添加新功能都要修改已有代码。
改进设计(符合开闭原则)
from abc import ABC, abstractmethodclass Shape:@abstractmethoddef draw(self):passclass Circle(Shape):def draw(self):print("draw circle")class Rectangle(Shape):def draw(self):print("draw rectangle")# 使用:
shapes = [Circle, Rectangle]
for shape in shapes:shape().draw()
在这个改进的设计中,我们将不同的形状分别设计成独立的类,并且这些类都继承了抽象基类 Shape
。每个形状类都实现了自己的 draw
方法。
- 对扩展开放:当我们需要添加新的形状时,比如三角形,只需要创建一个新的类
Triangle
并实现draw
方法,而不需要修改Shape
类或其他形状类。 - 对修改关闭:已经实现的
Circle
和Rectangle
类不会因为我们添加新形状而被修改。
这种设计方式遵循了开闭原则,使得系统更容易扩展,并且减少了对已有功能的影响,从而提高了系统的稳定性和可维护性。
3. 里氏替换原则(Liskov Substitution Principle,LSP)
里氏替换原则(Liskov Substitution Principle, LSP)是面向对象设计的五大原则之一(即 SOLID 原则中的“L”)。它的核心思想是:子类对象应该能够替换掉基类对象,并且不改变程序的正确性。
原则解释
- 子类型替换:LSP 要求子类必须能够替代其基类而不影响程序的行为。换句话说,基类中的方法应该能够在子类中正确地实现,而不会导致系统行为的不一致。
- 一致性:子类应当在行为上与基类保持一致。子类可以增加新的行为,但不能改变基类中已有行为的定义或语义。
- 作用对象:类与其子类;
- 作用:确保系统的可替换性、可扩展性和可维护性。遵循 LSP 可以避免子类引入的错误,保持系统行为的一致性。
举例说明
假设我们在开发一个计算面积的程序,有一个 Rectangle
类和一个 Square
类。最开始,我们可能设计了如下的类结构。
初始设计(违反里氏替换原则)
class Rectangle:def __init__(self, width, height):self.width = widthself.height = heightdef set_width(self, width):self.width = widthdef set_height(self, height):self.height = heightdef area(self):return self.width * self.heightclass Square(Rectangle):def set_width(self, width):self.width = widthself.height = widthdef set_height(self, height):self.width = heightself.height = heightdef print_area(rectangle):rectangle.set_width(5)rectangle.set_height(4)print(rectangle.area())# 使用:
rect = Rectangle(2, 3)
print_area(rect) # 输出: 20square = Square(2, 2)
print_area(square) # 输出: 16
在这个设计中,Square
类继承了 Rectangle
类,并且重写了 set_width
和 set_height
方法,以保证正方形的宽和高总是相等的。然而,这个设计违背了 LSP,因为当我们用 Square
替换 Rectangle
时,print_area
函数的行为发生了变化。
Rectangle
的print_area
期望在设置宽度和高度后,计算一个普通矩形的面积。但对于Square
,在设置宽度后,高度会自动调整,这导致area
的计算结果不同。
改进设计(符合里氏替换原则)
为了符合 LSP,我们应该避免让子类改变基类的行为。可以通过重新设计类结构,使用组合而非继承来解决这个问题。
class Shape:def area(self):passclass Rectangle(Shape):def __init__(self, width, height):self.width = widthself.height = heightdef area(self):return self.width * self.heightclass Square(Shape):def __init__(self, side):self.side = sidedef area(self):return self.side * self.sidedef print_area(shape):print(shape.area())# 使用:
rect = Rectangle(5, 4)
print_area(rect) # 输出: 20square = Square(4)
print_area(square) # 输出: 16
在这个改进设计中,Rectangle
和 Square
都继承了 Shape
类,各自实现了自己的 area
方法。
- 子类型替换:现在,无论传递
Rectangle
还是Square
对象给print_area
函数,程序的行为都符合预期。 - 一致性:
Square
和Rectangle
的行为保持一致,各自计算自身的面积,而不会因类的不同导致不一致的行为。不一致的行为主要体现在 “修改一个属性导致另一个属性的意外改变”,这打破了对类行为的一致性预期,导致使用子类替换基类时,程序无法按照原来的逻辑正确运行。这正是违反里氏替换原则的体现。
遵循 LSP 的几个要点
- 行为一致性:子类不应改变基类中已经实现的行为,子类的行为应该是基类行为的扩展或特化,而不是反转或破坏。
- 方法重写:如果子类重写了基类的方法,重写的方法必须符合基类方法的预期和约定。
- 接口设计:当设计基类时,要确保其接口足够通用,子类能够方便地继承并扩展这些接口,而不会违反 LSP。
里氏替换原则的好处
- 增强代码的可替换性:子类可以在不修改客户端代码的情况下替换基类,从而提高系统的灵活性。
- 减少代码的耦合度:通过遵循 LSP,可以减少系统中不必要的耦合,降低代码的维护成本。
- 提高系统的稳定性:遵循 LSP 可以避免子类引入的不一致行为,从而提高系统的稳定性和可靠性。
LSP 是面向对象设计中的关键原则之一,它保证了继承关系的正确性和一致性,有助于创建稳定且可扩展的系统。
4. 接口隔离原则(Interface Segregation Principle,ISP)
接口隔离原则(Interface Segregation Principle, ISP)是 SOLID 原则中的“ I ”,它的核心思想是:客户端不应该被迫依赖它们不使用的接口。这意味着一个接口应该只包含客户所需的方法,避免包含那些不被使用的方法,从而使得接口更加专注和简洁。
原则解释
- 接口的专一性:一个接口应该只包含一个特定功能的相关方法。接口中的方法应该是相关的,避免将不相关的功能集中到一个接口中。
- 减少依赖:通过将接口划分成多个专门的接口,减少类对不必要接口的依赖。每个接口应该只提供客户端所需要的功能。
- 接口的契约:每个接口应该清晰地定义一个合同,客户只需要知道接口提供了哪些功能,而不需要了解其他不相关的功能。
举例说明
假设我们在设计一个打印机系统,有一个 MultiFunctionPrinter
类,这个类实现了打印、扫描、复印等多种功能。我们可以设计如下的接口:
初始设计(违反接口隔离原则)
class MultiFunctionPrinter:def print_document(self, document):print("Printing document:", document)def scan_document(self, document):print("Scanning document:", document)def copy_document(self, document):print("Copying document:", document)class DocumentProcessor:def __init__(self, printer):self.printer = printerdef process(self, document):self.printer.print_document(document)# DocumentProcessor 不需要 scan_document 和 copy_document 方法# 但是由于接口不够专一,这里的设计就违背了 ISP
在这个设计中,DocumentProcessor
类依赖于 MultiFunctionPrinter
的接口,但它只需要 print_document
方法。由于 MultiFunctionPrinter
实现了多个不相关的功能,这导致 DocumentProcessor
被迫依赖它不需要的功能。
改进设计(符合接口隔离原则)
为了符合 ISP,我们应该将 MultiFunctionPrinter
的接口拆分成多个专门的接口,每个接口只包含特定的功能:
class Printer:def print_document(self, document):passclass Scanner:def scan_document(self, document):passclass Copier:def copy_document(self, document):passclass DocumentProcessor:def __init__(self, printer: Printer):self.printer = printerdef process(self, document):self.printer.print_document(document)
在改进设计中,Printer
、Scanner
和 Copier
分别定义了专门的接口,DocumentProcessor
只依赖于 Printer
接口。这样,DocumentProcessor
只依赖于它需要的接口,避免了不必要的依赖。
遵循 ISP 的几个要点
- 接口分离:将一个大接口拆分成多个小接口,每个小接口只包含一个特定的功能。
- 客户端专用接口:设计接口时要考虑到客户的实际需求,确保接口提供的功能符合客户的要求。
- 避免冗余方法:接口中不应包含未被客户端使用的方法,避免将多个不相关的功能集中到一个接口中。
接口隔离原则的好处
- 提高代码的可维护性:接口专一化使得类的职责更加明确,减少了不必要的依赖,提高了代码的可维护性。
- 增强代码的灵活性:客户端只依赖于所需的接口,使得代码在需求变化时更加灵活。
- 降低系统的耦合度:通过减少类对不必要接口的依赖,降低了系统的耦合度,使得系统更易于扩展和修改。
接口隔离原则有助于创建更加模块化和可维护的系统,通过减少类对不必要功能的依赖,提高了系统的灵活性和稳定性。
5. 依赖倒置原则(Dependency Inversion Principle,DIP)
依赖倒置原则(Dependency Inversion Principle, DIP)是 SOLID 原则中的“ D ”。它的核心思想是:高层模块不应依赖于低层模块,而应依赖于抽象;抽象不应依赖于细节,细节应依赖于抽象。换句话说,这个原则要求高层模块和低层模块都应依赖于抽象接口,而不是具体的实现类,从而减少系统的耦合度,提高灵活性和可维护性。
原则解释
- 高层模块依赖于抽象:系统中的高层模块(即业务逻辑模块)不应直接依赖于低层模块(即具体实现模块)。高层模块应该依赖于抽象接口,这样可以避免高层模块因为低层模块的变化而受到影响。
- 抽象不依赖于细节:抽象接口应该定义高层模块和低层模块之间的契约,而具体的实现细节应该在低层模块中实现。抽象接口应提供对低层模块功能的抽象,而不依赖于具体的实现细节。
- 细节依赖于抽象:具体实现应该依赖于抽象接口,而不是依赖于其他具体实现。这可以确保系统的具体实现可以随时被替换,而不会影响到依赖于这些接口的高层模块。
举例说明
假设我们在开发一个报告生成系统,有一个 ReportGenerator
类需要依赖一个 Printer
类来打印报告。最开始,我们可能设计了如下的类结构:
初始设计(违反依赖倒置原则)
class Printer:def print(self, document):print("Printing:", document)class ReportGenerator:def __init__(self, printer):self.printer = printerdef generate_report(self, data):document = f"Report: {data}"self.printer.print(document)
在这个设计中,ReportGenerator
直接依赖于 Printer
类的具体实现。这导致 ReportGenerator
和 Printer
之间的耦合度较高。如果将来我们需要更改 Printer
的实现,可能会对 ReportGenerator
造成影响,从而导致系统的灵活性和可维护性降低。
改进设计(符合依赖倒置原则)
为了符合 DIP,我们应该引入一个抽象接口,使 ReportGenerator
和 Printer
之间的依赖关系通过抽象接口来建立。
from abc import ABC, abstractmethodclass Printer(ABC):@abstractmethoddef print(self, document):passclass ConsolePrinter(Printer):def print(self, document):print("Printing to console:", document)class ReportGenerator:def __init__(self, printer: Printer):self.printer = printerdef generate_report(self, data):document = f"Report: {data}"self.printer.print(document)
在改进设计中,Printer
被定义为一个抽象接口(使用 ABC
和 abstractmethod
进行定义),ConsolePrinter
是 Printer
接口的一个具体实现。ReportGenerator
只依赖于 Printer
抽象接口,而不是 ConsolePrinter
的具体实现。这样,当我们需要更改打印的实现方式时,只需创建一个新的实现类,而不需要修改 ReportGenerator
类。
遵循 DIP 的几个要点
- 使用抽象接口:高层模块和低层模块之间的依赖关系应该通过抽象接口来建立,避免直接依赖具体实现。
- 分离接口与实现:将接口和实现分开,使得具体实现可以在不改变高层模块的情况下进行替换或修改。
- 依赖注入:将依赖关系的注入(例如,通过构造函数注入)从高层模块中提取出来,以实现更好的模块解耦。
依赖倒置原则的好处
- 增强系统的灵活性:通过依赖抽象接口而不是具体实现,使得系统可以更加灵活地进行扩展和修改。
- 减少系统的耦合度:降低高层模块对低层模块的依赖,提高了系统的可维护性和稳定性。
- 提高代码的可测试性:依赖于抽象接口使得系统组件更容易进行单元测试,因为可以通过模拟对象(mock objects)来替代具体的实现类进行测试。
依赖倒置原则有助于创建更加模块化和可扩展的系统,通过减少对具体实现的依赖,提高了系统的灵活性和可维护性。
这篇关于面向对象设计的五大原则:SOLID原则(聚合和耦合)_v0.1.0的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!