青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品

隨筆-341  評論-2670  文章-0  trackbacks-0

手把手教你寫腳本引擎(一)——挑選語言的特性

 

陳梓瀚

華南理工大學軟件本科05

vczh@163.com

http://m.shnenglu.com/vczh/

 

腳本引擎的作用在于增強程序的可配置性。從游戲到管理系統都需要腳本,甚至連工業級產品的Office3DS Max以及AutoCAD等都添加了屬于自己的腳本語言。DHTML的出現讓我們可以在網頁代碼中嵌入腳本語言,PHPASP等技術的出現讓我們可以將一個應用程序的界面換成網頁,而邏輯使用腳本語言編寫。現在腳本語言的種類繁多,Python的發展讓BOOST庫擁有了對Python的支持,Rails框架的出現壯大了Ruby的實力,LUA更是被大量應用在游戲開發中。Windows甚至提供了wscript以便讓我們能夠調用javascriptvbscript的代碼。


既然有了這么多可供選擇的腳本引擎,為什么我們仍然要開發自己的腳本引擎呢?首先,我們并不能保證現有的腳本引擎能夠滿足我們做出來的系統。因為我們所需要的腳本可能很簡單,用現有的腳本引擎比較浪費。或者我們的腳本復雜,但是功能比較“神奇”(譬如SQL)以至于沒有能夠滿足我們需要的腳本引擎。因為腳本并不一定是通用語言,腳本僅僅是為了滿足我們增強系統的可配置性而出現的。其次,腳本引擎足夠復雜,可以訓練我們的編程能力。在我們的業余時間里面開發出來的程序并不完全是為了滿足某個應用的需要而產生的,有可能是我們為了自身的提高而進行的摸索。開發腳本引擎足以成為鍛煉的方法之一。

 

計算機語言作為一個計算的定義,在我們開發腳本引擎之前需要先進行了解。對于目前流行的若干種語言,我們可以抽象出一組正交屬性來描述他們。

 

一、命令式與描述式

 

一門語言是命令式或者描述式取決于這門語言是用來告訴計算機怎樣做還是做什么的。舉個例子,SQLProlog是描述式語言,而C++C#等則是命令式語言。我們在使用SQL的時候告訴服務器的是我們需要滿足什么條件的數據項,而不是告訴服務器我們需要通過什么計算來獲得自己所需要的數據項。描述式的語言的優點在于其可讀性好。C# 3.0為數據查詢加入了LINQ讓我們可以在C#中書寫類似SQL的代碼查詢數據。

 

另一個比較模糊的例子則是HaskellHaskell很難區分是命令式語言還是描述式語言。因為從形式上來說我們告訴編譯器的是我們想做什么而不是我們想怎么做,但是Haskell給我們的工具的粒度太細以至于我們為了告訴編譯器做什么的同時仍然需要考慮一個問題是如何被解決的。

 

二、按值計算與惰性計算

 

惰性計算的語言很少出現以至于可能很多人都不知道“原來語言可以是這個樣子的”。惰性計算的精神是不去執行沒用的代碼。什么是沒用的代碼呢?只要是這段代碼的值不對外界產生任何影響,譬如沒有往屏幕、硬盤或者是其他什么地方寫點什么數據,就是沒有用的。當然,至于這段代碼中間做了些什么事情那是不管的。

 

舉一個比較簡單的例子,假設現在有如下代碼:

function PrintAndReturn(Message,Result)

{

    Print(Message);

    return Result;

}

function DoSomething(BoolA,BoolB)

{

    If(BoolA || BoolB) Print(“!”);

}

DoSomething(PrintAndReturn(“Hello”,true),PrintAndReturn(“World”,false));

DoSomething函數傳入兩個參數,都是布爾類型的。如果這兩個參數其中有一個是true的話那么就往屏幕上打出一個感嘆號。PrintAndReturn函數接受兩個參數,往屏幕上打出第一個參數,函數返回第二個參數。

 

對于一門按值計算的語言,也就是我們平常見到的那種,執行的結果是“HelloWorld!”。因為為了調用DoSomething我們需要首先獲得兩個布爾值。

 

對于一門惰性計算的語言,執行的結果是“Hello!”。因為DoSomething在對BoolA || BoolB進行求值的時候計算了BoolA,發現是true,于是BoolB這個參數就沒有用了,因此PrintAndReturn(“World”,false)也就不會執行了,導致“World”不會顯示在屏幕上。

 

