abstract class 人{ ... abstract function get_well(); } class 男 extends 人 { //无论你是神马男,服务你,你就能获得10点舒服度. private $well = 10; function get_well(){ return $this->well(); } } abstract class 装饰男类型 extends 人 { protected $人; function __construct(人 $人){ $this->人 = $人; } } class 捶背装饰 extends 类型男装饰{ function get_well(){ return $this->人->get_well()+30; } } class 洗发装饰 extends 类型男装饰{ function get_well(){ return $this->人->get_well()+20; } } class 洗褪装饰 extends 类型男装饰{ //老子不喜欢别人碰我的毛裤. function get_well(){ return $this->人->get_well()-20; } } //创建捶背,能给予的舒服指数 - -嘻嘻. $人 = new 捶背装饰(new 男); $人->get_well(); // 10+30 = 40 //来来来,全能选手,捶背、洗发、洗腿一起来 $人 = new 洗脚装饰(new 洗发装饰(new 捶背装饰(new 男()))); //10+30+20-20 = 40,注意顺序,由里到外执行.
装饰模式,既(组合+继承),基类方法一定要尽量少,不然子类可能有它不该有的方法.直接类继承,她只可能是一种形态,而她的多种形态可能一并拥有的时候,应该运用组合.
继承即单一多态,组合既多种多态.
这个例子中,你可以添加女,然后把装饰男类型改为装饰通用类型,但每个get_well()都要多一个判断是男还是女(如果给予的舒服程度不一样).
这只是确保不可能出现在男,女之外的第三种人,如果基类为动物,给予服务的可能是鸡,鹅,鸭,那么装饰类型应该运用工厂模式,动物形态和装饰形态一一对应.方便拓展.
除了服务类型,服务男的样子也很重要,这就多了一种装饰,现在有装饰男类型和相貌男类型,这种情况怎么破,其实类似.
复制代码 代码如下:
//如何获取捶背的帅哥麦?,
$人 =new 男类型(new 捶背(new 帅哥麦(new 男())));
4.3 外观模式
即给外部系统提供清晰接口
例如当Model层写得很混乱,但是里面的方法还能用,那我们的Controller层应该列举一些清晰的访问方法来供View层访问.外观模式,强调的是清晰的访问接口.
5 执行任务
5.1 策略模式
给类添加功能.对象要显式的调用它.
继续刚才的洗脚男和人的故事吧...你丫的爽完了要给钱吧?支付宝?微信?现金?
这个付款方式有多种,实现方法不应该放在人类中,而是应该委托给别的类
abstract class 人 { protectd $支付方式; function set_支付方式(){...} function 付款(金额){ return $支付方式->付款($金额); } } abstract class 付款{ abstract function 付款($金额); } class 支付宝付款 extends 付款{ function 付款($金额){ return 外接支付宝付款流程($金额); } } ... //使用 $男 =new 男(); ///爽爽爽 ... //结账 $支付宝支付账单 = new 支付宝付款($金额); $人 = new 男(); $人->set_支付方式(new 支付宝付款()); $人->付款();
5.2 观察者模式
当被观察者发生变化,观察者需要被通知.
当数据发生变化,页面需要被通知.
使用步骤:
观察者加载到被观察者中.
被观察者通知观察者.
例如登陆类(被观察)状态改变,要出发邮件系统和日志系统(观察者)
interface 被观察者{ function attach(观察者); function detatch(观察者); function notify(); } class Login implements 被观察者{ private $观察者; function __construct(){ $this->观察者 = array(); } function attach($观察者){ $this->观察者 = $观察者; } function detach($观察者){ //删除某个观察者的操作; } function notify(){ foreach ($this->观察者 as $单个观察者){ $单个观察者->update($this); } } } interface 观察者{ function update(被观察者); } abstract class Login_观察者 implements 观察者{ private $login; function __construct (Login $login){ $this->login = $login; $login->attach($this); } function update(观察者 $观察者){ if ($观察者 ===$this->login){ $this->do_update($观察者); } } abstract function do_update(Login $login); } class 邮件观察者 extends 登陆观察者 { function do_update(Login $login){ //判断条件 发送邮件 } } class 日志观察者 extends 登陆观察者 { function do_update(Login $login){ //判断条件 记录到日志; } } //使用 $login = new Login(); new 邮件观察者 ($login); new 日志观察者 ($login);
PHP有内置的SPL实现上述的观察者模式.
5.3 访问者模式