策略模式和工厂模式在促销系统下的应用


设计模式为我们提供了在某些应用场景下特定问题的解决方案,今天我们利用策略模式和工厂模式实现一个简单的促销系统。
实现语言:PHP
我们先来回顾一下策略模式和工厂模式的基本知识

策略模式

Gof 对策略模式的作用的说明如下

定义一系列的算法,把他们一个个封装起来,并且使他们可以相互替换。本模式使得算法可独立于他的客户而变化。

策略模式对算法进行抽象,使得类可以方便地实现不同的算法,而调用他的客户端完全感觉不出来。

工厂模式

Gof 整理的工厂模式主要有抽象工厂模式和工厂方法模式,大家还经常提到一种简单工厂模式,工厂模式将类的创建进行封装,使得类的创建和使用可以相互分离,独立变化。

问题系统描述

要实现一个促销系统,实现对单件产品和购物车产品集合的促销。针对单件产品,促销策略有降价、打折、赠品,限制条件有无条件限制、商品满足一定价格;针对购物车产品集合,促销策略有打折、降价、赠品、低价免单、免邮,限制条件有无条件限制、商品集合总价满一定价格、商品集合满一定数量。

基本数据表及类设计

促销系统核心数据表设计

  1. promotion表,存储信息包括促销类别(针对单件产品的促销活动还是针对产品集合的促销活动),优惠策略,限制条件。

促销系统核心类设计

  1. Promotion 类,对应promotion数据表,实现某个限制条件下针对单件产品或产品集合的某个促销策略。
  2. Benefit 系列类,提供在产品和产品集合上不同的促销优惠的算法。
  3. Checkout 系列类,提供在产品和产品集合上不同的限制条件的算法。

除了以上核心类类外,还有以下基础类,比如Product(产品类,实现一件产品的具体功能)、Collection(集合类,某些类对象的集合,可以继承Collection实现比如产品集合等具体集合类)。

促销系统设计思路

通过以上分析,我们知道实现促销的对象有两个:单件产品和产品集合;主要功能也有两个:判断是否满足限制条件和实现某种促销优惠。限制条件和促销优惠这两个功能具有稳定性,具体实现算法具有可变性。这样我们可以针对限制条件和促销优惠抽象出来其接口,使用策略模式进行封装,使得判断条件和促销优惠的算法可以独立于客户端进行变化和增减,并通过工厂模式创建具体算法实例。我们还应该设计针对 Promotion 类接口,抽象出来提供给客户端调用算法的方法。

促销系统实现

接口设计

interface Promotion
{   /** Promotion 类的接口,标注 Promotion 类向系统提供的具体功能*///返回促销策略的类别信息public function getBenefit();//返回促销策略的值public function getBenefitValue();//返回限制条件的信息public function getLimit();//返回限制条件的值public function getLimitValue();//创建此次促销产生的优惠信息的对象public function makeBenefit($attribute);//判断某个产品集合是否满足促狭的限制条件 public function itemsCheckout(Collection $items, $limitValue);//判断某个产品是否满足促销的限制条件public function productCheckout(Product $product, $limitValue);//针对某个产品集合计算促销策略public function itemsPromotionCalculate(Collection $items);//针对某个产品计算促销策略public function productPromotionCalculate(Product $product);
}interface PromotionBenefit{/** 促销优惠策略的接口,提供针对产品集合和单件产品的促销优惠的方法 */// 计算产品集合的促销优惠public function itemsPromotionCalculate(Promotion $promotion, Collection $items);// 计算单件产品的促销优惠public function productPromotionCalculate(Promotion $promotion, Product $product);
}interface PromotionCheckout{/** 促销限制条件的接口,提供针对产品集合和单件产品的判断限制条件方法*///判断产品集合的限制条件public function itemsCheckout(Collection $items, $limitValue);//判断单件产品的限制条件public function productCheckout(Product $product, $limitValue);
}

限制条件策略的实现

根据上面的限制条件策略的接口,接下来我们来实现无限制条件、满一定价格、满一定数目这些限制条件的策略

