php设计模式之工厂模式(工厂模式实现计算器功能)

因为工作的需要,陆陆续续接触了很不多不同的框架,也慢慢渗透了一些设计模式。
当初选择入手php,就是因为脚本语言,简单易学。正因为是脚本语言很多人拿着php做着面对过程的事情。在工作中我维护过一个function写了1000行的代码,还要一行一行下来开始寻找bug。当时我就想设计模式所带来的好处。

什么是工厂

简单的抽象成生活的一个例子就是,你需要一双鞋子,但是你不用关心鞋子的构造,采用什么布料,怎么制作。这时候你就需要告诉制造鞋子工厂说 我需要一双运动鞋。
你 > client 发送请求
鞋子工厂 > factory 接受请求 再返回一双鞋给用户

案例

我们现在要用php实现一个计算器的功能,具有加减乘除的功能,很多刚学php或者刚毕业的应届生一看到这就乐了,这不是很简单吗,一个php脚本几行代码就能搞定的东西。

//主程序代码
$parmasFirst = 1;
$parmasSecond = 2;
$operator = '/';
$result = 0;
switch($operator){    
case '+': $result = $parmasFirst + $parmasSecond ;break;   
case '-': $result = $parmasFirst - $parmasSecond ;break;   
case '*': $result = $parmasFirst * $parmasSecond ;break;    
case '/': {
        if (empty($parmasSecond)) { 
           throw new Exception('error parms ');        
        }       
 $result = $parmasFirst / $parmasSecond;       
 break;   
 }    
default:throw new Exception('error operator');
}
echo $result;

我们在设想一下,随着计算器的不断升级,不断有开平方、立方根……的复杂算法,你该怎么去维护代码?

你们会想 不就是在switch继续加分支。我们假设如果业务不断的扩展一个,功能被扩充了几千行代码的switch,你该怎么去维护?
第一种方法就是对switch里面的代码进行封装

封装后的代码

switch($operator)
    {
        case '+':
        {
            $operatorAddObj = new OperatorAdd($parmasFirst,$parmasSecond);
            $result = $operatorAddObj->getResult();
            break;
        }
        case '-':
        {
            $operatorAddObj = new OperatorSub($parmasFirst,$parmasSecond);
            $result = $operatorAddObj->getResult();
            break;
        }
        case '*':
        {
            $operatorAddObj = new OperatorMul($parmasFirst,$parmasSecond);
            $result = $operatorAddObj->getResult();
            break;
        }
        case '/':
        {
            $operatorAddObj = new OperatorDiv($parmasFirst,$parmasSecond);
            $result = $operatorAddObj->getResult();
            break;
        }
        default:throw new Exception('error operator');
    }

abstract class Operator
{
    protected $a,$b;
    public function __construct($parmasFirst,$parmasSecond)
    {
        $this->a = $parmasFirst;
        $this->b = $parmasSecond;
    }

    abstract function getResult();
}
class OperatorAdd extends Operator
{
    public function getResult()
    {
        return $this->a + $this->b;
    }
}
class OperatorSub extends Operator
{
    public function getResult()
    {
        return $this->a - $this->b;
    }
}
class OperatorMul extends Operator
{
    public function getResult()
    {
        return $this->a * $this->b;
    }
}
class OperatorDiv extends Operator
{
    public function getResult()
    {
        if(empty($this->b)) {
            throw new Exception('error parms b=0');
        }
        return $this->a / $this->b;
    }
}

但是你还是没有脱离switch这个架构。现在很多php框架都倡导模块化的设计理念,我们需要对不同的运算算法封装成独立一个模块,而不是通过switch不断去扩充,我们添加功能也不会去修改 主程序代码,以便带来不必要的问题bug.

通过工厂模式实现计算器

我们稍微把主程序的代码的swith进行修改

    //主程序代码
    $parmasFirst = 1;
    $parmasSecond = 1;
    $operator = '/';
    $result = 0;
    $result = OperatorFactory::create($parmasFirst,$parmasSecond,$operator)->getResult();
    echo $result;

class OperatorFactory
{
    public static $operator = array(
        '*' => 'mul',
        '/' => 'div',
        '-' => 'sub',
        '+' => 'add'
    );
    public static function create($a,$b,$operator)
    {
        if(!array_key_exists(strtolower($operator),self::$operator)) {
            throw new Exception('no operator');
        }
        $operatorName = self::$operator[$operator];
        $operatorObj = 'Operator' . ucfirst($operatorName);
        if(!class_exists($operatorObj)) {
            throw new Exception('no class');
        }
        return new $operatorObj($a,$b);
    }
}

这样有个好处就是 你去修改 其中一个算法的时候,不会去破坏其他算法的内部结构,而且你新增一个算法也很方便,也不用去更改主程序代码,只要稍微修改工厂模式就可以。便于代码维护

之后不断更新不同的设计模式来实现不同的案例和应用场景。 我已经给自己定好了目标了! 做一名phper,我要不断学习,并且在其中找到快乐,我才会在这条坑上越陷越深!

你可能感兴趣的:(php设计模式之工厂模式(工厂模式实现计算器功能))