當然,對于上面舉的這個例子來說,這種語言有著惰性計算的屬性并不合理。一門語言為了不具有二義性,在存在惰性計算的同時必須對自己的類型系統進行改造。關于這方面的資料可以查閱Haskell語言中Monad的原理。Haskell作為一門惰性計算的語言,在不關心求值順序的同時,仍然保證結果的一致性。上面這個例子,如果程序對||的求值是從右操作數開始的話,那么輸出的結果就變成“HelloWorld!”了。惰性計算的好處在于可以在邏輯上表達無窮大的對象,而在實際的計算過程中并不需要將這個無窮大的對象一次性計算出來,而是需要哪里算到哪里。舉個例子:

 

function MakeArray(Index)

{

    return [Index]++MakeArray(Index+1);

}

function Sum(Array,Count)

{

    Result=0;

    for i=0 to Count-1

        Result+=Array[i];

   return Result;

}

Print(Sum(MakeArray(1),10));

在這個例子中,[Index]代表一個只有一個元素的數組,其內容是Index,而++操作符將兩個數組接起來。于是MakeArray(1)就產生了一個無窮長的數組,其內容是[1,2,3,4,…]Sum計算數組的前若干個數字的和。對于一門惰性計算的語言,這個例子將輸出55,因為我們需要的僅僅是前10個數字,因此MakeArray只需要遞歸10次就自動挺下來了。而對于一門按值計算的語言來說,將發生死循環而出現不可停機現象。

 

三、強類型、弱類型與無類型

 

一門語言是無類型當且僅當一個固定的符號的類型可以在運行時改變。譬如如下代碼:

TheVariable=1;

TheVariable=”I am a string!”;

第一行創建了一個int類型的TheVariable變量,而第二行則將TheVariable修改成了字符串類型。一門無類型語言的對象類型可以是數值、字符串、數組、類、閉包、函數指針等等的東西。

 

只要不是無類型的,那必然就是強類型或者弱類型的了。強類型與弱類型的分界線比較明顯。只要存在隱式類型轉換的語言則是弱類型的,譬如C語言能將int隱式轉換為double。不存在隱式轉換的語言也是存在的,譬如Haskell。在Haskell里面不能創建一個實數類型的名字但是綁定一個整數的值上去。因為整數跟實數的類型是不同的,而且不存在隱式轉換。

 

四、函數與閉包

 

    凡是支持閉包的語言必然是支持函數的,但是并不是所有支持函數的語言都支持閉包,而且也并不是所有的語言都有函數。Windows的批處理文件所能理解的語言就是不支持函數的語言的一個例子。

 

    至于什么是閉包呢?閉包就是可以保持函數執行的上下文的一種強大的函數指針。舉個例子:

    function Add(a)

    {

        return function(b)

        {

            Return a+b;

        }

    }

    Inc=Add(1);

    Inc10=Add(10);

    Print(Inc(5));

    Print(Inc10(5));

    這個例子將輸出615。執行Inc=Add(1);的時候,Add函數返回了一個新的函數,這個函數接受參數b并返回參數ab相加的結果。返回的這個函數將參數a記了下來。所以IncInc10在執行的時候,雖然執行的是同一個函數,但是這個函數所看到的a確是不同的。a的值的不同代表著IncInc10執行函數的不同。這也就是閉包是可以保持函數執行的上下文的由來了。當然,一門不支持閉包的語言是不能允許上面這種寫法的。

 

    這四種屬性是區分語言特征的重要屬性。至于一門語言是否支持面向對象的寫法或者支持元編程或者泛型之類的東西,并不是十分重要的特性,雖然我們使用起來的感覺非常不同。

 

    那么我們如何選擇我們所需要的特性呢?對于一個簡單的事務腳本來說,我們只需要非常簡單的特性諸如選擇結構和循環結構,和簡單的計算功能。計算功能可以支持表達式也可以不支持表達式。一門不支持表達式的語言看起來就像MASM支持的那種有宏的匯編語言。就像前些日子CSDN抄得很熱的概念DSL一樣,我們在設計一門腳本語言的時候,想的不應該是這門語言如何如何強大,而應該是這門語言應該如何更好地表達領域相關的信息。

 

    下面這幅圖片顯示的是筆者在高中的時候開發的一款RPG的地圖編輯器。眾所周知,RPG是需要劇情的,因此編輯器需要在地板上或人物上設置陷阱引發腳本的執行。

 

RPG由于劇情復雜,需要的控制方法也就很多,因此供給RPG使用的腳本至少應該支持選擇和循環等。而且有的時候需要使用腳本來完成某些動畫(譬如上圖中的開門腳本),因此腳本也就需要函數了。至于為什么上面的腳本使用Pascal的語法僅僅是因為筆者當時Delphi用得比較多。這也是筆者第一次實現的一款腳本引擎。

 

