SOLID - Принцип подстановки Барбары Лисков

Принципы проектирования классов SOLID в PHP

Принцип подстановки Барбары Лисков. (Liskov Substitution Principle, LSP)

Это третий из пяти принципов SOLID и наиболее сложный для понимания.
В оригинале данный принцип звучит так:

Пусть q(x) является свойством, верным относительно объектов x некоторого типа T.
Тогда q(y) также должно быть верным для объектов y типа S, где S является подтипом типа T.

Роберт Мартин перефразировал это так:

Функции, которые используют базовый тип, должны иметь возможность использовать подтипы базового типа, не зная об этом.

Иными словами -

Поведение наследуемых классов не должно противоречить поведению, заданному базовым классом, то есть поведение наследуемых классов должно быть ожидаемым для кода который использует базовый класс.

Приведем пример

index.php

$bird    = new Bird();
$birdRun = new BirdRun($bird);
$birdRun->run();

bird.php

/**
* Реально используемый в коде класс
*/
class Bird {
    public function fly() {
        $flySpeed = 10;
        return $flySpeed;
    }
}

/**
* Дочерний класс от Bird. 
* Не изменяет поведение, но дополняет.
* Не нарушает принцип LSP
*/
class Duck extends Bird {

    public function fly() {
        $flySpeed = 8;
        return $flySpeed;
    }
    
    public function swim() {
        $swimSpeed = 2;
        return $swimSpeed;
    }
}

/**
* Дочерний класс от Bird. 
* Изменяет поведение.
* Нарушает принцип LSP
*/
class Penguin extends Bird {

    public function fly() {
          //die('i can`t fly (((');  // не типичное поведение - die или exception
          return 'i can`t fly ((('; // не типичное поведение - возвращаем string, а не integer
    }

    public function swim() {
        $swimSpeed = 4;
        return $swimSpeed;
    }
}

birdRun.php

class BirdRun {

    private $bird; 
    public function __construct(Bird $bird) {
        $this->bird = $bird;
    }

    public function run() {
        $flySpeed = $this->bird->fly();
    }
}

index.php

$bird = new Bird();
//$bird = new Duck();
//$bird = new Penguin();
$birdRun = new BirdRun($bird);
$birdRun->run();

После замены использования Bird на Duck код будет работать как и прежде - принцип LSP соблюден.
После замены Bird на Penguin код меняет свое поведение, следовательно в данном случае принцип LSP нарушен.

Следовать этому типу очень важно при проектировании новых типов с использованием наследования.
Этот принцип предупреждает о том, что изменение унаследованного производным типом поведения очень рискованно.

Добавить отзыв (пожелание, комментарий)