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

隨筆-91  評論-137  文章-0  trackbacks-0
上一篇,首先需要修正的是在DFA生成算法中的傳播部分,應該需要有個循環一直傳播到不能傳播為止,在多次實驗中表明,有些展望符是通過第2,3,4甚至更多次傳播得來的。

應此,相應的make函數變成了
    bool LALR1::make()
    {
        vector<LALR1Production> v;
        v.push_back(inputProductions[begin][0]);
        pStart = closure(v);
        pStart->idx = Item::inc();
        context.states.insert(pStart);
        items.push_back(pStart);

        queue<Item*> q;
        q.push(pStart);

        vector<Item*> changes;

        bool bContinue = false;
        while (!q.empty())
        {
            Item* pItem = q.front();
            vector<Production::Item> s;
            symbols(pItem, s);
            select_into(s, vts, compare_production_item_is_vt, push_back_unique_vector<Production::Item>);
            select_into(s, vns, compare_production_item_is_vn, push_back_unique_vector<Production::Item>);
            for (vector<Production::Item>::const_iterator i = s.begin(), m = s.end(); i != m; ++i)
            {
                Item* pNewItem = NULL;
                if (go(pItem, *i, pNewItem))
                {
                    long n = itemIndex(pNewItem);
                    if (n == -1)
                    {
                        pNewItem->idx = Item::inc();
                        q.push(pNewItem);
                        items.push_back(pNewItem);
                        context.states.insert(pNewItem);
                    }
                    else
                    {
                        items[n]->mergeWildCards(pNewItem, bContinue);
                        changes.push_back_unique(items[n]);
                        destruct(pNewItem, has_destruct(*pNewItem));
                        Item_Alloc::deallocate(pNewItem);
                    }
                    edges[pItem].push_back_unique(Edge(pItem, n == -1 ? pNewItem : items[n], *i));
                }
            }
            q.pop();
        }
        while (bContinue)
        {
            vector<Item*> v;
            v.reserve(changes.size());
            bContinue = false;
            for (vector<Item*>::const_iterator i = changes.begin(), m = changes.end(); i != m; ++i)
            {
                vector<Production::Item> s;
                symbols(*i, s);
                for (vector<Production::Item>::const_iterator j = s.begin(), n = s.end(); j != n; ++j)
                {
                    Item* pNewItem = NULL;
                    if (go(*i, *j, pNewItem))
                    {
                        long n = itemIndex(pNewItem);
                        if (n == -1) throw error<const char*>("unknown item", __FILE__, __LINE__);
                        else
                        {
                            items[n]->mergeWildCards(pNewItem, bContinue);
                            v.push_back_unique(items[n]);
                            destruct(pNewItem, has_destruct(*pNewItem));
                            Item_Alloc::deallocate(pNewItem);
                        }
                    }
                }
            }
            changes = v;
        }
    }
在merge函數中,會檢測有沒有新生成的展望符來決定是否繼續傳播下去。

一個示例
下面我們用一個例子來說明LALR1 DFA是如何生成的,首先它的文法如下
S -> L "=" R
  | R "+"
  | R
  ;

L -> "*" R
  |  "id"
  ;

R -> L
  ;
根據之前的算法,我們先來看自生的部分

