Ruby 实现装饰器模式

了解过Python的同学都知道,Python从语言层面支持装饰器模式,并配以语法糖 『@xx』的形式来提供更好的开发体验。本文的目标就是在Ruby中的实现对标Python的装饰器。

在开始之前我们还是来重温一下,Python中装饰器这个概念:

装饰器本质上是一个函数,它可以让其他函数在不需要做任何代码变动的前提下增加额外功能,装饰器的返回值也是一个函数对象。它经常用于有切面需求的场景,比如:插入日志、性能测试、事务处理、缓存、权限校验等场景。装饰器是解决这类问题的绝佳设计,有了装饰器,我们就可以抽离出大量与函数功能本身无关的雷同代码并继续重用。

总的来说,就是实现面向切面的AOP编程。

Python装饰器

既然是要对标Python,那我们来看看Pyhton的装饰器,装饰器在Python中分类两大类,一类是较为常见的函数装饰器,另一类是类装饰器,文本只针对函数装饰器。

函数装饰器

def logger(func):
    def wrapper(*args):
        print('execute %s function' % func.__name__)
        return func(*args)
    return wrapper

@logger
def talkto(someone):
    print('talk to %s' % someone)

talkto('jam')

''' 输出
execute talkto function
talk to jam
'''

上面的Python代码就是装饰器的例子,可以看出来,其实利用了闭包去是先携带上下文中的func函数对象,再返回包装函数,然后让包装函数接受原来的参数,并在包装函数最后返回原函数的结果。

Ruby装饰器

我们相应在Ruby中实现完全相同的装饰器模式,恐怕是不行了,Python使用的语法糖『@』在Ruby中是实例变量的关键字,无法重载,所以要使用其他的方法。

要达到的效果:

class Demo

  extend MyDecorator

  def logger(method, *args)
    puts "before execute #{method.original_name}"
    yield
    puts 'after execute'
  end

  wrap :logger
  def talk_to(someone)
    puts 'talk to ' + someone
  end
end

Demo.new.talk_to '特朗普'

# 输出:
# before execute talk_to
# talk to 特朗普
# after execute

上面的代码,在Demo类的talk_to方法外面包裹了装饰器,装饰方法是 logger。 包装是通过类宏wrap方法提供的,它会将传入的symbol方法名,作为装饰方法。

在装饰方法logger 使用统一的接口:

  • 至少接受两个参数,一个是被包装的方法的Method对象,另一个是该方法的参数。
  • 通过yield关键字执行被包装方法,(这样在yield前后的代码就是前置后置执行的了)。

在上面的例子中虽然没有写出来的,其实可以通过想yield中传递参数来覆盖,原有方法的参数。

因为装饰方法的第一个参数是Method对象,所以如果你愿意的话,可以不通过yield而使用method.call 去执行被包装的talk_to方法。

实现

上面的装饰器例子的实现,全都是基于扩展了MyDecorator这个模块,那么MyDecorator中究竟做了什么呢,我们来看看:

module MyDecorator

  def method_added(method_name)

    unless decorator_methods.empty?

      decorator_method = decorator_methods.pop

      new_name = "#{method_name}_without_decorator"

      alias_method new_name, method_name

      define_method method_name do |*args|
        method(decorator_method).call method(new_name), *args do |p = args|
          method(new_name).call(*p)
        end
      end

    end

  end

  def wrap(decorator)
    decorator_methods << decorator
  end

  def decorator_methods
    @decorator_methods ||= []
  end

end

MyDecorator 模块目前非常的简单之定义了三个方法。

  • method_added 挂在类中方法定义是的钩子方法
  • wrap 定义装饰的类宏,将装饰方法压入 装饰器队列中。
  • decorator_methods 存储装饰器方法

method_added 中通过每次弹出一个队里中的装饰方法,再将原方法重命名,然后通过动态定义方法 define_mehtod 在原方法外包裹一层,这就实现了装饰器。

总结

装饰器是在Python原生支持,并且可以通过语法糖提供便捷的实用方法。Ruby中虽然原生不支持,但是因为语言本身的DSL能力非常强所以我们可以做出一个类Python的装饰器。因为Ruby是通过block来模拟闭包的,所以在Ruby实现中我们也使用了block,并且因为Ruby是纯OOP的,函数非一等公民,只能通过函数对象去模拟函数的传递和调用。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 194,242评论 5 459
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 81,769评论 2 371
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 141,484评论 0 319
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 52,133评论 1 263
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 61,007评论 4 355
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 46,080评论 1 272
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 36,496评论 3 381
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 35,190评论 0 253
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 39,464评论 1 290
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 34,549评论 2 309
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 36,330评论 1 326
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 32,205评论 3 312
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 37,567评论 3 298
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 28,889评论 0 17
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,160评论 1 250
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 41,475评论 2 341
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 40,650评论 2 335

推荐阅读更多精彩内容

  • 一、异同对比选择1、Python和ruby的相同点: * 都强调语法简单,都具有更一般的表达方式。python是缩...
    沃伦盖茨阅读 4,126评论 2 24
  • 两本不错的书: 《Python参考手册》:对Python各个标准模块,特性介绍的比较详细。 《Python核心编程...
    静熙老师哈哈哈阅读 3,350评论 0 80
  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,494评论 18 139
  • 阳台上的那盆栀子花,是需要很多阳光,很多水分,很多关爱的一盆花。 岑静记不清哭了多久了,突然门外响起笃笃的敲门声。...
    惠风和畅806阅读 285评论 0 0
  • 我们在职场中求职的时候,面试官就是老大,只要不和他的心意,绝对不能成功入职,以下就是面试官最不喜欢的六种面试者。不...
    乐仲雨阅读 479评论 0 0