一区二区久久-一区二区三区www-一区二区三区久久-一区二区三区久久精品-麻豆国产一区二区在线观看-麻豆国产视频

jQuery is DSL (Part 1 - DSL)

  jQuery剛剛出來的時候,我沒有太多關注它,覺得這不過是Yet Another JavaScript Library。早期的jQuery專注于DOM節點的篩選與操作,不提供眾多的基礎類擴展,更不提供UI組件,因此體積能夠做到很小。然而,我實在看不出它和我熟悉的Prototype比有什么明顯的優勢——jQuery能做的各項獨立的操作,Prototype都能做。

  后來用jQuery的人越來越多,并且大家都愛用它的鏈式方法調用,甚至還把這種寫法推廣到其它語言中去。例如ASP.NET MVP Omar AL Zabir就把他的服務器端C#組件設計為支持鏈式方法調用的。這時候我才開始關注jQuery,并且逐漸喜歡上了鏈式方法調用的寫法,也在我自己的JavaScript組件中實現類似的API(參考AsyncOverload)。最后,我突然明白到,這其實就是一種Internal DSL嘛!

  在這篇文章里,我準備先討論Internal DSL,在下一篇文章里面再解釋為什么jQuery是Internal DSL。現在我們就從最根本的問題開始吧——

什么是Internal DSL?

  DSL是指Domain Specific Language,也就是用于描述和解決特定領域問題的語言。例如說,我們有專門描述字符串特征的正則表達式,有專門描述數據庫查詢的SQL,有專門描述XML結構的DTD和XSD,甚至有專門描述XML變換的XSLT,這些都是DSL。

當然,并非我們關注的領域都有現成的DSL,這時候我們有三個選擇:

  1. 使用通用語言描述該領域的問題(non-DSL)
  2. 發明一門全新的語言描述該領域的問題(External DSL
  3. 在一門現成語言內實現針對領域問題的描述(Internal DSL

例如說,我們現在要描述一個很簡單的金融領域問題,“我在花旗銀行存款$200”這樣一句話對應的三種法寫法可能是:(假設已經存在I和CitiBank兩個實體實例)

  1. I.DepositTo(new USD(200), CitiBank); /* C# */
  2. I deposit 200USD to CitiBank /* E-DSL */
  3. I.deposit(200.USD()).to(CitiBank); /* I-DSL */

第1種做法的成本最低,你只需要有OO的思想就可以了,你總能把實體類設計出來,但可能和人類描述此領域問題的思維方式有一定偏差(為什么USD可以new?為什么不是deposit [something] to [somewhere]?)。

 

第2種做法的成本最高,你需要寫一個全新的解釋器,至少是寫一組全新的規則,然后讓YACC這類工具幫你生成一個解釋器,但這樣出來的語法最貼近人類思維方式,甚至就如同自然語言一樣流暢。

第3種做法術語上述兩者的折中方案,如果語法不太復雜可以使用Builder模式實現語法分析,寫出來的語法相當貼近自然語言,但還是有學習門檻。由于腳本語言有相當的靈活性,所以現在很多人傾向于選擇在腳本語言內實現Internal DSL。

如何構造Internal DSL?

常見的兩種Internal DSL實現方法是Method ChainingFunction Sequence。如果我們需要描述一臺機器的硬件組成,兩種實現方式的代碼分別如下:

/* Method Chaining */
computer()
  .processor()
    .cores(2)
    .i386()
  .disk()
    .size(150)
  .disk()
    .size(75)
    .speed(7200)
    .sata()
  .end();

/* Function Sequence */
computer();
  processor();
    cores(2);
    processorType(i386);
  disk();
    diskSize(150);
  disk();
    diskSize(75);
    diskSpeed(7200);
    diskInterface(SATA);

無論是哪一種寫法,中間都必須寫一個分析器層。就如同語法分析器需要使用狀態機一樣,Internal DSL的實現也必須內置一個狀態機,以記錄當前執行到什么狀態了,并且接下來可以轉移到哪些有效狀態。

由于這不是一篇專門講語法分析器和狀態機實現的文章,所以我們把關注點保持在API層面就可以了,不深入討論其實現細節和成本。我們知道鏈式方法調用能夠實現Internal DSL就夠了,至于jQuery是如何利用好這一點的,我們在下一篇文章里再作討論。

小結

在這篇文章里,我們了解了Internal DSL與External DSL之間的區別,同時還了解到實現Internal DSL的具體方式,這為我們接下來討論jQuery的Internal DSL式接口做好了鋪墊。在下一篇文章里,我們將深入地來看看為什么jQuery的接口要如此設計,它能為用戶帶來了怎樣的便利,同時它自身的實現上又有什么優勢。

    it知識庫jQuery is DSL (Part 1 - DSL),轉載需保留來源!

    鄭重聲明:本文版權歸原作者所有,轉載文章僅為傳播更多信息之目的,如作者信息標記有誤,請第一時間聯系我們修改或刪除,多謝。

    主站蜘蛛池模板: 欧美人禽猛交狂配免费看 | 激情小视频网站 | 好吊妞视频haodiaoniucom | 成人午夜免费视频毛片 | 91在线视频免费播放 | 免费一区二区三区视频狠狠 | 日韩专区中文字幕 | 网站色哟哟 | 亚洲精品在线播放视频 | 狠狠色噜噜狠狠狠97影音先锋 | 俺也去老色官网 | 婷婷开心综合 | 国产福利资源 | 中日韩美中文字幕 | 激情小说色图 | 女女同性一区二区三区四区 | 精品中文字幕久久久久久 | www.小视频| 香蕉人人超人人超免费看视频 | 日韩在线中文字幕 | 成人精品 | 日本一区二区三区四区五区 | 成人成人性区 | 国产性色视频在线高清 | 成年网站未满十八禁毛片免费 | 99久久精品国产免看国产一区 | 亚洲国产精品免费在线观看 | 一级黄色a毛片 | 成人国产精品2021 | 激情综合五月亚洲婷婷 | 亚洲福利一区福利三区 | 东京加勒比中文字幕波多野结衣 | 美女黄板视频 | 四色婷婷 | 深爱激动网婷婷狠狠五月 | 激情在线视频 | 亚洲综合激情另类专区 | 一区二区三区四区亚洲 | 欧美一级一一特黄 | 色播六月 | 97人人在线视频 |