首先我們寫出這個文法的增廣文法
begin -> . S (#)
求取它的閉包得到
begin -> . S
wildCards:

S -> . L "=" R
wildCards:

S -> . R "+"
wildCards:

S -> . R
wildCards:

L -> . "*" R
wildCards:
"=" "+" 
L -> . "id"
wildCards:
"=" "+" 
R -> . L
wildCards:
"+" # 
我們觀察到,其中有5個可轉移的符號,分別為S、L、R、"*"和"id",我們分別用go函數對這5個轉移符號求出新的狀態

首先用符號S求出新狀態
begin -> S
wildCards:
由于這個狀態不在原有列表中,應此它是一個新生成的狀態,我們為它添加一條通過符號S轉移的邊。

接下來用符號L求出新狀態
S -> L . "=" R
wildCards:

R -> L
wildCards:
"+" # 
這個狀態也不在原有列表中,應此它也是一個新生成的狀態,我們為它添加一條通過符號L轉移的邊。

然后用符號R求出新狀態
S -> R . "+"
wildCards:

S -> R
wildCards:
這個狀態也不在原有列表中,應此它也是一個新生成的狀態,我們為它添加一條通過符號R轉移的邊。

然后用符號*求出新的狀態
L -> "*" . R
wildCards:
"=" "+" 
R -> . L
wildCards:
"+" # "=" 
L -> . "*" R
wildCards:
"=" "+" # 
L -> . "id"
wildCards:
"=" "+" # 
同樣的它也不在原有的列表中,我們同樣為其添加一條通過符號*轉移的邊。

然后是符號id的
L -> "id"
wildCards:
"=" "+" 
同樣不在列表中,我們為其添加一條通過符號id轉移的邊。

這樣,從start狀態轉移出來的5條邊就生成好了,下面來看看這5個新生成的狀態又會生成一些什么呢
begin -> S
wildCards:
由第一個狀態可知,它沒有任何的邊。

S -> L . "=" R
wildCards:

R -> L
wildCards:
"+" # 
第二個狀態則有一個=的轉移,它生成了一個新狀態
S -> L "=" . R
wildCards:

R -> . L
wildCards:
"+" # "=" 
L -> . "*" R
wildCards:
"=" "+" # 
L -> . "id"
wildCards:
"=" "+" # 

S -> R . "+"
wildCards:

S -> R
wildCards:
第三個狀態有一個+的轉移,它生成了一個新狀態
S -> R "+"
wildCards:

L -> "*" . R
wildCards:
"=" "+" 
R -> . L
wildCards:
"+" # "=" 
L -> . "*" R
wildCards:
"=" "+" # 
L -> . "id"
wildCards:
"=" "+" # 
第四個狀態有4個轉移,分別為R、L、*和id

1.通過符號R轉移到新狀態
L -> "*" R
wildCards:
"=" "+" 

2.通過符號L轉移到新狀態
R -> L
wildCards:
"+" # "=" 

3.通過*則可轉移到它自己

4.通過id轉移到第5個狀態

第五個狀態則沒有任何的轉移。

S -> L "=" . R
wildCards:

R -> . L
wildCards:
"+" # "=" 
L -> . "*" R
wildCards:
"=" "+" # 
L -> . "id"
wildCards:
"=" "+" # 
第六個狀態有4個轉移,分別為R、L、*和id

1.通過符號R可轉移到新狀態
S -> L "=" R
wildCards:

2.通過符號L可轉移到狀態9

3.通過符號*可轉移到狀態4

4.通過符號id可轉移到狀態5

第6、7、8個狀態都沒有任何轉移

然后讓我們來看下changes列表里有哪些東西,根據上一篇的算法可知,所有已存在的狀態都在changes列表里,應此它里面應該會有4、5和9三個狀態。

至此,整個自生的部分完成了,下面我們將其畫成一張圖


下面是傳播部分
在第一次傳播時changes列表里有3個狀態,分別對這3個狀態用go函數求出新的展望符,并把它們合并到原有的狀態上。

首先看狀態4,它有4個狀態轉移符,分別是R、L、*和id

1.通過符號R可轉移到狀態8,同時它的展望符如下
L -> "*" R
wildCards:
"=" "+" # 

2.通過符號L可轉移到狀態9,同時它的展望符如下
R -> L
wildCards:
"+" # "=" 

3.通過符號*可轉移到它自己,同時它的展望符如下
L -> "*" . R
wildCards:
"=" "+" # 
R -> . L
wildCards:
"+" # "=" 
L -> . "*" R
wildCards:
"=" "+" # 
L -> . "id"
wildCards:
"=" "+" # 

4.通過符號id可轉移到狀態5,同時它的展望符如下
L -> "id"
wildCards:
"=" "+" # 

然后我們來看一下狀態5和9,它們沒有任何狀態轉移符,應此它們不會傳播任何展望符。

現在changes列表里有4個狀態,分別為8、9、4和5,又由于第8個狀態已經產生了新的展望符#應此需要繼續傳播

第二次傳播

首先先看狀態8和9,它們沒有任何狀態轉移符,應此它們不會傳播任何展望符。

然后來看狀態4,同樣的它有4個狀態轉移符,分別為R、L、*和id。

1.通過符號R可轉移到狀態8,同時它的展望符如下
L -> "*" R
wildCards:
"=" "+" # 

2.通過符號L可轉移到狀態9,同時它的展望符如下
R -> L
wildCards:
"+" # "=" 

3.通過符號*可轉移到它自己,同時它的展望符如下
L -> "*" . R
wildCards:
"=" "+" # 
R -> . L
wildCards:
"+" # "=" 
L -> . "*" R
wildCards:
"=" "+" # 
L -> . "id"
wildCards:
"=" "+" # 

4.通過符號id可轉移到狀態5,同時它的展望符如下
L -> "id"
wildCards:
"=" "+" # 

最后我們來看狀態5,它沒有任何狀態轉移符,應此它不會傳播任何展望符。

現在changes列表里同樣有4個狀態,分別為8、9、4和5,由于沒有一個狀態產生了新的展望符,應此它將不會繼續傳播下去了。

現在整個文法的DFA就生成完畢了,讓我們來修改一下原先的那張圖來看看最終的DFA是什么樣的。


整個示例就先介紹到這里,在接下來的一篇文章中將會通過幾個示例來介紹closure和go函數的原理,希望這種由粗到細的講解順序能夠被讀者所接受。最后完整的代碼可到http://code.google.com/p/qlanguage下載。
posted on 2013-05-30 23:04 lwch 閱讀(1580) 評論(2)  編輯 收藏 引用 所屬分類: QLanguage

評論:
# re: QParserGenerator代碼分析二(A fix&An example) 2013-05-30 23:36 | eryar
圖片看不全……  回復  更多評論
  
# re: QParserGenerator代碼分析二(A fix&An example) 2013-05-31 10:12 | lwch
@eryar
這個好像是模板的問題,改了一下圖片的大小。  回復  更多評論
  
青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <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>
            欧美日本韩国| 亚洲精品一区二区在线| 亚洲精品乱码久久久久久日本蜜臀| 国产女人精品视频| 国产视频一区欧美| 国内精品99| 亚洲精品视频在线播放| 亚洲美女视频| 亚洲一区欧美二区| 久久久久久精| 亚洲另类自拍| 欧美亚洲在线播放| 欧美黄免费看| 国产香蕉久久精品综合网| 伊人久久婷婷| 亚洲一级特黄| 美女在线一区二区| 日韩亚洲一区二区| 久久av资源网站| 欧美日韩免费网站| 在线电影一区| 欧美一区二区三区另类| 欧美成人中文字幕| 亚洲中无吗在线| 欧美mv日韩mv亚洲| 国产日韩在线不卡| 亚洲视频中文字幕| 欧美成人在线免费观看| 欧美自拍偷拍| 亚洲欧洲日本mm| 亚洲一区亚洲| 欧美国产综合| 亚洲国产岛国毛片在线| 久久久精品一区| 亚洲美女免费视频| 久久亚洲欧美| 国产在线一区二区三区四区| 在线综合亚洲| 亚洲国产精品久久| 久久久久久夜精品精品免费| 国产精品国产a级| 亚洲精品国产拍免费91在线| 久久久久久久波多野高潮日日| 一区二区三区视频免费在线观看| 欧美xx视频| 亚洲日产国产精品| 亚洲国产日韩美| 久久尤物视频| 永久555www成人免费| 久久国产精彩视频| 午夜性色一区二区三区免费视频| 欧美午夜影院| 亚洲一区二区三区在线看| 亚洲国产精品欧美一二99| 久久影院午夜片一区| 在线看不卡av| 欧美国产另类| 久久综合伊人77777麻豆| 激情五月综合色婷婷一区二区| 久久久亚洲欧洲日产国码αv| 亚洲欧美综合国产精品一区| 国产精品久久午夜夜伦鲁鲁| 性色一区二区| 欧美夜福利tv在线| 国产尤物精品| 欧美国产三级| 欧美日韩国产成人高清视频| 亚洲乱码一区二区| 亚洲乱码国产乱码精品精可以看| 欧美另类久久久品| 亚洲一区欧美激情| 香蕉乱码成人久久天堂爱免费| 国产欧亚日韩视频| 蜜臀av在线播放一区二区三区 | 夜夜嗨av一区二区三区| 欧美日韩视频在线一区二区| 亚洲一区二区三区免费在线观看| 亚洲视频一区二区在线观看 | 日韩午夜三级在线| 一本色道久久88综合亚洲精品ⅰ| 国产精品嫩草99a| 久久综合电影| 欧美福利视频网站| 久久精品视频免费播放| 亚洲一区二区三区中文字幕在线| 国产精品入口日韩视频大尺度| 香蕉视频成人在线观看| 久久国产精品亚洲va麻豆| 亚洲高清不卡在线| 一区二区三区久久| 一区在线影院| 日韩视频久久| 国内不卡一区二区三区| 亚洲日本欧美在线| 国产日韩精品一区二区三区在线| 麻豆成人在线| 欧美午夜免费影院| 另类亚洲自拍| 国产精品久久久久久久久久久久| 久久久久国产精品麻豆ai换脸| 欧美国产日产韩国视频| 欧美专区中文字幕| 欧美日韩亚洲国产精品| 久久久综合激的五月天| 欧美日韩中文在线观看| 欧美91视频| 国产一区二区高清| 亚洲午夜精品17c| 日韩午夜免费| 老鸭窝亚洲一区二区三区| 性欧美长视频| 欧美色精品在线视频| 亚洲盗摄视频| 一区精品久久| 欧美一二三视频| 亚洲欧美资源在线| 欧美精品一区视频| 欧美成人高清视频| 黄色一区二区三区四区| 亚洲永久免费视频| 亚洲欧美精品在线观看| 欧美久久久久久蜜桃| 亚洲第一精品夜夜躁人人躁| 很黄很黄激情成人| 欧美一区二区性| 欧美在线视频在线播放完整版免费观看 | 亚洲欧美三级伦理| 午夜精品一区二区三区在线视| 欧美精品一区二区在线观看| 男女视频一区二区| 精品av久久707| 久久久精品国产免大香伊| 久久精品人人爽| 国产欧美一级| 性色av香蕉一区二区| 久久福利一区| 精品51国产黑色丝袜高跟鞋| 久久免费精品视频| 亚洲国产日本| 亚洲伊人久久综合| 国产精品系列在线播放| 午夜精彩国产免费不卡不顿大片| 欧美一区二区三区四区视频| 国产午夜精品美女视频明星a级| 小嫩嫩精品导航| 欧美女主播在线| 亚洲免费观看高清在线观看 | 亚洲欧洲综合另类在线| 亚洲精品视频免费观看| 欧美人成在线| 亚洲淫性视频| 久久综合中文| 亚洲看片一区| 欧美亚洲成人精品| 午夜精品福利一区二区三区av | 亚洲午夜电影在线观看| 午夜天堂精品久久久久| 黑人操亚洲美女惩罚| 欧美α欧美αv大片| 一本色道婷婷久久欧美| 欧美综合二区| 亚洲国产mv| 国产精品久久久久久妇女6080 | 亚洲欧美另类在线观看| 久久九九久久九九| 亚洲精品久久久久久久久久久久 | 久久人人九九| 亚洲日本一区二区三区| 午夜在线精品| 亚洲人永久免费| 国产精品一级在线| 欧美成人国产| 99国产精品国产精品久久| 久久久国产成人精品| 99精品国产高清一区二区| 国产精品亚洲综合| 欧美黑人在线观看| 久久国产婷婷国产香蕉| 夜夜爽99久久国产综合精品女不卡 | 国产一区二区日韩精品| 欧美国产亚洲精品久久久8v| 午夜精品久久久久久久99樱桃| 欧美第一黄网免费网站| 欧美一区二区在线播放| 一区二区三区四区五区在线| 伊人久久成人| 国产女人水真多18毛片18精品视频| 欧美大色视频| 另类国产ts人妖高潮视频| 亚洲欧美在线高清| 99精品欧美一区| 亚洲二区免费| 久久夜色精品亚洲噜噜国产mv| 亚洲一区二区高清视频| 亚洲国产岛国毛片在线| 国产一区再线| 国产毛片一区二区| 欧美天堂亚洲电影院在线观看| 美女黄色成人网| 久久久久九九九九|