首页  >  问答  >  正文

利用多态性同时返回多个结果和使用多种行为的方法

<p>我正在使用 Laravel 9 并且我有一个请求可以包含:</p> <ul> <li>名为 SEASON 的参数,值可以是 <code>array</code> 或 null 所以 <code>SEASON</code> 参数可以是 <code>array</code> 也可以是 <code>null</code></li> <li>名为 EXPIRY 的参数可以是 <code>array</code> 也可以是 <code>null</code></li> </ul> <p>我有两个类,一个用于 <code>SEASON</code> 功能,另一个用于 <code>EXPIRY</code> ,它们都从 <code>Repository</code> 扩展。两者都有一个名为 <code>execute</code> 的方法,该方法返回一个数组</p> <pre class="brush:php;toolbar:false;">abstract class Repository { abstract public function execute(): array; } class Expiry extends Repository { public function execute() { return ['The Request contain Expiry Parameter, and seasonal behaviours is done']; } } class Season extends Repository { public function execute() { return ['The Request contain Season Parameter, and expiry behaviours is done']; } }</pre> <p>如果我的请求包含SEASON,我想调用Season类的execute方法,或者如果我的请求包含Expiry,我想调用expiry的execute方法。或者调用它们并将执行的执行返回合并到一个数组中,以便我可以得到结果。</p> <pre class="brush:php;toolbar:false;">['The Request contain Expiry Parameter, and seasonal behaviours is done', 'The Request contain Expiry Parameter, and expiry behaviours is done']</pre> <p>这就是我在控制器中尝试过的:</p> <pre class="brush:php;toolbar:false;">public function bootstrap($data) { $parseTopics = Helper::parseTopicsRequest(); $basicProgram = new BasicProgramRepository(); $seasonalProgram = new SeasonalProgramRepository($parseTopics['SEASONAL']); $object = count($parseTopics['SEASONAL']) ? $seasonalProgram : $basicProgram; // Polymorphism return $object->execute(); }</pre> <p>问题1: 我不确定我是否应该使用这种方式或类似的方式来解决我的需求:</p> <pre class="brush:php;toolbar:false;">$employe = new Program(new BasicProgramRepository());</pre> <p>预期结果: 预期结果取决于我是否有季节参数和到期时间。我想要实现的是使用不同的行为(execute方法)</p>
P粉187677012P粉187677012380 天前525

全部回复(1)我来回复

  • P粉086993788

    P粉0869937882023-09-06 18:46:06

    如果你想实现多态方法,最好创建存储库或仅用于管理该逻辑的东西。

    这是示例。

    class SampleRepository
    {
        /**
         * repository instance value
         *
         * @var string[] | null
         */
        private $sampleArray; // maybe here is SEASON or EXPIRY or null
    
        /**
         * constructor
         *
         * @param string[] | null $sampleArray
         */
        public function __construct($sampleArray)
        {
            $this->sampleArray = $sampleArray;
        }
    
        /**
         * execute like class interface role
         *
         * @return array
         */
        public function execute()
        {
            return (!$this->sampleArray) ? [] : $this->getResult();
        }
    
        /**
         * get result
         *
         * @return array
         */
        private function getResult()
        {
            // maybe pattern will be better to manage another class or trait.
            $pattern = [
                "SEASON" => new Season(),
                "EXPIRY" => new Expiry()
            ];
            return collect($this->sampleArray)->map(function($itemKey){
                $requestClass = data_get($pattern,$itemKey);
                if (!$requestClass){ // here is space you don't expect class or canIt find correct class
                    return ["something wrong"];
                }
                return $requestClass->execute();
            })->flatten();
        }
    }

    你可以这样调用。

    $sampleRepository  = new SampleRepository($sampleValue); // expect string[] or null like ["SEASON"],["SEASON","EXPIRY"],null
        $result = $sampleRepository->execute(); // [string] or [string,string] or []

    此方法仅适用于您的参数指定值。 如果Season类和Expiry类的返回结果几乎相同,那么最好在trait上进行管理。 (即示例代码中的 $pattern)

    尝试一些。

    我读了评论,所以关注..

    例如,它更愿意只获取 getResult() 的结果。 因此,某些模式和如此多的逻辑不应该写在 getResult() 上;

    如果您使用特征,这是示例。 首先,您需要创建管理行为类。

    行为.php

    <?php 
    namespace App\Repositories;
    
    class Behavior
    {
        use Behavior\BehaviorTrait;
        // if you need to add another pattern, you can add trait here.
    }

    然后,您需要在同级位置创建Behavior目录。 你移动该目录,你就创建了这样的特征文件。

    <?php
    
    namespace App\Repositories\Behavior;
    
    trait BehaviorTrait
    {
        public static function findAccessibleClass(string $itemKey)
        {
          return data_get([
            "SEASON" => new Season(),
            "EXPIRY" => new Expiry()
          ],$itemKey);
        }
    }

    findAccessibleClass() 方法负责查找正确的类。

    然后,你可以像这样调用这个方法。

    private function getResult()
        {
            return collect($this->sampleArray)->map(function($itemKey){
                $requestClass = Behavior::findAccessibleClass($itemKey); // fix here.
                if (!$requestClass){ // here is space you don't expect class or canIt find correct class
                    return ["something wrong"];
                }
                return $requestClass->execute();
            })->flatten();
        }

    如果 getResult() 中的代码太多,最好将负责的代码分开。

    要创建Behavior Trait,getResult不需要负责行为逻辑。简而言之,它将很容易测试或修复。

    希望一切顺利。

    回复
    0
  • 取消回复