面向对象设计 SOLID 原则和python例子
admin
2024-05-13 18:15:43
0

面向对象设计 SOLID 原则共有 5 个,它们分别是:

1、依赖倒置原则 (Dependency Inversion Principle, DIP) :指的是高层模块 (high-level modules) 不应该依赖于低层模块 (low-level modules), 两者都应该依赖于抽象 (abstractions)。也就是说要以抽象为基础,而不是以实现为基础来进行编程

我个人的理解是,要先抽象出一个接口类,然后接口不实现具体的实现,但要定义输入的参数,起到让调用的人作为参考的作用。


# 一个例子:class LowLevelModule:def low_level_method(self):passclass HighLevelModule:def __init__(self):self.low_level_module = LowLevelModule()def high_level_method(self):self.low_level_module.low_level_method()# 在上面的代码中,HighLevelModule 类依赖于 LowLevelModule 类。为了遵循 DIP,我们应该将这种依赖关系反转过来:class LowLevelModule:def low_level_method(self):passclass HighLevelModule:def __init__(self, low_level_module):self.low_level_module = low_level_moduledef high_level_method(self):self.low_level_module.low_level_method()# 在这个例子中,HighLevelModule 类不再依赖于具体的实现类 LowLevelModule,而是依赖于抽象(接口)。

2、里氏替换原则 (Liskov Substitution Principle, LSP) :指的是如果一个程序中使用的是基类 (base class) 的对象,那么在不改变程序正确性的前提下,这个程序中基类对象的地方都可以使用其子类 (subclass) 对象来替换

我个人的理解是,要能够像调用父类一样调用子类,例如父类有一个方法是怎么调用的,子类也要有这个方法,并且是一样的调用这个方法


# 以下是一个例子: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(Rectangle):def __init__(self, side):self.width = sideself.height = sidedef calculate_area(shape: Shape) -> float:return shape.area()rectangle = Rectangle(2, 3)
print(calculate_area(rectangle))  # 6square = Square(2)
print(calculate_area(square))  # 4
# 在上面的代码中,我们定义了一个 Shape 类和两个子类 Rectangle 和 Square。
# 这两个子类都可以替换成基类 Shape,并且都可以正常工作。# 在函数 calculate_area 中,我们传入了一个 Shape 类型的参数,这个函数可以处理任何继承
# 自 Shape 的子类,而不需要考虑具体的子类是什么,这就是里氏替换原则的体现。# 里氏替换原则的要求是,子类需要满足的条件是,不破坏父类的约束条件,并且满足自己的约束条件。

3、开放封闭原则 (Open-Closed Principle, OCP) :指的是一个类或模块应该对扩展开放,对修改封闭。也就是说,当需要新增功能时,应该通过增加新的代码来实现,而不是修改现有的代码。

我个人的理解是:由于一个类可能会被多个地方都使用到,如果随便对这个类进行修改,那么可能很多引用这个类的地方都需要进行修改,这个工作量是很大的,而且还容易没有全部地方都进行修改,从而引发 BUG。因此可以选择对这个类进行继承,将新功能在继承的子类里进行实现。也可以写一个名字跟原来的接口的名字不同的新接口。个人感觉重新写一个新名字的接口的方法更简单,但是如果要增加的新接口很多的话,还是使用子类继承更好。


以下是一个例子: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 Circle(Shape):def __init__(self, radius):self.radius = radiusdef area(self):return 3.14 * self.radius ** 2class ShapeCalculator:def calculate_area(self, shape):return shape.area()calculator = ShapeCalculator()
rectangle = Rectangle(2, 3)
print(calculator.calculate_area(rectangle)) # 6
circle = Circle(2)
print(calculator.calculate_area(circle)) # 12.56# 在上面的代码中,我们定义了一个基类 Shape, 两个子类 Rectangle 和 Circle。
# 在类 ShapeCalculator 中,我们有一个函数 calculate_area,该函数可以处理任何继承
# 自 Shape 的子类,而不需要考虑具体的子类是什么,这样我们可以在不修改现有代码的情况
# 下新增新的图形类型,这就是开放封闭原则的体现。# 这种设计方式可以保证现有代码不会因新增功能而改变,从而降低维护代码的成本。