那么,我們如何選擇腳本語言的特性呢?我們要考慮一下系統的復雜度,因為腳本語言的特性跟我們想提供給腳本語言的庫是有很大關系的。

 

舉個例子,如果提供給腳本的庫經常需要調用到腳本的函數的話(比如GUI,比如可以給腳本用的類似YACC的東西等),那么腳本最好具有閉包的特性,沒有的話至少也得有函數指針這種類型。如果提供給腳本的庫的大部分函數都可以接受很多種不同類型的對象的話,那么腳本最好是無類型的。如果庫很龐大,大到不得不用命名空間和類來提供的話,那腳本無論如何都要有類的。

 

對于某些專用領域的語言,一般都采用類似自然語言(但是具有嚴格定義)的外觀來組織腳本,最好的例子就是SQL了。如果從語言的角度看,SQLselect是一個具有很多參數,而且大部分參數都具有缺省值的函數,而且大部分函數都是一些lambda表達式。因為lambda表達式出現得太多,因此就需要簡化lambda表達式的語法了。所以最終出現在我們面前的語法就是select中到處都可以寫有參數的表達式,而且這些參數來自于select的表名和重命名。

 

如果腳本本身需要非常快的話,那么最好使用強類型或者弱類型。因為這兩種特性的語言的每一個符號都是有確定的類型的,虛擬機的開發不僅有很多方法,而且還有可能做成JIT(也就是編譯成機器碼)。在這種情況下,庫的供給就要非常注意了。因為在大部分情況下腳本都是在跟庫打交道的,所以交互的部分要詳細考慮。

 

如果腳本僅僅是用來做一些簡單的配置工作的話,那么表達式可以全免,用命令的外觀設計語法。而且在大多數情況下連函數都可以免。這樣的話這門語言就剩下變量、分支和循環了,就跟Windows的批處理一樣。

 

最后一個需要提及但是大部分情況下不用管的屬性就是腳本的計算能力。這個計算能力說的不是計算的速度,而是解決的問題的范圍。這個屬性就是圖靈完備了。通俗地講,對于任何一個數學問題,如果只要C語言算得出來腳本語言都算得出來的話,那么這門腳本語言就是圖靈完備的了。當然,因為C語言也是圖靈完備的,而且圖靈完備的計算能力在有限線程的計算機中是最高的,因此不存在一個數學問題,某種語言算得出來而C語言算不出來。那么如何判斷一門語言是不是圖靈完備的呢?

 

簡單的來說,有數組的語言就是圖靈完備的,有閉包的語言也是圖靈完備的。如果數組也沒有,閉包也沒有,那么有結構(C語言的structPascalrecord)和有指向結構的指針的語言也是圖靈完備的。因為閉包的內部結構也是一些保留環境的struct,因此只要能表達遞歸數據結構的語言都是圖靈完備的。

 

這一篇文章就先將到這里了。下一篇文章將會講述如何實現最簡單的命令型腳本語言,再下一篇文章開始將會有幾篇文章講述如何實現一門有數組和函數的弱類型腳本語言,接著會對這門語言進行擴充。

posted on 2008-07-07 07:45 陳梓瀚(vczh) 閱讀(21687) 評論(12)  編輯 收藏 引用 所屬分類: 腳本技術

評論:
# re: 手把手教你寫腳本引擎(一)——挑選語言的特性 2008-07-07 10:02 | sflypig
有一個人寫了一本書叫“自己動手寫操作系統”。
Vczh應該寫一本書叫“手把手教你寫腳本引擎”。  回復  更多評論
  
# re: 手把手教你寫腳本引擎(一)——挑選語言的特性 2008-07-07 17:13 | 空明流轉
我說vc,我下次干脆開一個專題叫手把手教你寫Shader編譯器好了。。。
最近就在干這個。。。正在給shader挑選語法糖。。。  回復  更多評論
  
# re: 手把手教你寫腳本引擎(一)——挑選語言的特性 2008-07-07 17:29 | 路過
學習一下,現在的小DD都這么厲害啦。  回復  更多評論
  
# re: 手把手教你寫腳本引擎(一)——挑選語言的特性 2008-07-07 19:19 | 陳梓瀚(vczh)
語法糖得你挑,我對Shader的認識又不深入,是你才天天寫Shader……  回復  更多評論
  
# re: 手把手教你寫腳本引擎(一)——挑選語言的特性 2008-07-07 22:28 | 空明流轉
@陳梓瀚(vczh)
我是說,我寫一個Shader的手把手,和你這個遙相呼應一下。  回復  更多評論
  
