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

隨筆-341  評論-2670  文章-0  trackbacks-0
    這篇短文的Idea來源于一篇論文。這篇論文的題目是Higier-Order Functions for Parsing,Graham Hutton寫的。論文中使用了一種叫Miranda的函數(shù)式語言來講述如何使用高階函數(shù)開發(fā)語法分析器。

    高階函數(shù)很多語言都支持,譬如JavaScript啊,C#的lambda expression啊,或者是我自己做的語言Vczh Free Script 2.0。不過Miranda是惰性計算的語言,我們常用的語言都不具有惰性計算的特性。因此我閱讀了這篇文章之后,自己用Vczh Free Script 2.0寫了一個等價的小規(guī)模的語法分析器。結(jié)構(gòu)跟論文中所提到的那個有所區(qū)別,不過相同的經(jīng)驗(yàn)可以直接應(yīng)用在JavaScript里面或其它語言(例如Python等)的lambda expression里。C#我不知道行不行,沒去考證。

    這里首先要解決一個問題,就是如何引用沒被定義的名字的問題。譬如如下文法:

    Term=<number> | "(" Exp ")"
    Factor=[ Factor ("*" | "/") ] Term
    Exp=[Exp ("+" | "-") ] Factor

    如果我們直接把這些東西寫進(jìn)代碼的話,那就會遇到Exp沒有定義的問題。因此每一個小parser我都定義為一個數(shù)組,這個數(shù)組只有一個元素。在運(yùn)算的時候每次都把元素取出來執(zhí)行,就可以模擬惰性計算在這里起到的作用了。

    好了,現(xiàn)在我們開始制作。我對Parser的定義是這樣的。一個Parser是一個只有一個函數(shù)的數(shù)組。這個函數(shù)接受一個輸入,返回一個結(jié)果的數(shù)組。因?yàn)檎Z法可能有歧義,所以返回多個結(jié)果是允許的。每一個結(jié)果由兩部分組成,第一部分是分析的結(jié)果,第二部分是分析到這里為止還剩下的字符串。

    首先,我們需要一個Fail,這個Fail無論輸入什么都返回空:
