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

隨筆-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| 午夜精品福利在线| 亚洲欧美日韩精品久久奇米色影视 | 久久久久久久激情视频| 久久久精品日韩欧美| 久久国产欧美日韩精品| 老牛国产精品一区的观看方式| 久久免费视频在线| 欧美福利视频网站| 欧美天天视频| 国产一区二区高清视频| 亚洲第一精品夜夜躁人人躁| 日韩视频不卡中文| 欧美在线不卡视频| 欧美高清视频在线播放| 日韩视频免费观看| 美女图片一区二区| 久久精品网址| 欧美电影免费观看大全| 亚洲精品乱码| 中文久久精品| 久久久久久97三级| 欧美日韩 国产精品| 国产亚洲二区| 亚洲综合精品四区| 老司机免费视频一区二区| 亚洲精品一区二区三区蜜桃久| 亚洲精品中文字幕在线| 久久深夜福利| 国产亚洲一区二区三区在线播放| 一区二区不卡在线视频 午夜欧美不卡'| 亚洲一区二区三区涩| 欧美承认网站| 理论片一区二区在线| 国内精品伊人久久久久av一坑| 亚洲一区二区伦理| 欧美高清视频一二三区| 六月丁香综合| 亚洲精品影视在线观看| 欧美激情第8页| 久久综合九色| 亚洲精品视频免费在线观看| 免费成人美女女| 美女任你摸久久| 亚洲激情网址| 99国产精品久久久久久久久久| 欧美日韩免费网站| 国产精品99久久99久久久二8| 亚洲欧洲一区二区天堂久久| 久久天天躁狠狠躁夜夜av| 国产精品99久久不卡二区| 欧美国产一区二区三区激情无套| 最新成人av在线| 亚洲免费视频网站| 亚洲国内精品| 亚洲欧美高清| 最新亚洲激情| 欧美一区二区三区啪啪| 在线观看亚洲精品视频| 亚洲精品自在在线观看| 国产一区二区日韩精品| 99国产精品自拍| 91久久精品一区| 久久精品中文字幕一区| 亚洲欧美日韩一区在线| 欧美成人精品在线观看| 欧美一区二区在线看| 欧美日本在线一区| 蜜臀久久99精品久久久久久9 | 欧美成人精品一区二区三区| 亚洲专区在线视频| 欧美精品一区在线发布| 久久露脸国产精品| 欧美午夜欧美| 中文av一区二区| 亚洲视频每日更新| 欧美日韩1080p| 亚洲欧洲精品一区二区三区| 黑丝一区二区| 久久婷婷国产综合精品青草| 久热精品视频在线| 韩日欧美一区二区| 久久最新视频| 在线观看日韩国产| 玖玖视频精品| 亚洲精品社区| 欧美一区激情| 亚洲国产成人久久综合一区| 久久在线视频在线| 亚洲精选大片| 久久精品九九| 亚洲精品123区| 亚洲日本中文字幕| 日韩午夜电影| 亚洲电影av| 国产精品日韩欧美| 久久成人精品无人区| 久久婷婷色综合| 正在播放欧美视频| 国产综合自拍| 国产精品电影在线观看| 久久久久成人精品| 夜夜嗨av一区二区三区四区| 久久女同互慰一区二区三区| 99国产精品国产精品久久| 欧美吻胸吃奶大尺度电影| 久久嫩草精品久久久精品| 一区二区三区久久久| 最新精品在线| 久久久久欧美精品| 欧美一区二区三区的| 亚洲一区二区三区中文字幕 | 国产亚洲精品bv在线观看| 男人插女人欧美| 久久亚洲春色中文字幕| 欧美专区福利在线| 欧美一区二区成人6969| 香蕉免费一区二区三区在线观看| 99国产精品99久久久久久| 国产偷久久久精品专区| 欧美日韩精品久久久| 欧美日韩大陆在线| 欧美三日本三级少妇三2023| 欧美黑人多人双交| 欧美视频免费在线观看| 国产精品视频一二| 国产女精品视频网站免费 | 欧美99久久| 欧美精品日韩一本| 国产精品高潮粉嫩av| 国产一区二区久久| 亚洲韩日在线| 销魂美女一区二区三区视频在线| 久久久久久国产精品mv| 欧美韩国一区| 午夜激情亚洲| 欧美人成免费网站| 狠狠综合久久av一区二区小说| 亚洲激情黄色| 美玉足脚交一区二区三区图片| 亚洲精品视频啊美女在线直播| 亚洲免费在线视频| 欧美午夜免费| aa级大片欧美三级| 欧美激情无毛| 免费在线观看精品| 国产亚洲欧美aaaa| 欧美一级一区| 中文久久精品| 国产精品sm| 午夜精品亚洲| 一本色道久久综合亚洲91| 欧美欧美在线| 在线一区二区三区做爰视频网站| 欧美 日韩 国产在线| 久久精精品视频| 在线观看欧美精品| 免费人成网站在线观看欧美高清| 欧美影院一区| 亚洲激情视频在线观看| 黄色精品网站| 亚洲精品社区| 亚洲女女女同性video| 国产一区欧美日韩| 亚洲欧美日韩精品久久| 亚洲色无码播放| 国产欧美91| 亚洲高清视频中文字幕| 欧美日韩国产综合视频在线| 99精品热6080yy久久| 在线视频精品一| 国内精品久久久久久久影视蜜臀 | 欧美黄色一区| 国产精品久久久久久久久久久久久 | 亚洲综合视频一区| 亚洲第一主播视频| 在线视频你懂得一区| 伊人夜夜躁av伊人久久| 99国产精品| 日韩特黄影片| 免费视频亚洲| 免费看黄裸体一级大秀欧美| 国产精品久久久久久妇女6080| 欧美大香线蕉线伊人久久国产精品| 欧美亚洲第一页| 99re这里只有精品6| 国内伊人久久久久久网站视频 | 亚洲在线视频观看| 亚洲午夜羞羞片| 欧美日韩一区二区三区四区在线观看 | 香蕉久久a毛片| 欧美日韩国产一级片| 国产乱人伦精品一区二区| 亚洲免费电影在线| 99精品视频免费观看视频| 免费成人网www| 久久在线播放| 日韩一区二区免费高清|