/*
* 实现无条件限制策略,因为无条件限制,所以接口的两个方法均直接返回true
*/
class CheckoutNone implements PromotionCheckout{public function itemsCheckout(Collection $items, $limitValue){return true;}public function productCheckout(Product $product, $limitValue){return true;}
}/*
* 实现对单件商品和商品集合满一定价格的限制条件策略
*/
class CheckoutFullPrice implements PromotionCheckout{public function itemsCheckout(Collection $items, $limitValue){return $items->reduce(function($amount, $item){return $amount + $item->price * $item->qty;}, 0) >= $limitValue;}public function productCheckout(Product $product, $limitValue){return $product->getBasePrice() >= $limitValue;}
}/*
* 实现对商品集合满一定数目的限制条件策略
*/
class CheckoutFullNumber implements PromotionCheckout{public function itemsCheckout(Collection $items, $limitValue){return $items->sum('qty') >= $limitValue;}public function productCheckout(Product $product, $limitValue){//因为单件商品没有满足一定价格的限制条件,所以直接返回falsereturn false;}
}

限制条件策略工厂的实现

我们把限制条件策略类的实例化工作教给工厂类来实现,我们来看一下限制条件策略工厂类的实现

class PromotionCheckoutFactory{private static $_instances;/*** @return PromotionCheckout*/public static function makePromotionCheckout($limit){if(!isset(self::$_instances[$limit])){switch($limit){case Promotion::LIMIT_NONE : self::$_instances[$limit] = new CheckoutNone(); break;case Promotion::LIMIT_FULL_PRICE : self::$_instances[$limit] = new CheckoutFullPrice();break;case Promotion::LIMIT_FULL_NUMBER : self::$_instances[$limit] = new CheckoutFullNumber(); break;}}return isset(self::$_instances[$limit]) ? self::$_instances[$limit] : null;}
}

促销优惠策略的实现

实现了促销优惠策略的接口,接下来我们就来实现降价、打折、低价免单、赠品、免邮等具体促销优惠策略。

class BenefitPriceReduction implements PromotionBenefit{public function itemsPromotionCalculate(Promotion $promotion, Collection $items){/** 实现产品集合的降价促销策略*/ if($promoiton->getBenefit() != Promotion::BENEFIT_PRICE_REDUCTION){return false;}$size = max(count($promoiton->getBenefitValue()), count($promoiton->getLimitValue()));$benefitValues = array_pad($promoiton->getBenefitValue(), $size, 0);$limitValues = array_pad($promoiton->getLimitValue(), $size, false);$benefits = array_combine($limitValues, $benefitValues);/** 促销支持组合设定,比如这里可以表示满100减10,满200减25,满300减50,则$benefit的值为 *[*    100 => 10,*    200 => 25,*    300 => 50*]*/$attribute = false;foreach($benefits as $limitValue => $benefitValue){$benefitValue = floatval($benefitValue);if($promoiton->itemsCheckout($items, $limitValue) && $benefitValue > 0){$goodsAmount = $items->reduce(function($amount, $item){return $amount + $item->price * $item->qty;}, 0);$data = ['base_price' => $goodsAmount,'old_price' => $goodsAmount,'new_price' => round($goodsAmount - $benefitValue, 2),'benefit_price' => $benefitValue,];if(!$attribute || $attribute['benefit_price'] < $data['benefit_price']){$attribute = $data;$attribute['select_limit_value'] = $limitValue;$attribute['select_benefit_value'] = $benefitValue;}}}if($attribute){return $promoiton->makeBenefit($attribute);}return false;}public function productPromotionCalculate(Promotion $promoiton, Product $product){/** 实现单件产品的降价促销策略*/ if($promoiton->getBenefit() != Promotion::BENEFIT_PRICE_REDUCTION){return false;}$size = max(count($promoiton->getBenefitValue()), count($promoiton->getLimitValue()));$benefitValues = array_pad($promoiton->getBenefitValue(), $size, 0);$limitValues = array_pad($promoiton->getLimitValue(), $size, false);$benefits = array_combine($limitValues, $benefitValues);$attribute = false;foreach($benefits as $limitValue => $benefitValue){$benefitValue = floatval($benefitValue);if($promoiton->productCheckout($product, $limitValue) && $benefitValue > 0){$data = ['base_price' => $product->getBasePrice(),'old_price' => $product->getBasePrice(),'new_price' => round( $product->getBasePrice() - $benefitValue, 2),'benefit_price' =>$benefitValue,];if(!$attribute || $attribute['benefit_price'] < $data['benefit_price']){$attribute = $data;$attribute['select_limit_value'] = $limitValue;$attribute['select_benefit_value'] = $benefitValue;}}}if($attribute){return $promoiton->makeBenefit($attribute);}return false;}
}class PromotionDiscount implements PromotionBenefit{public function itemsPromotionCalculate(Promotion $promotion, Collection $items){/** 实现针对组合产品的打折优惠*/...}public function productPromotionCalculate(Promotion $promotion, Product $product){/** 实现针对单件产品的打折优惠*/...   }
}class BenefitLowerFree implements PromotionBenefit{public function itemsPromotionCalculate(Promotion $promotion, Collection $items){/** 实现针对产品集合的低价免单*/...}public function productPromotionCalculate(Promotion $promotion, Product $product){/** 单件产品不支持低价免单策略*/return false;}
}class BenefitFreeShipping implements PromotionBenefit{public function itemsPromotionCalculate(Promoiton $promotion, Collection $items){/** 判断产品集合是否满足免邮限制条件*/...}public function productPromotionCalculate(Promotion $promotion, Product $product){/** 单件产品不支持免邮策略*/return false;}
}class BenefitGift implements PromotionBenefit{/** @return false | $gift* false 此产品集合不满足赠品条件* $gift 此产品集合此次促销的赠品*/public function itemsPromotionCalculate(Promotion $promotion, Collection $items){/** 实现产品集合的赠品优惠*/...}/** @return false | $gift* false 此产品不满足赠品条件* $gift 此产品集合此次促销的赠品*/public function productPromotionCalculate(Promotion $promotion, Product $product){/** 实现单件产品的赠品优惠*/...}
}

优惠策略工厂的实现

我们实现了具体的促销优惠策略,接下来实现优惠策略的工厂

class PromotionBenefitFactory{private static $_instances;/*** @return PromotionBenefit*/public static function makePromotionStrategy($benefit){if(!isset(self::$_instances[$benefit])){switch($benefit){case Promotion::BENEFIT_DISCOUNT : self::$_instances[$benefit] = new BenefitDiscount(); break;case Promotion::BENEFIT_PRICE_REDUCTION : self::$_instances[$benefit] = new BenefitPriceReduction(); break;case Promotion::BENEFIT_LOWER_FREE : self::$_instances[$benefit] = new BenefitLowerFree(); break;case Promotion::BENEFIT_FREE_SHIPPING : self::$_instances[$benefit] = new BenefitFreeShipping(); break;case Promotion::BENEFIT_GIFT : self::$_instances[$benefit] = new BenefitGift(); break;}}return isset(self::$_instances[$benefit]) ? self::$_instances[$benefit] : null;}
}

Promotion 类的实现

Promotion 类应该是对应 promotion 表的 ORM, 除此之外,Promotion 类还应该实现 Promotion 接口,实现促销系统所需的方法

/*
* Note:PromotionInterface 指的是 Promotion 接口,因为 PHP 不允许类和接口名字一样,所以需要将 Promotion 接口 use 的时候 as 成 PromotionInterface
*/
class Promotion extends Model implements PromotionInterface
{protected $table = 'promotions';protected $primaryKey = 'pmid';/*** @var PromotionBenefit*/private $promotionStrategy;/*** @var PromotionCheckout*/private $promotionCheckout;const TYPE_PRODUCT = 'product';const TYPE_CART = 'cart';const LIMIT_NONE = 'no_limit';const LIMIT_FULL_PRICE = 'full_price';const LIMIT_FULL_NUMBER = 'full_number';const BENEFIT_DISCOUNT = 'discount';const BENEFIT_PRICE_REDUCTION = 'price_reduction';const BENEFIT_LOWER_FREE = 'lower_free';const BENEFIT_FREE_SHIPPING = 'free_shipping';const BENEFIT_GIFT = 'gift';public function __construct(){parent::__contruct();$this->promotionStrategy = PromotionStrategyFactory::makePromotionStrategy($this->getBenefit());$this->promotionCheckout = PromotionCheckoutFactory::makePromotionCheckout($this->getLimit());}public function getBenefit(){return $this->benefit;}/*** @return array*/public function getBenefitValue(){return $this->benefit_value;}public function getLimit(){return $this->limit;}/*** @return array*/public function getLimitValue(){return $this->limit_value;}public function makeBenefit($attribute){return new Benefit($attribute, $this);}public function itemsPromotionCalculate(Collection $items){if($this->promotionStrategy && $this->type == self::TYPE_CART){return $this->promotionStrategy->itemsPromotionCalculate($this, $items);}}public function productPromotionCalculate(Product $product){if($this->promotionStrategy && $this->type == self::TYPE_PRODUCT){return $this->promotionStrategy->productPromotionCalculate($this, $product);}}public function itemsCheckout(Collection $items, $limitValue){if($this->promotionCheckout){return $this->promotionCheckout->itemsCheckout($items, $limitValue);}}public function productCheckout(Product $product, $limitValue){if($this->promotionCheckout){return $this->promotionCheckout->productCheckout($product, $limitValue);}}
}

是否应该使用访问者模式

至此,我们主要利用了策略模式的工厂模式实现了促销系统的主要功能。在实现的过程中我考虑过需不需要使用访问者模式来对代码进行进一步抽象,因为访问者模式的作用是

表示一个作用于某对象结构中的各元素的操作。它使你可以在不改变的类的前提下定义作用于这些元素的新操作。

因为我们发现的主要作用对象是单件商品和商品集合,然后我们定义了这两个作用对象上的操作:判断限制条件和进行促销优惠。似乎是可以使用访问者模式进行抽象,将判断限制条件和进行促销优惠的的接口合并成对操作对象的接口,通过在此接口上的扩展实现对判断限制条件和进行促销优惠的扩展。这样做貌似也可以行得通,不过在实例中,将限制条件和促销优惠进行更加具体的抽象,代码更加容易理解,也更加易读,在操作对象上面操作行为的类别相对固定的情况下,使用访问者模式有点过度抽象。

策略模式和工厂模式在促销系统下的应用相关推荐

