OOP 五大原则, 5 Principles of Object-oriented Programming

1. Single Responsibility Principle (SRP)

单一职责原则

定义

  • Every module/class/function should have single responsibility and only change for 1 reason .
  • 一个class/module 只能有一项职责,也只因为一个原因而发生改变。

优势

  • 提高reusability.
  • 提高 maintainability

示例

class SaleOrders:
    """
    A class keeps information about a sales order
    """
    # ✅, meets SRP
    def saveOrder(order: Order):
        pass

    # ❌, Violate for SRP. Because there will be different type clients to use this class 
    # and there will be different reasons to modify this class
    def exportOrderAsHtml() :
        pass

2. Open-Closed Principle (OCP)

开闭原则

定义

  • Each software entity should be open for extension, but closed for modification.
  • 一个实体(类、函数、模块等),可以不断衍生,但不经常修改。

优势

  • 提高extensibility
  • Easy to maintain and upgrade 易于维护和升级

❌ 错误示例

 """
 This design violates the OCP because if the rules change, 
 the OrderValidation class has to be modified, tested, and compiled.
 """
class OrderValidation:

    def _rule1(self, order: Order):
        pass

    def _rule2(self, order: Order):
        pass

    def _rule3(self, order: Order):
        pass

    def validate(self, order: Order):
        if order.amount > 1000:
            return self._rule1(order)

        if order.amount > 500:
            return self._rule2(order)

        return self._rule3(order)

✅ 正确示例

   """
   The new design will satisfy the OCP, 
   because if the rules change,  we can just create a new Validator class
   """
from abc import ABC, abstractmethod

class Validator(ABC):

    @abstractmethod
    def validate(self, order: Order):
        raise NotImplementedError


class Validator1(Validator):

    def validate(self, order: Order):
        pass


class Validator2(Validator):

    def validate(self, order: Order):
        pass


class Validator3(Validator):

    def validate(self, order: Order):
        pass

class OrderValidation:

    def __init__(self, validator: Validator):
        self._validator = validator

    def validate(self, order):
        return self._validator.validate(order)

3. Liskov Substitution Principle (LSP)

里氏代换原则

定义

  • Objects in a program should be replaceable with instances of their subclasses without altering the correctness of that program.
  • 子类可以替换父类,且不影响程序正确性。是开闭原则的补充。

优势

  • 提高reusability.
  • 提高 maintainability

❌ 错误示例

 """
 This design will violate LSP.
 If a client uses a reference variable of type Rectangle to call the setSize() method 
 to assign different values of height and width, then results will be different
 if the variable references to a Rectangle object than to a Square object.
 """
class Rectangle:

    def __init__(self):
        self._height = None
        self._width = None

    def set_size(self, height: int, width: int):
        self._height = height
        self._width = width

class Square(Rectangle):

    def set_size(self, height: int, width: int):
        self._height = height
        self._width = height

4. Interface Segregation Principle (ISP)

接口隔离原则

定义

  • Clients should not be forced to depend on interfaces that they do not use.
  • 解决不同的问题的时候,利用不同的接口。不要局限于单一接口。

优势

  • Reduce coupling, 降低耦合。
  • Reduce interdependency, 降低依赖。

❌ 错误示例

"""
This violates ISP because clients are forced to depend on
methods they do not use: HighWay does not use stat_engine(),
and ParkingLot does not need accelerate().
"""
class Vehicle:

    def stat_engine(self):
        pass

    def accelerate(self):
        pass

    def get_speed(self):
        pass

    def get_vehicle_type(self):
        pass

    def is_ticketed(self):
        pass

✅ 正确示例

"""
A better design is to design smaller interfaces for different types of clients
as shown in the following figure
"""
class DriverVehicle:

    def stat_engine(self):
        pass

    def accelerate(self):
        pass

class HighwayVehicle:

    def get_speed(self):
        pass

    def get_vehicle_type(self):
        pass

class ParkingLotVehicle:

    def is_ticketed(self):
        pass

5. Dependency Inversion Principle (DIP)

依赖倒转原则

定义

  • High level modules should not depend on low level modules, both should depend on abstraction.
    Also, abstraction should not depend on details, details should depend on abstractions.
  • 针对接口编程,依赖于抽象而不依赖于具体。

优势

  • Reduce coupling, 降低耦合。
  • Reduce interdependency, 降低依赖。

❌ 错误示例

"""
Making an EBookReader class to use PDFBook class is a violation of DIP because it requires to change 
the EBookReader class to read other types of e-books. 
"""
class PDFBook:

    def read(self):
        pass

class EBookReader:

    def __init__(self, pdfBook: PDFBook):
        self._pdfBook = pdfBook

✅ 正确示例

"""
A better design is to let EBookReader  use an interface EBook and let PDFBook and other types of
e-book classes implement EBook. Now adding or 
changing e-book classes will not require any change to EBookReader class.
"""
from abc import abstractmethod, ABC

class EBook(ABC):

    @abstractmethod
    def read(self):
        raise NotImplementedError

class PDFBook(EBook):

    def read(self):
        pass

class TXTBook(EBook):

    def read(self):
        pass

class EBookReader:

    def __init__(self, eBook: EBook):
        self._eBook = eBook

    def read(self):
        self._eBook.read()

参考资料

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 225,208评论 6 524
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 96,502评论 3 405
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 172,496评论 0 370
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 61,176评论 1 302
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 70,185评论 6 401
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 53,630评论 1 316
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 41,992评论 3 431
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 40,973评论 0 280
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 47,510评论 1 325
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 39,546评论 3 347
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 41,659评论 1 355
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 37,250评论 5 351
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 42,990评论 3 340
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 33,421评论 0 25
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 34,569评论 1 277
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 50,238评论 3 382
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 46,732评论 2 366

推荐阅读更多精彩内容