این اصل که توسط Barbara Liskov ابداع شده است حاکی از آن است که وقتی دست به نوشتن اینترفیسها میزنیم، نه تنها باید روی نوع ورودی کلاس ایمپلیمنتشده از روی یک اینترفیس دقت کنیم، بلکه باید خروجی کلیهٔ کلاسهایی هم که از یک اینترفیس استفاده میکنند یکسان باشد. در یک کلام، سابکلاس باید بتواند جایگزین سوپرکلاس شود بدون آنکه مشکلی ایجاد گردد. به عبارتی، Liskov Substitution Principle یا به اختصار LSP حاکی از آن است که هر گونه پیادهسازی Abstraction (انتزاع)، که یکی از اصول چهارگانهٔ OOP است، باید بدون آنکه باگی ایجاد شود در هر جایی استفاده شود.
برای روشنتر شدن کاربرد LSP، در ادامه مثال کلاسیک مستطیل و مربع را در قالب کد توضیح خواهیم داد به طوری ساختار این پروژه به صورت زیر خواهد بود:
liskov-substitution-principle/
├── index.php
├── Quadrilateral.php
├── Rectangle.php
└── Square.phpلازم به یادآوری است که برای شلوغ نشدن سورسکدهای مورد استفاده در این آموزش، از نوشتن کلیهٔ تگهای آغازین php؟> خودداری کردهایم. به منظور روشن شدن کاربرد LSP، ساختار پروژهٔ زیر را مد نظر قرار میدهیم:
ابتدا پوشهای به نام liskov-substitution-principle ساخته سپس در فایلی تحت عنوان Rectangle.php کلاسی با نام Rectangle نوشتهایم:
class Rectangle
{
protected $width;
protected $height;
public function setHeight($height)
{
$this->height = $height;
}
public function getHeight()
{
return $this->height;
}
public function setWidth($width)
{
$this->width = $width;
}
public function getWidth()
{
return $this->width;
}
public function area()
{
return $this->getHeight() * $this->getWidth();
}
}همانطور که میبینیم، داخل این کلاس دو پراپرتی تحت عناوین width$ و height$ تعریف کردهایم سپس با استفاده از متدهای اصطلاحاً Setter آنها را به ترتیب مقداردهی کرده و در فانکشن ()area نیز از متدهای به اصطلاح Getter استفاده نمودهایم تا مساحت مستطیل محاسبه گردد و در ادامه فایل دیگری ساختهایم با نام Square.php که قرار است داخل آن کلاسی به اسم Sqaure بنویسیم که از کلاس Rectangle ارثبری میکند:
require_once 'Rectangle.php';
class Square extends Rectangle
{
public function setHeight($value)
{
$this->width = $value;
$this->height = $value;
}
public function setWidth($value)
{
$this->width = $value;
$this->height = $value;
}
}با توجه به این که در هندسه مربع نوعی مستطیل تلقی میگردد، از همین روی ما هم به عنوان دولوپر فکر میکنیم که با ارثبری کردن از روی کلاس مرتبط با مستطیل میتوانیم به هدف خود دست یابیم که حداقل روی کاغذ چنین چیزی مشکلی ایجاد نمیکند اما در ادامهٔ آموزش خواهید دید که اینطور نیست!
حال برای تست هر دو کلاس والد و فرزند، فایلی میسازیم به اسم index.php و آن را به صورت زیر تکمیل میکنیم:
require_once 'Rectangle.php';
require_once 'Square.php';
$rectangle = new Rectangle();
$rectangle->setWidth(3);
echo $rectangle->area();
echo '<br>';
$square = new Square();
$square->setWidth(3);
echo $square->area();در خطوط ابتدایی هر دو فایل را ایمپورت کردهایم و در ادامه آبجکتی تحت عنوان rectangle$ از روی کلاس Rectangle ساخته و از طریق متدهای Setter این کلاس، width$ و height$ مستطیل را مقداردهی کردهایم و در نهایت هم با استفاده از فانکشن ()area مساحت را در خروجی چاپ کردهایم (با درج یک دستور <br> یک فاصله ایجاد کرده و همچون دستورات قبل، آبجکتی از روی کلاس Square ایجاد کردهایم.) با اجرای این فایل در لوکالهاست با خروجی زیر مواجه خواهیم شد:
6
9در تفسیر خروجی فوق میتوان گفت که مساحت مستطیل به درستی برابر با ۶ محاسبه شده است و در ارتباط با شیئ مربع نیز انتظار میرود که مساحت برابر با همان مقدار ۶ باشد اما در کمال تعجب میبینیم که مقدار ۹ به دست آمده است! برای درک بهتر این موضوع، مجدد کلاس Square را مد نظر قرار میدهیم:
require_once 'Rectangle.php';
class Square extends Rectangle
{
public function setHeight($value)
{
$this->width = $value;
$this->height = $value;
}
public function setWidth($value)
{
$this->width = $value;
$this->height = $value;
}
}برای محاسبهٔ مساحت مربع باید هر دو پراپرتی width$ و height$ را برابر با عدد واحدی قرار دهیم که در همین راستا داخل فانکشن ()setHeight مقدار هر دو پراپرتی را برابر با ۲ قرار دادهایم اما وقتی که فانکشن ()setWidth فراخوانی میشود، برای آنها مقدار ۳ را در نظر گرفته میشود و چون کدها از بالا به پایین اجرا میشوند، همین میشود که خروجی ۹ مشاهده خواهد شد. حال بار دیگر کدهای فوق را به صورت زیر تغییر میدهیم:
$square = new Square();
$square->setHeight(3);
$square->setWidth(2);
echo $square->area();در حقیقت فقط جای مقادیر ورودی برای width$ و height$ را تغییر دادهایم که در صورت اجرای اسکریپت فوق، با خروجی زیر مواجه خواهیم شد:
4برای رفع این مشکل، قانون LSP را به صورت زیر اِعمال میکنیم:
interface Quadrilateral
{
public function setHeight($h);
public function setWidth($w);
public function area();
}در فایلی تحت عنوان Quadrilateral.php یک اینترفیس ساختهایم تحت عنوان Quadrilateral به معنی «چهارضلعی» که در آن اصول کار تعریف شده است به طوری که هر دو کلاس مستطیل و مربع از آن ایمپلیمنت خواهند شد:
require_once 'Quadrilateral.php';
class Rectangle implements Quadrilateral
{
protected $width;
protected $height;
public function setHeight($height)
{
$this->height = $height;
}
public function getHeight()
{
return $this->height;
}
public function setWidth($width)
{
$this->width = $width;
}
public function getWidth()
{
return $this->width;
}
public function area()
{
return $this->getHeight() * $this->getWidth();
}
}کلاس مربع نیز به صورت زیر تغییر مییابد:
require_once 'Quadrilateral.php';
class Square implements Quadrilateral
{
protected $side;
public function setHeight($height)
{
$this->side = $height;
}
public function setWidth($width)
{
$this->side = $width;
}
public function getSide()
{
return $this->side;
}
public function area()
{
return pow($this->side, 2);
}
}حال مجدد فایل index.php را مد نظر قرار میدهیم:
require_once 'Rectangle.php';
require_once 'Square.php';
$rectangle = new Rectangle();
$rectangle->setHeight(2);
$rectangle->setWidth(3);
echo $rectangle->area();
echo '<br>';
$square = new Square();
$square->setHeight(2);
$square->setWidth(3);
echo $square->area();و به عنوان خروجی داریم:
6
9در مثالی دیگر که پایبند به این اصل است، یک Super Class به نام Bird داریم که متد walk را پیادهسازی میکند:
class Bird {
protected string $name;
public function __construct(string $name)
{
$this->name = $name;
}
public function walk() {
return "{$this->name} is walking.";
}
}همچنین یک کلاس به نام FlyingBird داریم که کلاس Bird را extend کرده و یک متد جدید به نام fly را نیز به آن اضافه میکند:
class FlyingBird extends Bird {
protected int $flySpeed;
public function __construct(string $name, int $flySpeed)
{
parent::__construct($name);
$this->flySpeed = $flySpeed;
}
public function fly() {
return "{$this->name} is flying at a speed of {$this->flySpeed}.";
}
}اکنون دو کلاس جدید تعریف میکنیم که هر کدام از کلاسهای بالا را extend کنند و بالتبع متدهای نوشته در آن را نیز در خود دارند:
class Dove extends FlyingBird {
protected string $color;
public function __construct(string $name, int $flySpeed, string $color)
{
parent::__construct($name, $flySpeed);
$this->color = $color;
}
}
class Penguin extends Bird {
protected string $color;
public function __construct(string $name, string $color)
{
parent::__construct($name);
$this->color = $color;
}
}بر اساس قانون Liskov، باید قادر باشیم تا شئای از سوپرکلاس Bird را با شئای از سابکلاسهای آن، که شامل کلاسهای FlyingBird، Penguin، و Dove میشود، جا به جا کنیم بدون آن که برنامه دچار مشکل شود.
برای مثال تصور کنید تابعی به شکل زیر در برنامه داریم که در پارامتر ورودی، شئای از نوع Bird را میپذیرد(که به این معناست که هر شئای که Bird را extend کند نیز میپذیرد):
function birdWalking(Bird $bird) {
return $bird->walk();
}اکنون باید بتوانیم بدون این که اشکالی در روند برنامه رخ دهد، یک شئ از سابکلاس Bird را به جای شئای از خود این سوپرکلاس، به این تابع بفرستیم.
به مثال زیر دقت نمایید:
function birdWalking(Bird $bird) {
return $bird->walk();
}
$bird = new Bird('Birdie');
$flyingBird = new FlyingBird("Hawk", 5000);
$dove = new Dove("Rock dove", 3000, "Pale grey");
$penguin = new Penguin("Emperor penguin", "black-white");
echo birdWalking($bird); // output: Birdie is walking.
echo birdWalking($flyingBird); // output: Hawk is walking.
echo birdWalking($dove); // output: Rock dove is walking.
echo birdWalking($penguin); // output: Emperor penguin is walkingدر فراخوانی تابع بالا با اشیاء مختلف، هیچ خطایی در برنامه ایجاد نشد و همهی آنها خروجی مرتبط را به درستی برگرداندند، که این نشان دهنده پایبندی به قانون Liskov است.
همین عمل را میتوانید با سوپرکلاس FlyingBird و سابکلاس آن یعنی Dove انجام دهید. برنامه شما باید بتواند یک شئ از کلاس Dove را جایگزین یکی شئ از کلاس FlyingBird کند.
جمعبندی
در واقع، مشکلی که گاهی در توسعهٔ نرمافزار مبتنی بر OOP پیش میآید این است که دولوپر نیاز دارد تا بسته به نوع خروجی شروط مختلفی در نظر بگیرد تا مثلاً اگر خروجی آرایه بود یک کار انجام شود و اگر آبجکت بود کاری دیگر و این در حالی است که اگر از قانون LSP پیروی کنیم، هندل کردن شروطی از این دست به سادگی امکانپذیر خواهد شد و این در حالی است که پیچیدگی سورسکد نیز به حداقل میرسد.