  1. 策略模式与工厂模式比较

    这段时间看了一些设计模式,看到策略模式与工厂模式的时候,总是感觉他们很相似,不要区分,到具体的场景了你可能还在徘徊到底用工厂还是策略呢?这几天就想写一篇关于策略模式与工厂模式区别的文章,但一直没思路, ...

  2. 设计模式-策略模式,观察者模式,装饰者模式,静态工厂模式,工厂方法模式

    设计模式这个东西,永远不是单单从书本上就能获取到的东西.曾经看到一个比喻,比喻的就很巧妙,文艺复兴时期的教学方式,那时候诞生了很多巨匠,达芬奇,莫开朗基罗,拉斐尔都是在这个时期绽放光芒的巨星.有一种说 ...

  3. 设计模式-策略模式和工厂模式结合使用

    怎么把策略模式和工厂模式结合起来使用 如果大家对策略模式和工厂模式不是很了解的话可以先看前面文章 策略模式:https://www.jianshu.com/p/958281936901 工厂模式:ht ...

  4. java策略模式 工厂模式_策略模式和工厂模式搭配使用

    策略模式和工厂模式的搭配使用可以很好地消除代码if-else的多层嵌套 需求 针对店下商铺,有这样一个需求,对用户客户分为了普通客户.vip客户.超级vip用户.专属vip用户4个等级,每当用户购买商 ...