1 Fail=[func(Input)
2 {
3   return [];
4 }];
5 

    然后,我們需要一個Ch來檢查輸入的前綴是否跟定義的一樣:
 1 Ch=func(c)
 2 {
 3   return [func(Input)
 4   {
 5     if(#Input>=#c)
 6       if(Input[0:#c]==c)
 7         return [[c,Input[#c:#Input-#c]]];
 8     return Fail[0](Input);
 9   }];
10 };
    Ch的使用方法是這樣的:Ch(字符串)。譬如Ch("vczh")就返回了一個parser,這個parser在輸入"vczh123"的時候返回[ ["vczh" , "123"] ]。原因是這樣的。因?yàn)镃h("vczh")是沒有歧義的,所以返回包含一個結(jié)果的數(shù)組。這個結(jié)果又是一個數(shù)組,數(shù)組的兩個元素分別是分析的結(jié)果("vczh")和剩余的字符串("123")。

    為了方便,我們建立一個Regex來檢查輸入的前綴是否滿足一個正則表達(dá)式:
 1 Regex=func(Expression)
 2 {
 3   Expression=regexppure(Expression);
 4   return [func(Input)
 5   {
 6     local Match=matchhead(Expression,Input);
 7     if(Match!=null)
 8       return [[text(Match),Input[#text(Match):#Input-#text(Match)]]];
 9     else
10       return [];
11   }];
12 };
    Regex與Ch是類似的。實(shí)際上Regex可以用其他的手段組合起來。因?yàn)槲覀儸F(xiàn)在制作的分析器是可以分析Type-2文法的,遠(yuǎn)遠(yuǎn)比正則表達(dá)式所能表達(dá)的Type-3文法強(qiáng)大很多。不過為了使用方便這么做也不是壞事。

    接下來我們定義了一個Seq來表示多個parser串聯(lián):
 1 Seq=func({}Parsers)
 2 {
 3   return [func(Input)
 4   {
 5     local Result=[[[],Input]];
 6     for(p in Parsers)
 7     {
 8       local NewResult=[];
 9       for(r in Result)
10       {
11         for(pr in p[0](r[1]))
12           NewResult=NewResult++[[r[0]++[pr[0]],pr[1]]];
13       }
14       Result=NewResult;
15     }
16     return Result;
17   }];
18 };
    串聯(lián)的意思其實(shí)很簡單。Seq(Ch("1") , Ch("2") , Ch("3"))就等于說輸入必須由Ch("1")、Ch("2")和Ch("3")構(gòu)成。為什么要這么做呢,因?yàn)镾eq跟循環(huán)和分支配合起來的話會非常強(qiáng)大,詳見下文。

    好了,有了Seq我們就需要Alt:
 1 Alt=func({}Parsers)
 2 {
 3   return [func(Input)
 4   {
 5     local Result=[];
 6     for(p in Parsers)
 7       Result=Result++p[0](Input);
 8     return Result;
 9   }];
10 };
    Alt是分支的意思,譬如Alt(Ch("1") , Ch("2") , Ch("3")的意思是輸入可以是Ch("1")或Ch("2")或Ch("3")。

    然后我們就需要循環(huán)Any:
 1 Any=func(Parser,Max)
 2 {
 3   return [func(Input)
 4   {
 5     local Result=[[[],Input]];
 6     local Current=0;
 7     do
 8     {
 9       Produce=0;
10       if(#Result[Current][0]!=Max)
11         for(r in Parser[0](Result[Current][1]))
12         {
13           Result=Result++[[Result[Current][0]++[r[0]],r[1]]];
14         }
15       Current=Current+1;
16     }
17     while(Current<#Result);
18     return Result;
19   }];
20 };
    Any的第一個參數(shù)是Parser,第二個參數(shù)是最大循環(huán)次數(shù),-1代表無限循環(huán)。這樣的話,Any(Ch("1"),4)就可以接受""、"1"、"11"、"111"和"1111"了。如果4改為-1的話,那么多少個1都行了。如果要限制最少循環(huán)次數(shù)怎么辦呢?嘿嘿,用Seq(X,X,X,Any(X,-1))吧,最少就3次了。如果你嫌麻煩的話可以再開發(fā)一個函數(shù)去簡化這個過程。在這里我就不詳細(xì)討論了。

    為了方便,我們讓Rep(X)=Any(X,-1),Opt(X)=Any(X,1):
1 Opt=func(Parser)
2 {
3   return Any(Parser,1);
4 };
5 
6 Rep=func(Parser)
7 {
8   return Any(Parser,-1);
9 };

    最后我們需要一個Using:
 1 Using=func(Parser,Handler)
 2 {
 3   return [func(Input)
 4   {
 5     local Result=Parser[0](Input);
 6     for(r in Result)
 7       r[0]=Handler(r[0]);
 8     return Result;
 9   }];
10 };
    Using很好理解的,給他一個Parser和一個函數(shù)Handler,當(dāng)Parser完成以后會把結(jié)果送給Handler進(jìn)行轉(zhuǎn)換(譬如進(jìn)行四則運(yùn)算的求值),然后把Handler函數(shù)的執(zhí)行結(jié)果當(dāng)成Parser的分析結(jié)果返回。

    說到這里如果還不是很清楚的話,有兩個辦法。
    1:自己使用JavaScript或者其他語言重寫一次
    2:閱讀文章開頭的論文(有鏈接)
    好了,現(xiàn)在我們展示一下如何使用這些函數(shù)來對一個四則運(yùn)算式子進(jìn)行求值。

    重新提一下四則運(yùn)算的文法:
    Term=<number> | "(" Exp ")"
    Factor=[ Factor ("*" | "/") ] Term
    Exp=[Exp ("+" | "-") ] Factor
    我們這個語法分析器跟boost::spirit一樣,不支持左遞歸哈。所以我們手動修改一下文法:
    Term=<number> | "(" Exp ")"
    Factor=Term ( ("*" | "/") Term )*
    Exp=Factor ( ("+" | "-") Factor )*

    好了,有了這個文法,我們用代碼把它們表達(dá)出來:
 1 CreateParser=func()
 2 {
 3   return [Fail];
 4 };
 5 
 6 SetParser=func(Object,Parser)
 7 {
 8   Object[0]=Parser[0];
 9 };
10 
11 Pass=func(Index)
12 {
13   return func(Params)
14   {
15     return Params[Index];
16   };
17 };
18 
19 Calculator=func(Params)
20 {
21   local Result=Params[0];
22   for(pair in Params[1])
23     if(pair[0]=="+")
24       Result=Result+pair[1];
25     else if(pair[0]=="-")
26       Result=Result-pair[1];
27     else if(pair[0]=="*")
28       Result=Result*pair[1];
29     else if(pair[0]=="/")
30       Result=Result/pair[1];
31     else
32       throw("Unknown operator:"++pair[0]);
33   return Result;
34 };
35 
36 Term=CreateParser();
37 Factor=CreateParser();
38 Exp=CreateParser();
39 
40 SetParser(Term,Alt(Regex("\\d+(.\\d+)?"),Using(Seq(Ch("("),Exp,Ch(")")),Pass(1))));
41 SetParser(Factor,Using(Seq(Term,Rep(Seq(Alt(Ch("*"),Ch("/")),Term))),Calculator));
42 SetParser(Exp,Using(Seq(Factor,Rep(Seq(Alt(Ch("+"),Ch("-")),Factor))),Calculator));
43 Parser=Exp;
    ·Pass是的作用是分析道["(" , Exp , ")"]的時候把Exp返回,因?yàn)槔ㄌ柺遣恍枰摹?br>    ·Calculator的作用是傳入[1 [ ["+" , 2] , ["-" , 3] , ["+" , 4] ]]的時候吧表達(dá)式當(dāng)成1+2+3+4進(jìn)行計算。
    ·Term、Factor和Exp都是用了Using來處理返回的結(jié)果。這樣的話就等于將語義綁定到語法上。

    好了,讓我們看一看運(yùn)行結(jié)果吧。以下是主程序:
1 for(r in Parser[0](read("Input:")))
2 {
3   write("================================================================================");
4   writeln("RESULT :",r[0]);
5   writeln("REMAIN :",r[1]);
6 }

    輸入:(1+2)+(3+4),屏幕上會出現(xiàn):
 1 Input:(1+2)*(3+4)
 2 ================================================================================
 3 RESULT :
 4 REMAIN :(1+2)*(3+4)
 5 ================================================================================
 6 RESULT :3
 7 REMAIN :*(3+4)
 8 ================================================================================
 9 RESULT :21
10 REMAIN :
11 

    萬歲!
posted on 2008-05-21 00:57 陳梓瀚(vczh) 閱讀(8192) 評論(5)  編輯 收藏 引用 所屬分類: 腳本技術(shù)

評論:
# re: 使用高階函數(shù)開發(fā)語法分析器 2008-05-21 03:19 | haskell
強(qiáng)大  回復(fù)  更多評論
  
# re: 使用高階函數(shù)開發(fā)語法分析器 2008-05-21 04:28 | 空明流轉(zhuǎn)
喂,vc,你看我樓上的人叫啥名字。  回復(fù)  更多評論
  
# re: 使用高階函數(shù)開發(fā)語法分析器 2008-05-21 05:07 | 陳梓瀚(vczh)
不就haskell么……  回復(fù)  更多評論
  
# re: 使用高階函數(shù)開發(fā)語法分析器 2008-05-21 16:49 | 空明流轉(zhuǎn)
@陳梓瀚(vczh)
切,你敢叫OCaml啊...  回復(fù)  更多評論
  
# re: 使用高階函數(shù)開發(fā)語法分析器 2016-03-10 19:26 | aaaron7
感覺思想和 monadic parser combinator 那篇論文中很像,只是那篇論文里用一個 State monad 來實(shí)現(xiàn)了這個隊列  回復(fù)  更多評論
  
青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <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>
            最近中文字幕日韩精品| 中文在线一区| 欧美乱人伦中文字幕在线| **欧美日韩vr在线| 国产婷婷色一区二区三区在线 | 国产精品稀缺呦系列在线| 欧美成人在线免费视频| 欧美日韩和欧美的一区二区| 欧美日韩免费一区二区三区视频 | 欧美精品www在线观看| 欧美激情视频一区二区三区免费 | 亚洲欧美日韩视频二区| 欧美亚洲网站| 久久理论片午夜琪琪电影网| 欧美 日韩 国产一区二区在线视频| 亚洲高清一二三区| 亚洲国产另类 国产精品国产免费| 亚洲国产精品成人综合| 日韩视频一区二区在线观看| 亚洲欧美成aⅴ人在线观看| 久久在线免费观看视频| 欧美三级视频在线| 国产主播一区二区三区四区| 日韩亚洲视频| 久久成人国产| 亚洲人成7777| 久久久噜噜噜久久中文字幕色伊伊 | 国产精品久久久久久久久免费| 国内成+人亚洲| 亚洲一区二区三区乱码aⅴ| 噜噜噜在线观看免费视频日韩| 亚洲伦理在线免费看| 久久激情久久| 国产精品丝袜xxxxxxx| 亚洲日本中文字幕| 久久久成人网| 亚洲伊人久久综合| 欧美三级日本三级少妇99| 亚洲国产人成综合网站| 久久精品人人做人人综合| 一区二区三区视频在线观看| 欧美韩日一区二区| 亚洲国产精品久久人人爱蜜臀 | 亚洲精品久久久久久久久久久久久| 欧美在线你懂的| 亚洲电影av在线| 亚洲欧洲一二三| 久久久视频精品| 亚洲欧美视频一区| 国产精品a久久久久| 亚洲伦理在线观看| 亚洲成色999久久网站| 久久久久久久久久久一区| 国产欧美精品一区二区色综合 | 亚洲无线视频| 亚洲精品久久在线| 欧美国产精品v| 亚洲精品美女免费| 亚洲国产精品久久久| 久久综合网络一区二区| 亚洲东热激情| 欧美激情欧美激情在线五月| 欧美亚洲综合久久| 亚洲日本va午夜在线电影| 美玉足脚交一区二区三区图片| 好男人免费精品视频| 久久精品国产亚洲精品| 性欧美1819sex性高清| 国产精品香蕉在线观看| 欧美一区二区私人影院日本| 香蕉免费一区二区三区在线观看| 国产视频在线观看一区| 久久亚洲国产成人| 麻豆精品一区二区av白丝在线| 亚洲黑丝在线| 亚洲黄色av一区| 欧美午夜精品电影| 久久国产精品久久精品国产| 亚洲欧美bt| 亚洲国产精品久久久久婷婷884| 亚洲国产成人av在线| 欧美日韩高清在线一区| 午夜精品亚洲一区二区三区嫩草| 欧美一区二区三区四区夜夜大片 | 欧美一区二区日韩| 精品99一区二区| 亚洲国产一区二区a毛片| 欧美性猛交99久久久久99按摩| 欧美一级淫片aaaaaaa视频| 欧美一区二区三区四区在线| 亚洲人成在线影院| 亚洲一区免费看| 在线观看亚洲a| 夜夜精品视频一区二区| 韩国av一区二区| 亚洲乱码精品一二三四区日韩在线 | 亚洲永久在线| 久久精品视频在线免费观看| 麻豆成人综合网| 亚洲精品之草原avav久久| 一区二区欧美亚洲| 极品中文字幕一区| 一区二区三区国产| 伊人狠狠色丁香综合尤物| 99国产成+人+综合+亚洲欧美| 国产视频自拍一区| 日韩亚洲国产欧美| 伊人成人在线视频| 亚洲欧美在线高清| 99精品热视频| 久久精品国产精品亚洲精品| 亚洲一级在线| 欧美激情a∨在线视频播放| 久久精品99国产精品酒店日本| 欧美激情精品久久久| 久久久一区二区| 国产精品一区免费观看| 亚洲欧洲视频在线| 在线播放精品| 欧美一级免费视频| 亚洲视频在线播放| 欧美成人资源| 欧美+亚洲+精品+三区| 国产伦精品一区二区三区视频黑人| 91久久精品国产91久久| 影音先锋久久精品| 欧美主播一区二区三区| 欧美一区二区三区视频免费播放| 欧美日韩美女| 亚洲精品国产系列| 日韩一级大片在线| 欧美成人久久| 欧美韩日一区二区三区| 亚洲国语精品自产拍在线观看| 久久久久久一区| 久久久在线视频| 国内成人在线| 欧美在线观看网址综合| 久久精视频免费在线久久完整在线看| 国产精品每日更新在线播放网址| 一区二区三区日韩精品| 午夜精品三级视频福利| 国产女优一区| 久久国产一区二区三区| 另类天堂av| 亚洲精品极品| 欧美日韩一区二区三区高清| 日韩午夜剧场| 亚洲影音先锋| 国产欧美日韩一区二区三区| 亚洲专区一区二区三区| 久久成年人视频| 在线欧美电影| 欧美女人交a| 亚洲在线国产日韩欧美| 久久久91精品| 最新国产乱人伦偷精品免费网站| 欧美成人国产| 亚洲欧美日韩精品一区二区| 老司机精品久久| 9l视频自拍蝌蚪9l视频成人| 国产精品久久久久aaaa| 午夜精品久久久久| 狼人社综合社区| 宅男精品视频| 国产日韩欧美一二三区| 国产欧美精品一区二区三区介绍| 欧美日韩激情小视频| 亚洲欧洲一区二区三区久久| 亚洲中午字幕| 狠狠入ady亚洲精品| 欧美精品日韩一区| 午夜日韩电影| 亚洲欧洲综合| 久久久久亚洲综合| 一区二区三区欧美在线| 国产主播喷水一区二区| 欧美日韩午夜剧场| 久久久午夜电影| 中文一区在线| 亚洲第一久久影院| 欧美在线你懂的| 正在播放亚洲一区| 在线电影国产精品| 国产精品国产一区二区 | 欧美日韩视频一区二区三区| 性感少妇一区| 99视频精品在线| 欧美电影在线播放| 久久都是精品| 亚洲在线免费观看| 日韩亚洲欧美一区| 亚洲国产精品va在看黑人| 国产女人水真多18毛片18精品视频| 欧美国产日韩一区二区在线观看| 欧美在线不卡| 先锋a资源在线看亚洲| 一本一本久久| 99爱精品视频| 夜夜爽www精品|