面向对象设计原则 SOLID

SRP 单一职责原则

<?php

<<<EOF
SRP 单一职责原则

任何一个软件模块都应该只对某一类行为者负责
EOF;

<<<EOF
下面举一个错误示范

Animal 既有吃肉, 又有吃奶酪. 在某些情况下肯定就会导致错误, 比如 jerry 不吃肉, 但是因为 jerry 是 Animal 的实例, 所以 jerry 也可以吃肉了, 这不符合逻辑。
另外一种问题是修改的时候, 比如 tom 吃奶酪的方式是一口一个, 然后 tom 的开发者把 Animal 的吃奶酪方法改为一口一个, jerry 本来是小口吃也会受到影响.
EOF;

class Animal{

    private string $name;
    private string $type;

    function __construct(string $name,string $type){
        $this->name = $name;
        $this->type = $type;
    }

    function eatMeat(){
        echo $this->type . $this->name . " eat meat";
    }

    function eatCheese(){
        echo $this->type . $this->name . " eat cheese";
    }
}


<<<EOF
解决办法有两种, 第一种是直接将动物的属性和动物的操作相分离, 像下面这样猫的操作和老鼠的操作作为两个类互不影响, 同时他们的 animalInfo 都是 AnimalInfo 的实例, 保持了 Animal 的共性
EOF;

class AnimalInfo{

    private string $name;
    private string $type;

    function __construct(string $name,string $type){
        $this->name = $name;
        $this->type = $type;
    }

    function getName():string
    {
        return $this->name;
    }
    function getType():string
    {
        return $this->type;
    }

}


class Cat{

    private AnimalInfo $animalInfo;

    function __construct(AnimalInfo $animalInfo){
        $this->animalInfo = $animalInfo;
    }

    function eatMeat(){
        echo $this->animalInfo->getType() . ":" . $this->animalInfo->getName() . " eat meat";
    }

    function eatCheese(){
        echo $this->animalInfo->getType() . ":" . $this->animalInfo->getName() . " eat cheese";
    }
}

class Mouse{

    private AnimalInfo $animalInfo;

    function __construct(AnimalInfo $animalInfo){
        $this->animalInfo = $animalInfo;
    }

    function eatCheese(){
        echo $this->animalInfo->getType() . ":" . $this->animalInfo->getName() . " eat cheese";
    }
}

<<<EOF
上面的方法可以解决问题, 但是从 Animal 的使用者角度看, 原来需要管理一个类, 现在需要管理三个类, 而且得根据不同的情况创建不同的类的实例, 真的很麻烦.
为了解决上面的问题, 下面还有一种更好的方法, 那就是 Facade 设计模式, 可以看到使用者只需要管理 AnimalFacade 就可以做到和上面相同的事
EOF;

class AnimalFacade{

    private $animalImpl;

    function __construct(string $name,string $type){
        if($type == 'cat'){
            $this->animalImpl = new Cat(new AnimalInfo($name, $type));
        }elseif($type == 'mouse'){
            $this->animalImpl = new Mouse(new AnimalInfo($name, $type));
        }
    }

    function eatMeat()
    {
        $this->animalImpl->eatMeat();
    }

    function eatCheese()
    {
        $this->animalImpl->eatCheese();
    }

}




<<<EOF
本来到这里就结束了, 但是可能有的人会想到为什么不直接用多态来解决这个问题, 个人理解多态和上面的两种解决方法并不冲突, 然后第一种解法中的状态和操作相分离的思路确实非常的炫酷
EOF;


function test_1(){
    $tom = new Animal('Tom', 'cat');
    $tom->eatMeat();
    $jerry = new Animal('Jerry', 'mouse');
    $jerry->eatCheese();
}



function test_2(){
    $tom = new AnimalInfo('Tom', 'cat');
    $jerry = new AnimalInfo('Jerry', 'mouse');
    $cat = new Cat($tom);
    $mouse = new Mouse($jerry);
    $cat->eatMeat();
    $mouse->eatCheese();
}


function test_3(){
    $tom = new AnimalFacade('Tom', 'cat');
    $tom->eatMeat();
    $jerry = new AnimalFacade('Jerry', 'mouse');
    $jerry->eatCheese();
}

test_1();
test_2();
test_3();

OCP 开闭原则

<?php

<<<EOF
OCP 开闭原则
设计良好的计算机软件应该易于扩展, 同时抗拒修改

