PHP、Python和Javascript的装饰器模式对比


Posted in PHP onFebruary 03, 2015

修饰模式(Decorator Pattern),又叫装饰者模式,是面向对象编程领域中,一种动态地往一个类中添加新的行为的设计模式。就功能而言,修饰模式相比生成子类更为灵活,这样可以给某个对象而不是整个类添加一些功能。装饰模式非常适用于灵活扩展对象的功能,下面是装饰模式的UML图:

PHP、Python和Javascript的装饰器模式对比

例如,有一个技术论坛,用户通过留言进行沟通,由于刚开始论坛里都是熟人,几乎都不需要对留言的内容作出审核,接收留言的页面可以是这样:

class SaveMsg(){
 private $msg;
 public function __construct($msg){
 $this->msg=$msg;
 }
 public function __store(){
 //存入数据库
 }
}

后来,随着论坛逐渐出名,就有一些人在上面发链接,就需要对含有链接的消息进行过滤,论坛进一步发展,发现除开发垃圾链接的外,还有很多无用的灌水,到后来可能还有攻击等等各种不正常的帖子,所以对论坛帖子的管理,可以单独抽象出一个类进行管理,当需要扩充过滤规则时,可以进行动态扩充。

//基类
abstract class Filter{
 abstract public function isForbid();
}
//基础过滤类
class MsgFilter extends Filter{
 public $content;
 public function __construct($msg){
 $this->content=$msg;
 }
 public function isForbid(){
 if(preg_match("/https?/i",$this->content)){
 return [true,"Not Allowed Urls"];
 }else{
 return [false];
 }
 }
}
//装饰器,用来扩充功能
abstract class FilterDecorator extends Filter{
 protected $obj;
 public function __construct(Filter $obj){
 $this->obj=$obj;
 }
}
//新过滤器,判断是否重复发帖
class repeat extends FilterDecorator{
 public function isForbid(){
 if($this->obj->isForbid()[0] === true){
 //判定是否包含url
 return $this->obj->isForbid();
 }else if($this->obj->content == "this is a test"){
 //判定是否重复发帖
 return [true,"Repeat Posts"];
 }else{
 return [false];
 }
 }
}
$test = new MsgFilter("httpsfdjoafdsajof");
print_r($test->isForbid());//被禁止
$test2 = new repeat(new MsgFilter("this is a test"));
print_r($test2->isForbid());//被禁止

在python中,不存在抽象类和方法,实现就更加简单:

#!/usr/bin/env python
class Filter():
  pass
class MsgFilter(Filter):
  def __init__(self,msg):
    self.content=msg
  def isForbid(self):
    if('http' in self.content):
      return [True,"Not Allowed Urls"]
    else:
      return [False]
class FilterDecorator(Filter):
  def __init__(self,obj):
    self._obj=obj
class Repeat(FilterDecorator):
  def isForbid(self):
    if self._obj.isForbid()[0]:
      return self._obj.isForbid()
    elif self._obj.content == 'this is a test':
      return [True,"Repeat Posts"];
    else:
      return [False]
test = MsgFilter("this is a content have http urls")
print test.isForbid()
test2 = Repeat(MsgFilter('this is a test'))
print test2.isForbid()

Javascript中,没有严格的类,所有继承都基于原型,理解起来会稍费功夫:

function MsgFilter(msg){
 this.content=msg;
 this.isForbid=function(){
 if(this.content.match(/http/g)){
 return [true,"Not Allowed Urls"];
 }else {
 return [false];
 }
 }
}
function Repeat(obj){
 var _obj=obj;
 this.isForbid=function(){
 if(_obj.isForbid[0] === true){
 return _obj.isForbid();
 }else if(_obj.content=='this is a test'){
 return [true,"Repeat Posts"];
 }else{
 return [false];
 }
 }
}
var test = new MsgFilter("his is a content have http urls");
console.log(test.isForbid());
var test2 = new Repeat(new MsgFilter("this is a test"));
console.log(test2.isForbid());

由于Javascript缺少类的特性,继承对于它来说就显得有点鸡肋了,上面的代码看起来更像是对两个函数的处理, 在python中,有更加简单的添加装饰器的方法,直接通过”@”给函数自动添加装饰器,达到扩展功能的目的,如:

def Decorator(F):
  def newF(age):
    print "You Are Calling",F.__name__
    F(age)
  return newF