4、接口隔离原则 (Interface Segregation Principle, ISP) :指的是客户端不应该依赖它不需要的接口,即一个类对另一个类的依赖应该建立在最小的接口上。

我个人的理解是:假如一个接口类有很多的接口,但是实现这个接口类的具体实现类,其实有很多接口是使用不到的,因此如果要一一实现这些接口,既无法具体实现,也没有必要耗费大量的时间精力去实现这些接口,因此需要将接口类尽量分成多个接口类,每个接口类都比较精简,这样在实现的时候,就可以避免要去实现它不需要实现的接口。


# 以下是一个例子:class Document:def __init__(self, content):self.content = contentdef save(self):passdef print(self):passdef fax(self):passclass SimplePrinter:def print(self, document: Document):print(document.content)class Fax:def fax(self, document: Document):print(f"Faxing the document: {document.content}")# 在上面的代码中,我们定义了一个Document类,它包含三个方法:save,print,fax。
# 然而,并不是所有的客户端都需要使用这三个方法,比如 SimplePrinter只需要使用
#  print 方法,而不需要使用 save, fax 方法,这样就导致了客户端不必要的依赖。# 为了遵循 ISP,我们应该将这些接口拆分为更小的接口,比如:class PrintableDocument(ABC):@abstractmethoddef print(self):passclass SaveableDocument(ABC):@abstractmethoddef save(self):passclass FaxableDocument(ABC):@abstractmethoddef fax(self):passclass Document(PrintableDocument, SaveableDocument, FaxableDocument):def __init__(self, content):self.content = contentdef save(self):passdef print(self):print(self.content)def fax(self):passclass SimplePrinter:def print(self, document: PrintableDocument):print(document.content)class Fax:def fax(self, document: FaxableDocument):print(f"Faxing the document: {document.content}")
# 这样,我们就将 Document 类拆分成了三个更小的接口

5、单一职责原则:单一职责原则 (Single Responsibility Principle, SRP) :指的是一个类应该只有一个引起它变化的原因。也就是说,一个类只应该有一个负责其中的逻辑。

我个人的理解是:如果一个类需要实现多个职责,那么可能会导致这个类的功能不够明确,在实现时会增加很多不该在这个类中实现的功能。


# 以下是一个例子:class Employee:def __init__(self, name, salary):self.name = nameself.salary = salarydef increase_salary(self, percent):self.salary = self.salary + (self.salary * percent) / 100def save_employee(self):# save the employee to the databasepassdef send_email(self):# send email to the employeepass
# 在上面的代码中,类 Employee 负责三种职责:薪水计算、存储到数据库和发送电子邮件。# 为了遵循 SRP,我们应该将这些职责拆分成不同的类,比如:class Employee:def __init__(self, name, salary):self.name = nameself.salary = salaryclass SalaryCalculator:def increase_salary(self, employee: Employee, percent):employee.salary = employee.salary + (employee.salary * percent) / 100class EmployeeDB:def save_employee(self, employee: Employee):# save the employee to the databasepassclass EmailSender:def send_email(self, employee: Employee):# send email to the employeepass# 这样每个类只负责一个单一的职责,使得类更加简单,容易维护,这就是单一职责原则的体现。

这几个原则之间有没有互相矛盾的地方?

这几个原则之间并不会互相矛盾,而是相互补充和协同工作。

依赖倒置原则 (Dependency Inversion Principle, DIP) 强调的是高层模块不应该依赖低层模块,

而是应该依赖抽象。这个原则可以和单一职责原则 (Single Responsibility Principle, SRP) 协同工作,

因为如果低层模块有多个职责,那么就会增加高层模块的依赖。

里氏替换原则 (Liskov Substitution Principle, LSP) 强调的是子类应该能够替换父类。这个原则可以和

开放封闭原则 (Open-Closed Principle, OCP) 协同工作,因为如果父类是可扩展的,那么子类就可以扩展

父类的行为,而不会破坏程序的正确性。

接口隔离原则 (Interface Segregation Principle, ISP) 强调的是客户端不应该依赖它不需要的接口。