  5. 策略模式、工厂模式、装饰者模式总结解析

    今天在面试的时候被问到自己策略模式怎么用的时候有被问懵到,以至于明明是自己的代码在脑海里已经混乱了,而且面试官提出的还是没有更好的利用设计模式也让我思考了一下我之前的代码到底是怎么实现的,重新梳理下策 ...

  6. 策略模式和工厂模式的区别

    文章目录 策略模式和工厂模式的区别 相似点 差异 用途不一样 关注点不一样 UML图 实例 策略模式和工厂模式的区别 相似点 在模式结构上,两者很相似: 差异 用途不一样 工厂是创建型模式,它的作用就 ...

  7. java策略模式和工厂模式的区别

    这两天初学设计模式的时候发现工厂模式和策略模式写法上相似,感觉很疑惑.既然相似,为何还要专门写成两种不同的模式呢.在翻阅了各个高手的总结后,偶有一点启发,特此记下. 有一个高手说 工厂相当于黑盒子,策 ...

  8. 策略与简单工厂模式结合的实现--收银软件的代码及UML图

    策略模式和简单工厂模式的结合:把分支判断放到环境角色中. 解决简单工厂模式中提到的问题: ●关键:分支的switch依然去不掉 ●解决方法:依赖注入.反射.XML 简单工厂模式 策略模式 收银软件的策 ...