什么意思呢? 就是说设计好的软件, 如果要扩展它的功能是很容易的, 同时扩展它的功能时不需要修改已有的功能模块。
EOF;


<<<EOF
同样先举一个不好的例子, 在写功能的时候先写 controller 然后慢慢的 controller 的行数太长了, 又把 controller 的代码往 services 抽离, 这样每次需要改 controller, 就要改之前抽离的 service, 这就不符合开闭原则。
EOF;


class UserController
{

    private $responseData;

    function update(array $form)
    {
// 模拟重构
//        $updateData = [];
//        if(isset($form['name'])){
//            $updateData["name"] = $form['name'];
//        }
//        if(isset($form['aga'])){
//            $updateData["aga"] = $form['aga'];
//        }
//        if(isset($form['id'])){
//            UserModel::where('id', $form['id'])->update($updateData);
//            $this->response_data = ["code"=> 200, "msg"=> "更新成功"];
//        }else{
//            $this->response_data = ["code"=> 501, "msg"=> "参数错误, 缺少 id"];
//        }
        $this->responseData = UserService::update($form);
        return $this->responseData;
    }
}

class UserService
{
    static function update($form)
    {
        // 后续这里必定还会再改
        $updateData = [];
        if (isset($form['name'])) {
            $updateData["name"] = $form['name'];
        }
        if (isset($form['aga'])) {
            $updateData["aga"] = $form['aga'];
        }
        if (isset($form['id'])) {
            UserModel::where('id', $form['id'])->update($updateData);
            return ["code" => 200, "msg" => "更新成功"];
        } else {
            return ["code" => 501, "msg" => "参数错误, 缺少 id"];
        }
    }
}


<<<EOF
正确的方法是先定义 接口和 service , 然后扩展 service
下面的 ProjectInterface 是一个接口, 定义了一个 update 方法, 代表了不论 ProjectService 的功能如何扩展, update 是必定存在的, 同时 ProjectService 又可以很方便的扩展, 这就是符合了开闭原则
EOF;

interface ProjectInterface
{
    function update($form);
}

class ProjectService implements ProjectInterface
{
    function update($form)
    {
    }
}

// 这里模拟重构
class ProjectServiceV1 extends ProjectService
{
    function update($form)
    {
        $updateData = [];
        // TODO 这里懒得写了
    }
}


class ProjectController
{
    // 这里直接通过接口名称注入实现
    function update(ProjectInterface $service, $form)
    {
        $this->response_data = $service->update($form);
        return $this->response_data;
    }
}


<<<EOF
开闭原则在软件架构中的应用, 主要体现在一个逻辑分层的概念, 比如 MVC,  当代码需要修改时, 第一反应就是要修改那一层的代码, 而不是直接重构项目
EOF;

LSP 里氏替换原则

<<<EOF
LSP 里氏替换原则
操作父类的程序在操作子类时行为可以保持不变

里氏替换原则可以看作开闭原则的补充, 开闭原则中提到好的架构是方便扩展的, 里氏替换原则对这种扩展进行了限制, 要求子类对父类进行扩展时需要保证行为一致, 不能随意扩展。

这里行为一致其实指的就是,返回值的类型相同或者更严格, 而不是说子类的方法必须和父类一模一样。 所以不用纠结子类的返回值不一样了,是不是不符合里氏替换原则了。 因为面向对象的多态,指的就是同一类型的不同对象, 行为可能不同。 里氏替换原则只是要求行为一致,不是完全相同。
EOF;


<<<EOF
同样举一个不好的例子
EOF;

####### 太晚了, 懒得写了。

<<<EOF
里氏替换原则不仅可以用在代码层面,还可以用在架构层面, 比如日志系统... 写着写着忘了, 睡觉了, 晚安玛卡巴卡
EOF;
本作品采用《CC 协议》,转载必须注明作者和本文链接
《L03 构架 API 服务器》
你将学到如 RESTFul 设计风格、PostMan 的使用、OAuth 流程,JWT 概念及使用 和 API 开发相关的进阶知识。
《L01 基础入门》
我们将带你从零开发一个项目并部署到线上,本课程教授 Web 开发中专业、实用的技能,如 Git 工作流、Laravel Mix 前端工作流等。
讨论数量: 0
(= ̄ω ̄=)··· 暂无内容!

讨论应以学习和精进为目的。请勿发布不友善或者负能量的内容,与人为善,比聪明更重要!