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

2025-05-29 0 71

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

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

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

?

1

2

3

4

5

6

7

8

9
class SaveMsg(){

private $msg;

public function __construct($msg){

$this->msg=$msg;

}

public function __store(){

//存入数据库

}

}

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

?

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

29

30

31

32

33

34

35

36

37

38

39

40

41

42

43
//基类

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中,不存在抽象类和方法,实现就更加简单:

?

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26
#!/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中,没有严格的类,所有继承都基于原型,理解起来会稍费功夫:

?

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26
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中,有更加简单的添加装饰器的方法,直接通过”@”给函数自动添加装饰器,达到扩展功能的目的,如:

?

1

2

3

4

5

6

7

8

9

10
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)

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

收藏 (0) 打赏

感谢您的支持,我会继续努力的!

打开微信/支付宝扫一扫,即可进行扫码打赏哦,分享从这里开始,精彩与您同在
点赞 (0)

声明:本站所有文章,如无特殊说明或标注,均为本站原创发布。任何个人或组织,在未征得本站同意时,禁止复制、盗用、采集、发布本站内容到任何网站、书籍等各类媒体平台。如若本站内容侵犯了原著者的合法权益,可联系我们进行处理。

快网idc优惠网 建站教程 PHP、Python和Javascript的装饰器模式对比 https://www.kuaiidc.com/102480.html

相关文章

发表评论
暂无评论