# re: 手把手教你寫腳本引擎(一)——挑選語言的特性 2008-07-08 18:09 | cexer
多謝博主的分享!  回復  更多評論
  
# re: 手把手教你寫腳本引擎(一)——挑選語言的特性 2008-08-03 07:07 | Lnn
教教我吧,從最簡單的教起。。。我會很努力的。。。  回復  更多評論
  
# re: 手把手教你寫腳本引擎(一)——挑選語言的特性 2008-10-03 06:15 | 免費小說
用C++寫么,,,,,,,,  回復  更多評論
  
# re: 手把手教你寫腳本引擎(一)——挑選語言的特性[未登錄] 2010-09-23 01:28 | forrest
寫的不錯,贊一個!

首先關于第三點:強類型、弱類型與無類型。樓主說的太簡略,這里斗膽補充一下:無類型是指變量是無類型的,但是變量指向的對象實際上是有類型的。

不過話說回來,雖然說語言的編程范式(如面向對象或者面向過程),跟語言本身不是特別有關系(用面向過程的語言,一樣可以編寫面向對象的代碼),但是這是非常大的語法糖,所以也是一個非常重要的考慮方面。異常處理和泛型編程也是一樣的。

另外需要補充幾點的是:1. 語言本身的內存管理也是一個非常重要的考慮方面,即是否支持垃圾回收和引用計數。2. 語言的動態程度,這個與樓主的第三點有點重疊,但是支持多大程度的元編程確實是個問題。3. 參數和返回值的傳遞方式(按值傳遞還是按引用傳遞,etc.)。4. 作用域規則  回復  更多評論
  
# re: 手把手教你寫腳本引擎(一)——挑選語言的特性 2010-10-08 03:51 | tt
相對于硬件來說,一切都是描述式語言吧 -_-  回復  更多評論
  
# re: 手把手教你寫腳本引擎(一)——挑選語言的特性 2010-10-08 06:52 | 陳梓瀚(vczh)
@tt
代碼是給人看的,不是給硬件看的。這就是為什么硬件不能運行代碼,只能運行編譯后的二進制指令的原因了。所以描述式是針對人而言的哈。  回復  更多評論
  
# re: 手把手教你寫腳本引擎(一)——挑選語言的特性[未登錄] 2013-06-26 14:00 | alex
>>一門語言是無類型當且僅當一個固定的符號的類型可以在運行時改變
python is a strongly typed (also dynamically typed) language.
you can do this -
var = 1
var = "string"
but you cannot do this -
var = 1 + "string"  回復  更多評論
  
