Ruby DSL介绍及其在测试数据构造中的使用(2)

在(1)中介绍了DSL和普通的函数定义之间的区别。在(1)的最后提到,DSL分为内部DSL和外部DSL,我们再看一遍他们的定义:1. External DSLs 用不同于host语言的语言来编写,通过编译和解释器来翻译成host语言 2. Internal DSLs 将host语言转化为DSL本身。

第二种方式,相比较第一种来说,构建DSL更为简便,还可以利用host语言本身已有的语言特征和库等,缺点是定义DSL的时候会受到host语言的限制。但是,如果我们选择了一门语法友好、灵活的编程语言作为host语言的话,那么我们就可以放大内部DSL的优点,弱化它的缺点,最终达到效率和回报的最佳值,那么Ruby是一个好的选择。

Why Ruby?Obie Fernandez InfoQ论坛上列出了Ruby的一些feature:


当然了,Ruby的语法有很多值得玩味的地方,比如。。。(也许这需要好几篇文章来写了)。好吧,让我们直接一点:Ruby DSL的强大直接来自于Ruby强大的元编程能力。

 

 

 

“元”这个词来源于希腊语中表示”…之间, …之后, …超越’的前缀 ”meta”,有”超越”和”高阶”的意思。正如”元数据”表示”数据的数据”, “元模型”表示”模型的模型”, “元编程”表示”编程的编程”。

在《programming ruby 1.9》中说过:” Programming is all about building layers of abstractions.” 有些语言,比如C语言,更接近与机器本身,从而导致C语言跟应用之间的距离比较遥远。而有些语言,比如Ruby,提供了更高层次的抽象,从而让你在更接近于特定领域的地方开始编程。但是,当你使用元编程时,你就不再局限于语言本身的抽象层次了,你可以基于母语言构建新的抽象。于是,” In effect, you’re creating a new, domain-specific programming language—one that’s designed to let you express the concepts you need to solve your particular problem.”

扫描二维码关注公众号,回复: 763708 查看本文章

所以,从这个角度看,元编程是DSL的基础。几乎每一种Ruby DSL的实现都离不开元编程。下面我归纳一下Ruby DSL的几种方式以及其中利用到的元编程技巧:

 

1. Class Marcos



 你可以把”ofter_create”, ”has_many”看做一种声名。它的实现依赖于元编程中反射功能中的define_method或者instance_eval, 可以为函数动态添加方法。

 

 

2.  Methods on an Object



 =>



 这是一种构造层次数据的DSL,完全基于Builder::XmlMarkup的对象。我们可以发现,生成的XML中定义的关键字完全可以由对象的方法名来决定,但是这个对象是如何做到满足所有任意添加的关键字呢?这其中的实现就依赖于元编程中反射功能中的method_missing方法。

 

3.  自定义词汇



 Ruby on Rails中,你可以:

4.years  +  13.days  +  2.hours

或者我们可以为这些词汇定义上下文:



  第一种你可以叫做自定义词汇,第二种你可以叫做单位的实现,第三种你可以叫做上下文相关的词汇,但是本质上他们都是一样的。由于Ruby中没有函数,只有方法,意味着所有方法的使用都关联到一个对象。第一种实现,实际上调用者是Object的对象,第二种,调用者是4,13,2这些Fixnum的对象,第三种实现通过关键字task之后的关键字来调用不同的对象来调用这些方法。

这些DSL的实现依赖于Ruby开放类,block(其实相当于高阶函数)等特性。

 

基本上RubyDSL都是基于以上三种模式的组合和扩展,当然其实现背后的技巧可以非常不同和巧妙。

        (文章转载请注明出处)

猜你喜欢

转载自jdoc.iteye.com/blog/1269707
今日推荐