这个原则可以和依赖倒置原则 (Dependency Inversion Principle, DIP) 协同工作,因为如果低层模块的

接口不是客户端需要的,那么就会增加高层模块的依赖。

总之,这些原则是相互补充和协同工作的,它们的目的都是为了让代码更加简单,易于维护和扩展。遵循这些

原则可以使得代码具有高内聚性和低耦合性,这样可以降低程序的复杂性,提高程序的可维护性和可扩展性。

相关内容

热门资讯

4月份银行理财规模环比增加2.... 钛媒体App 5月16日消息,银行理财市场在4月份迎来规模与收益的双增长。据华源证券廖志明团队发布的...
【光明日报】黑龙江:免签红利释... 5月10日早上7时,一辆国际大巴缓缓停靠在黑龙江绥芬河公路口岸入境大厅前。游客们提着大包小裹,依次走...
又一跨国高端化工合作项目落子乐... 5月15日,福华化学携手瑞士特种化学品企业科莱恩打造的创新型高端磷系无卤阻燃剂项目(以下简称“福华科...
鸿蒙智行:已拥有1951家销售... IT之家 5 月 15 日消息,鸿蒙智行智界 V9 发布会正在进行,官方透露目前已拥有 1951 家...
黄金、白银,直线大跌! 5月15日晚间,贵金属价格突然大跌! 截至记者发稿时,现货黄金跌超2%,暂报4553美元/盎司附近。...
央视《焦点访谈》聚焦!万兴科技... 深圳商报·读创客户端首席记者 谢惠茜 5月14日,中央电视台《焦点访谈》推出专题节目《扩能提质强服务...
东方嘉富人寿董事长履职半年被换... 文|达摩财经 东方嘉富人寿再度进行人事调整。 5月13日,东方嘉富人寿发布公告称,自2026年4月...
重返西决!文班19+6卡斯尔3... 【搜狐体育战报】北京时间5月16日NBA季后赛,客场作战的马刺以139-109击败森林狼,总比分4-...
原创 美... 十万亿美债为什么还没有崩盘?或许答案在于,中国的存在让局势与众不同。现在的美债就像一张看似脆弱的网,...
原创 茅... 最近打开股票软件看白酒板块,是不是心里拔凉拔凉的? 茅台又回到1300元区间了,五粮液跌破90元,洋...
茅台宣布涨价 5月15日深夜,“i茅台”APP发布公告称,按照随行就市、供需适配、量价平衡、相对平稳的原则,贵州茅...
最高涨200元!茅台官宣4款产... 贵州茅台(600519.SH)凌晨宣布涨价几款产品。 茅台数字营销平台“i茅台”今日(5月16日)发...
面向地方国资产融转型全链条,X... 5月15日,XOD创新投融资模式3.0产品发布会在广州举办。该产品主要面向地方国资产融协同创新转型提...
2026Q1:10家上市商超9... 截至4月30日,所有A股上市公司2026年Q1财报全部出炉,传统商超也晒出自己的成绩单。10家披露的...
入主盟科药业失利后,拟借款2.... 来源:时代周报-时代在线 继去年试图通过定增入主盟科药业(688373.SH)失败后,海鲸药业再度出...
同比激增86%、规模突破760... 图片来源:界面图库 界面新闻记者 | 孙艺真 今年以来,证券行业融资补血热潮持续升温。前5个月...
促进青年消费,扶持青年创业,上... 5月14日,上海市政协团青界别、经济界跨界别活动在市政协全过程人民民主实践点举行。 今年初,团市委立...
苹果股价昨日创收盘新高,站上3... IT之家 5 月 16 日消息,苹果公司股价昨日(5 月 15 日)收于 300.23 美元,首次站...
杭州首批配售型保障房正式入市 杭州首批配售型保障房正式入市 价格约为周边商品房5折,18日开始报名 不能入市交易,可由政府指定机构...
“后巴菲特时代”,伯克希尔调仓... 当地时间5月15日,伯克希尔披露了2026年一季度美股持仓报告。这是伯克希尔在巴菲特卸任CEO并由阿...