@Decorator
#通过@给函数showAge添加装饰器Decorator
def showAge(age):
  print "hello , i am %d years old"%age
showAge(10)

装饰模式的目的是解决动态扩展功能的难题,装饰模式的本质是对对象的灵活处理,理解装饰模式,不仅能深入了解面向对象的程序设计,更能提高编程的思维能力。

PHP 相关文章推荐
一个php导出oracle库的php代码
Apr 20 PHP
php gd2 上传图片/文字水印/图片水印/等比例缩略图/实现代码
May 15 PHP
分享PHP header函数使用教程
Sep 05 PHP
php 使用file_get_contents读取大文件的方法
Nov 13 PHP
php读取der格式证书乱码解决方法
Jun 22 PHP
WordPress中"无法将上传的文件移动至"错误的解决方法
Jul 01 PHP
PHP接收json 并将接收数据插入数据库的实现代码
Dec 01 PHP
PHP使用http_build_query()构造URL字符串的方法
Apr 02 PHP
PHP日志LOG类定义与用法示例
Sep 06 PHP
Laravel find in set排序实例
Oct 09 PHP
laravel withCount 统计关联数量的方法
Oct 10 PHP
php+laravel 扫码二维码签到功能
May 15 PHP
php对象在内存中的存在形式分析
Feb 03 #PHP
浅析THINKPHP的addAll支持的最大数据量
Feb 03 #PHP
php格式化金额函数分享
Feb 02 #PHP
php可应用于面包屑导航的迭代寻找家谱树实现方法
Feb 02 #PHP
php生成唯一的订单函数分享
Feb 02 #PHP
php数组去除空值函数分享
Feb 02 #PHP
php截取字符串函数分享
Feb 02 #PHP
You might like
PHP6 先修班 JSON实例代码
2008/08/23 PHP
php获取发送给用户的header信息的方法
2015/03/16 PHP
laravel 解决后端无法获取到前端Post过来的值问题
2019/10/22 PHP
JavaScript For Beginners(转载)
2007/01/05 Javascript
xtree.js 代码
2007/03/13 Javascript
判断对象是否Window的实现代码
2012/01/10 Javascript
利用js实现在浏览器状态栏显示访问者在本页停留的时间
2013/12/29 Javascript
JQuery与JS里submit()的区别示例介绍
2014/02/17 Javascript
node.js中的fs.fchown方法使用说明
2014/12/16 Javascript
js获取及判断键盘按键的方法
2015/12/01 Javascript
jQuery 如何给Carousel插件添加新的功能
2016/04/18 Javascript
Jquery ui datepicker设置日期范围,如只能隔3天【实现代码】
2016/05/04 Javascript
微信小程序之数据双向绑定与数据操作
2017/05/12 Javascript
JS一个简单的注册页面实例
2017/09/05 Javascript
js数组常用最重要的方法
2018/02/04 Javascript
angularjs实现table增加tr的方法
2018/02/27 Javascript
如何在 JavaScript 中更好地利用数组
2018/09/27 Javascript
监听element-ui table滚动事件的方法
2019/03/26 Javascript
JavaScript箭头函数中的this详解
2019/06/19 Javascript
vue 导航内容设置选中状态样式的例子
2019/11/01 Javascript
js实现贪吃蛇游戏(简易版)
2020/09/29 Javascript
Python中Iterator迭代器的使用杂谈
2016/06/20 Python
Python内建函数之raw_input()与input()代码解析
2017/10/26 Python
python实现汉诺塔算法
2021/03/01 Python
如何使用python把ppt转换成pdf
2019/06/29 Python
Python3+Requests+Excel完整接口自动化测试框架的实现
2019/10/11 Python
如何更改 pandas dataframe 中两列的位置
2019/12/27 Python
win10安装tensorflow-gpu1.8.0详细完整步骤
2020/01/20 Python
python批量修改文件名的示例
2020/09/27 Python
加拿大城市本地限时优惠:Buytopia.ca
2018/09/19 全球购物
本科毕业生自我鉴定
2013/11/02 职场文书
数字化校园建设方案
2014/05/03 职场文书
幼儿教师师德承诺书
2014/05/23 职场文书
2016年寒假社会实践活动总结
2015/03/27 职场文书
公司年会晚会开幕词
2019/04/02 职场文书
Python中可变和不可变对象的深入讲解
2021/08/02 Python