青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>
            一区二区电影免费观看| 欧美+日本+国产+在线a∨观看| 亚洲美女免费精品视频在线观看| 国产网站欧美日韩免费精品在线观看 | 国产一区二区三区四区五区美女| 国产精品视频在线观看| 国产香蕉97碰碰久久人人| 国产一区日韩二区欧美三区| 精品动漫3d一区二区三区免费| 亚洲午夜久久久久久久久电影网| 久久精品国产一区二区三| 中日韩在线视频| 国产无遮挡一区二区三区毛片日本| 午夜精品久久久久久久久| 免费观看日韩| 久久免费高清| 亚洲肉体裸体xxxx137| 亚洲第一主播视频| 国产美女搞久久| 亚洲午夜激情网站| 99热在线精品观看| 亚洲一二三四久久| 久久婷婷蜜乳一本欲蜜臀| 亚洲国产成人精品视频| 国产精品99久久久久久www| 欧美伊人久久久久久久久影院| 久久亚洲欧美| 国产精品久久久久免费a∨| 欧美日韩黄色一区二区| 久久精品中文字幕一区| 欧美三级韩国三级日本三斤| 国产一区二区三区视频在线观看| 91久久精品一区| 欧美在线观看一区二区| 亚洲国产日韩欧美在线动漫| 怡红院精品视频| 黑人极品videos精品欧美裸| 日韩视频永久免费| 久热精品视频在线| 中日韩高清电影网| 欧美二区不卡| 亚洲高清在线观看一区| 久久精品一区二区国产| 一区二区不卡在线视频 午夜欧美不卡在 | 久久男人av资源网站| 欧美日韩综合视频| 亚洲片在线资源| 美女精品自拍一二三四| 午夜精品剧场| 国产精品视频yy9299一区| 国产精品99久久不卡二区| 亚洲二区在线观看| 伊人成人网在线看| 玖玖在线精品| 欧美激情一区二区三区四区| 国产人成精品一区二区三| 亚洲日韩欧美一区二区在线| 久久裸体视频| 欧美高清视频www夜色资源网| 亚洲国产午夜| 欧美激情亚洲自拍| 欧美一区二区三区久久精品茉莉花| 欧美一级电影久久| 一区二区日韩免费看| 欧美xart系列高清| 亚洲日本成人女熟在线观看| 久热国产精品视频| 久久婷婷综合激情| 在线观看日韩精品| 欧美大片一区| 欧美久久视频| 性亚洲最疯狂xxxx高清| 欧美在线免费视频| 亚洲大胆人体在线| 亚洲经典一区| 欧美天堂亚洲电影院在线播放 | 狠狠色丁香婷婷综合影院| 久久成人18免费网站| 欧美自拍偷拍| 欧美日韩亚洲网| 久久久久久综合网天天| 亚洲国产欧洲综合997久久| 久久婷婷丁香| 欧美激情视频在线免费观看 欧美视频免费一 | 午夜一区在线| 亚洲福利国产| 亚洲巨乳在线| 国产午夜精品美女视频明星a级 | 国产精品久久久久久久久免费| 西瓜成人精品人成网站| 欧美一区二区女人| 亚洲精品视频在线观看网站| 中文精品99久久国产香蕉| 国产欧美一区二区三区另类精品 | 亚洲宅男天堂在线观看无病毒| 国产精品美女主播| 欧美国产一区二区| 国产精品国产a级| 美腿丝袜亚洲色图| 国产精品不卡在线| 欧美激情在线观看| 国产精品网站在线| 亚洲国产精品ⅴa在线观看 | 噜噜噜久久亚洲精品国产品小说| 一区二区日本视频| 久久精品av麻豆的观看方式| 一卡二卡3卡四卡高清精品视频| 午夜伦欧美伦电影理论片| 夜夜嗨av一区二区三区中文字幕 | 国产婷婷色一区二区三区四区| 亚洲激情校园春色| 亚洲第一主播视频| 午夜精品久久久久久99热| 亚洲另类一区二区| 久久婷婷麻豆| 久久久久国产一区二区| 欧美午夜片在线观看| 亚洲国产1区| 在线播放亚洲| 久久精品免费| 欧美在线亚洲在线| 国产精品久久久| 亚洲欧洲一级| 亚洲欧洲日韩综合二区| 久久久九九九九| 欧美伊人久久大香线蕉综合69| 欧美久久久久久久| 中文精品视频| 欧美电影打屁股sp| 亚洲天堂男人| 欧美日韩国产区| 中文精品视频| 欧美高清视频一区二区| 精品51国产黑色丝袜高跟鞋| 欧美激情片在线观看| 欧美日韩国产麻豆| 免费在线成人| 欧美精品日本| 欧美二区视频| 亚洲国产视频直播| 久久久久国产精品一区二区| 欧美在线日韩| 国产日韩一区在线| 欧美亚洲一区三区| 久久爱91午夜羞羞| 国产深夜精品| 久久精品国语| 欧美成人有码| 亚洲激情影院| 欧美精品午夜视频| 亚洲毛片一区| 香蕉成人伊视频在线观看| 国产精品老女人精品视频| 亚洲一区999| 亚洲欧美视频在线观看| 国产精品一区二区在线观看| 亚洲欧美日韩国产成人| 久久精品91| 在线播放亚洲一区| 欧美精品电影在线| 亚洲视频一区在线观看| 久久成人精品电影| 影音先锋日韩有码| 欧美va天堂va视频va在线| 亚洲国内欧美| 亚洲欧美国产毛片在线| 国产精品久久毛片a| 久久精品99国产精品日本| 亚洲精品网址在线观看| 久久精品国产99国产精品| 亚洲国产视频一区| 国产美女精品一区二区三区| 久久中文久久字幕| 亚洲小视频在线| 欧美好吊妞视频| 欧美一区激情视频在线观看| 亚洲国产欧美久久| 国产精品久久福利| 免费亚洲婷婷| 亚洲欧美一区二区视频| 最新亚洲一区| 久久国产精品高清| 夜夜爽www精品| 永久免费视频成人| 国产精品午夜在线| 欧美国产日本韩| 香蕉精品999视频一区二区| 亚洲国产精品久久精品怡红院| 欧美在线一级视频| 久久精品亚洲一区| 亚洲美女精品久久| 久久视频免费观看| 亚洲综合成人在线| 亚洲免费成人av| 亚洲国产成人久久综合| 国产日韩欧美日韩| 国产精品日韩在线播放| 欧美日韩系列| 欧美国产三区| 久久欧美中文字幕|