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

隨筆-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 閱讀(1590) 評論(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>
            欧美日韩精品一区二区| 久久久久国产精品一区二区| 欧美日韩另类字幕中文| 欧美精品一区二区视频| 欧美91大片| 欧美乱大交xxxxx| 欧美精品久久久久久久久老牛影院| 欧美激情在线有限公司| 国产精品久久999| 国产精品一级| 亚洲国产精品ⅴa在线观看| 亚洲精品日韩精品| 亚洲一区二区三区在线| 欧美在线视频一区二区三区| 久久精品视频在线看| 欧美aa在线视频| 亚洲精品免费网站| 亚洲视频在线观看网站| 欧美一区久久| 欧美金8天国| 国产女人水真多18毛片18精品视频| 国产综合网站| 亚洲一区二区三区久久| 欧美福利视频一区| 久久视频在线看| 99国内精品久久| 久久精品观看| 亚洲欧洲一区二区在线播放| 亚洲精选中文字幕| 久久久久国产免费免费| 国产精品白丝黑袜喷水久久久| 国内精品嫩模av私拍在线观看| 亚洲精品久久久久久久久久久久| 久久成人一区| 在线亚洲免费视频| 欧美日韩123| 91久久精品国产91性色| 午夜亚洲精品| 亚洲精品久久久久久一区二区| 久久精品一区中文字幕| 国产精品久久久久久久久果冻传媒 | 亚洲乱码精品一二三四区日韩在线| 亚洲性线免费观看视频成熟| 欧美aa在线视频| 国产亚洲欧美aaaa| 亚洲欧美怡红院| 亚洲精品自在在线观看| 蜜臀av一级做a爰片久久| 国产日韩欧美另类| 午夜视频在线观看一区二区三区| 亚洲精品一区二区三区樱花 | 久久国产精品毛片| 99视频精品| 欧美日韩国产在线播放网站| 亚洲国产专区| 欧美高潮视频| 男人的天堂亚洲| 亚洲精品国产欧美| 最新日韩中文字幕| 欧美欧美在线| 亚洲一区二区三区四区五区午夜| 亚洲精品国久久99热| 欧美精品日韩一本| 日韩视频一区二区| 一级成人国产| 欧美色区777第一页| 亚洲男人的天堂在线观看| 亚洲一区二区在线视频| 国产精品手机在线| 久久久中精品2020中文| 久久久中精品2020中文| 亚洲人成在线免费观看| 91久久精品美女高潮| 欧美日韩国产一区| 午夜视频一区| 久久免费一区| 一区二区三欧美| 一区二区三区国产盗摄| 欧美综合国产| 另类国产ts人妖高潮视频| 亚洲电影专区| 99精品国产高清一区二区| 国产精品色网| 欧美激情视频一区二区三区不卡| 欧美黄在线观看| 亚洲欧美综合一区| 久久精品国产久精国产爱| 亚洲动漫精品| 亚洲午夜女主播在线直播| 国产一级一区二区| 欧美国产日韩一区| 国产精品亚洲片夜色在线| 久热精品视频在线| 欧美另类videos死尸| 久久国产精品一区二区三区四区| 美女精品自拍一二三四| 亚洲女人小视频在线观看| 久久久欧美一区二区| 亚洲欧美亚洲| 欧美久久电影| 久久免费国产精品| 欧美网站在线观看| 欧美bbbxxxxx| 国产乱人伦精品一区二区| 亚洲激情网站免费观看| 狠狠v欧美v日韩v亚洲ⅴ| 这里只有精品视频| 亚洲巨乳在线| 久久蜜臀精品av| 久久天天狠狠| 国产精品亚洲综合色区韩国| 最新成人av网站| 在线日韩欧美| 校园春色国产精品| 西瓜成人精品人成网站| 欧美视频亚洲视频| 亚洲麻豆视频| 日韩视频免费观看高清在线视频| 久久久久se| 久久亚洲欧美国产精品乐播| 国产精品美女久久久久av超清| 亚洲欧洲一区二区天堂久久 | 亚洲精品美女久久7777777| 揄拍成人国产精品视频| 欧美一级在线视频| 久久久国产精品一区| 国产一二三精品| 久久se精品一区二区| 久久福利视频导航| 国产精品一二一区| 亚洲一区国产一区| 午夜精品久久久| 国产精品亚发布| 亚洲欧美在线一区| 久久精品国语| 精品99一区二区| 久久久水蜜桃| 亚洲高清不卡| 亚洲久久一区| 欧美日韩一区三区四区| 欧美日韩国产探花| 亚洲欧美视频在线观看视频| 欧美日韩在线高清| 一本一本久久a久久精品综合麻豆| 艳妇臀荡乳欲伦亚洲一区| 欧美理论电影在线播放| 一区二区三区色| 久久精品综合一区| 亚洲第一网站免费视频| 欧美高清在线精品一区| 一区二区三区免费观看| 午夜精品理论片| 国内视频一区| 欧美成人黄色小视频| 日韩午夜av| 欧美在线免费看| 韩国精品久久久999| 男人的天堂亚洲| 亚洲一区二区免费视频| 久久久久这里只有精品| 亚洲电影免费在线| 欧美日韩影院| 欧美专区福利在线| 亚洲国产专区| 午夜精品久久久久久久久久久久 | 久久米奇亚洲| 日韩西西人体444www| 亚洲欧美激情四射在线日 | 欧美激情视频一区二区三区在线播放 | 久久亚洲春色中文字幕久久久| 亚洲第一精品夜夜躁人人爽 | 亚洲精品一区二区三区樱花| 国产精品久久久久999| 亚洲天堂成人在线视频| 国产精品v片在线观看不卡| 久久精品人人做人人爽| 亚洲日韩视频| 久久国产精品亚洲77777| 亚洲美女视频在线观看| 国产一区91精品张津瑜| 欧美精品1区2区| 欧美亚洲一区二区三区| 亚洲精品一区二区三区福利| 久久人人爽人人爽| 亚洲综合色自拍一区| 最新日韩精品| 国产一区二区| 国产精品久久久91| 欧美精品二区三区四区免费看视频| 中文av一区二区| 亚洲精品自在久久| 亚洲国产精品ⅴa在线观看| 久久婷婷综合激情| 欧美一区二区精品久久911| 99国产精品99久久久久久| 在线观看视频一区二区欧美日韩| 国产精品免费一区二区三区观看| 欧美日韩大片一区二区三区| 免费成人高清在线视频| 可以免费看不卡的av网站|