  9. java 策略模式和工厂模式区别_Java编程细节——什么是策略模式

    策略模式应该是Java设计模式中最简单的一种模式, 它的核心思想是,一个类的行为可以在运行时动态改变,有不同的实现逻辑. 其实具体的说,它也是基于面向接口编程的思想,通过定义不同的实现类逻辑来做到的. ...

最新文章

  1. Java基础—IO流
  2. 018_rate评分
  3. 用python计算准确率_Python中计算模型精度的几种方法,Pytorch,中求,准确率
  4. MySQL——在Linux下安装和卸载MySQL
  5. python使用近似公式计算e_python如何算自然底数e(方法二)
  6. python如何画图设置坐标轴_python matplotlib坐标轴设置的方法
  7. 队列 开源 php,消息队列 - 基于think-queue消息队列 – 基于ThinkPHP和Bootstrap的极速后台开发框架...
  8. 两百多的无线蓝牙耳机和一千多的AirPods,外观几乎一样,硬件差距在哪里?
  9. c php curl post,php curl post
  10. 海思3516D + IMX291图像闪烁问题定位
  11. it男如何像黑客一样聊天qq
  12. 图片转Excel表格在线工具,分享几款不错的工具!
  13. PMP考试敏捷知识点(9)
  14. 阻滞增长模型求解_阻滞增长模型研究解读.ppt
  15. EditText输入完成后自动关闭输入法
  16. macOS安装homebrew与更新gcc
  17. 一年级前一学期计算机应用题,计算机教学工作计划7篇
  18. 对象上下文语义分割:OCR论文笔记(Object-Contextual Representations for Semantic Segmentation )
  19. 用python可以免费下载音乐吗-使用python实现下载我们想听的歌曲,速度超快
  20. 程序员职业资格软考——软考,你不想软就得考 (值得一看的总结)

热门文章

  1. 适合学生党的无线充电宝有哪些?学生党最爱的无线充电宝推荐
  2. iOS开发中如何添加应用自己的字体
  3. pdf压缩工具在线,pdf在线压缩软件,在线压缩pdf文件大小的方法?
  4. 华为鸿蒙高清图片,华为P50 Pro真高清图曝光:“鸿蒙之光”超大主摄稳了
  5. 网络安全的昨天、今天、明天
  6. linux洋葱浏览器,Firefox 39正式发布:如丝般顺滑
  7. 华为m6能更新鸿蒙,华为M6怎么升级鸿蒙系统 M6平板升级鸿蒙系统教程
  8. 会计师事务所的相关介绍以及业务的具体说明
  9. Anaconda下载安装及简单使用
  10. 【Linux SPI Framework】