版權(quán)說明:本文檔由用戶提供并上傳,收益歸屬內(nèi)容提供方,若內(nèi)容存在侵權(quán),請進(jìn)行舉報或認(rèn)領(lǐng)
文檔簡介
1、<p><b> 1 緒論</b></p><p> 編譯技術(shù)是理論與實踐并重的課程,在大學(xué)本科計算機(jī)科學(xué)與技術(shù)系的學(xué)生的培養(yǎng)教育中有非常重要的地位。而其實驗課程要綜合運用一年級、二年級和三年級所學(xué)的多門課程的內(nèi)容和知識,用來完成一個小型編譯程序的設(shè)計和實現(xiàn)。從而鞏固和加強(qiáng)對詞法分析、語法分析、語義分析、代碼生成和報錯處理等理論的認(rèn)識和理解;培養(yǎng)在校大學(xué)生對一個完整的編譯系統(tǒng)
2、的獨立分析和設(shè)計開發(fā)的能力,進(jìn)一步的培養(yǎng)當(dāng)代大學(xué)生動手實踐能力以及獨立自主的開發(fā)設(shè)計能力。 </p><p> 完整的編譯器設(shè)計開發(fā)和實現(xiàn)是一個非常復(fù)雜的過程,為了簡潔明了的說明編譯器的基本原理,本程序采用了自定義一種類高級語言的高級語言的方法,先定義了該類高級語言的語法規(guī)則,然后再設(shè)計與實現(xiàn)了該語言的編譯器前端,并且分別從詞法分析,語法分析和語義分析等三個方面進(jìn)行了詳細(xì)的開發(fā)和設(shè)計。希望可以讓學(xué)生可以更加清楚
3、直觀的認(rèn)識和理解編譯原理的各個方面的知識。</p><p> 從以往的日常教學(xué)經(jīng)驗來看,我們大學(xué)課程中接觸和學(xué)習(xí)的最多的就是C語言和C++語言。C語言是計算機(jī)科學(xué)與技術(shù)等信息類專業(yè)的基礎(chǔ)必修課,大部分同學(xué)都掌握的比較扎實,所以,本程序所寫的詞法分析器,語法分析器,語義分析器大部分所用的語言都是C語言,C語言是面向過程的結(jié)構(gòu)化語言,程序設(shè)計結(jié)構(gòu)清晰,邏輯嚴(yán)謹(jǐn),能讓讓學(xué)生很好的理解和接受。C++語言也是計算機(jī)科學(xué)與
4、技術(shù)等專業(yè)的基礎(chǔ)課程,是C語言在面對對象思想方面的改進(jìn)版,它繼承了C語言的結(jié)構(gòu)化設(shè)計思路,邏輯層次感強(qiáng),同時又具備了面向?qū)ο笤O(shè)計的能力,也是一門非常優(yōu)秀的語言,很多學(xué)生對這門語言很感興趣,所以本程序也設(shè)計了幾個用C++語言編寫的詞法分析器,語法分析器,語義分析器,以備那些在C++語言方面熟練掌握的學(xué)生學(xué)習(xí)和借鑒。</p><p><b> 1.1 選題背景</b></p>
5、<p> 21世紀(jì)是飛速發(fā)展的高科技時代。在日常的生活和工作中各行各業(yè)都離不開計算機(jī)技術(shù)的運用。學(xué)好計算機(jī)技術(shù)可以讓你在以后的工作和生活中對電腦的使用得心應(yīng)手。所以計算機(jī)科學(xué)與技術(shù)成為了在大學(xué)里非常熱門的專業(yè)。但是大部分學(xué)生往往注重高級語言和軟件設(shè)計運用方面的學(xué)習(xí),對計算機(jī)的底層運用卻不是特別在意,往往不求甚解,馬馬虎虎。其實這些都是不好的學(xué)習(xí)習(xí)慣,像計算機(jī)原理,操作系統(tǒng)和編譯原理都是非常重要的課程,學(xué)好并且掌握它們能讓你在
6、遇到計算機(jī)硬件方面的問題的時候能夠輕松解決,了解計算機(jī)的底層運行原理更可以幫助我們在設(shè)計和開發(fā)軟件的時候避免一些很棘手的問題,使開發(fā)出來的軟件更健壯,運行效率更高。所以,為了增強(qiáng)大家對編譯原理課程的學(xué)習(xí)理解和動手實踐能力,本課題整理和設(shè)計了一些和編譯器有關(guān)的程序,方便大家學(xué)習(xí)和參考。</p><p><b> 1.2 選題意義</b></p><p> 目前世界
7、上的發(fā)達(dá)國家普遍都非常重視發(fā)展以計算機(jī)科學(xué)和通信技術(shù)為核心的信息技術(shù)、信息產(chǎn)業(yè)和信息技術(shù)的應(yīng)用,大部分經(jīng)濟(jì)比較發(fā)達(dá)的國家的信息產(chǎn)業(yè)的發(fā)展都非常迅速。當(dāng)前,我國正處于國民經(jīng)濟(jì)的高速發(fā)展時期。與此相伴隨著的,必然有信息技術(shù)應(yīng)用方面的高速發(fā)展,各行各業(yè)都將面臨著信息技術(shù)研究與發(fā)展的大課題以及信息化技術(shù)改造的大任務(wù)與大工程。所以,很多高校都非常重視學(xué)生在計算機(jī)科學(xué)技術(shù)方面的學(xué)習(xí)。信息技術(shù)基礎(chǔ)更是成為了大部分學(xué)校的公共基礎(chǔ)課。</p>
8、<p> 當(dāng)然,編譯原理作為一門比較復(fù)雜的計算機(jī)技術(shù),對非計算機(jī)專業(yè)的學(xué)生要求不是那么高,但是計算機(jī)科學(xué)與技術(shù)專業(yè)的學(xué)生就應(yīng)該好好學(xué)習(xí)和掌握。要求學(xué)生不但要學(xué)習(xí)課堂上的理論知識,也要加強(qiáng)動手實踐能力,所以,編譯原理的課程設(shè)計和實踐就顯得非常重要了,但是目前來說,很多學(xué)生的動手實踐能力很差,實驗課上設(shè)計的程序往往會遇見各種各樣的問題,比如程序的語法總是有問題,或者邏輯不嚴(yán)謹(jǐn),又或者經(jīng)常報錯,結(jié)果不符合設(shè)計要求等等。<
9、/p><p> 在這個教學(xué)實踐背景下,本課題決定整理和設(shè)計一些編譯原理實踐課程中常用的程序,方便學(xué)生更好的學(xué)習(xí)和理解編譯原理這門課程,同時提高他們的動手實踐的能力,能讓他們在自己設(shè)計編譯程序的時候少犯錯誤,少走彎路,更好的完成學(xué)校既定的教學(xué)任務(wù)和目標(biāo)。希望同學(xué)們能把本課題設(shè)計的程序當(dāng)做一個參考,更好的掌握編譯原理這門課的精髓,更好的提高自己。</p><p><b> 1.3
10、資料來源</b></p><p> 在設(shè)計過程之中曾多次對編譯原理的課程設(shè)計和編譯器的設(shè)計實現(xiàn)收集資料,在互聯(lián)網(wǎng)上和圖書館中查找編譯原理的相關(guān)信息,力求使本課題設(shè)計的程序能夠充分的貼近現(xiàn)實,滿足同學(xué)們的實踐需求。由于在收集資料的時候很多資料的內(nèi)容互相重復(fù),有資料中的程序也有互相借鑒,所以該課題所編寫的程序代碼可能有的也會互相類似,但是這應(yīng)該不會影響整個程序的可讀性以及準(zhǔn)確性。整個程序設(shè)計中的很多東西
11、都有參考教科書,所以,教科書也是一個很重要的資料來源。 </p><p><b> 1.4 程序特點</b></p><p> 本課題做的這個套程序主要針對的是編譯原理的實踐教學(xué)和程序設(shè)計方面,所以各個程序之間相對比較獨立,功能比較單一,主要就是三個功能,詞法分析,語法分析以及語義分析。最后統(tǒng)一在VS2010下做了一個簡單的界面,能把整個編譯過程演示一下。&l
12、t;/p><p> 本程序的主要特點為:</p><p><b> a)界面簡單</b></p><p> 本程序在開發(fā)的過程中,為了讓學(xué)生更直觀的學(xué)習(xí)和了解編譯原理內(nèi)部知識和編譯器的結(jié)構(gòu),使操作界面簡潔、統(tǒng)一,易學(xué)易用。采用人機(jī)對話方式,交互性強(qiáng)。</p><p><b> b)操作簡單</b>
13、;</p><p> 本系統(tǒng)盡量使用DOS操作界面,風(fēng)格一致,用戶只需熟悉DOS界面的基本操作,就能基本學(xué)會本程序的使用。在數(shù)據(jù)輸入過程中,本課題盡可能多地采用數(shù)據(jù)輸入確認(rèn),減少數(shù)據(jù)輸入錯誤,將鍵盤的錄入錯誤量減至最少。最后統(tǒng)一的編譯器程序界面也非常簡單,只有兩個按鈕,一個編譯一個運行,簡單實用。</p><p><b> c)注釋很詳細(xì)</b></p>
14、;<p> 程序內(nèi)部源代碼的注釋非常詳細(xì),可以讓學(xué)生簡潔直觀的判斷每一行代碼的功能以及作用,方便教學(xué)。</p><p><b> 2 開發(fā)環(huán)境</b></p><p> 2.1 C語言簡介</p><p> C語言是由20世紀(jì)70年代中期UNIX操作系統(tǒng)的發(fā)明者Dennis Ritchie在1970年于B語言的基礎(chǔ)上設(shè)
15、計和發(fā)明的。在1972年年初,Thompson等人在早期簡易的計算機(jī)PDP型計算機(jī)上運用C語言重寫了UNIX操作的系統(tǒng)內(nèi)部關(guān)鍵程序,可以這么說,C語言是伴隨著UNIX操作系統(tǒng)的誕生而誕生的。</p><p> 20世紀(jì)80年代中期,C語言被程序員廣泛使用,漸漸的演變成為個人計算機(jī)上非常流行和受歡迎的編程語言和工具。早在1983年7月,美國的國家標(biāo)準(zhǔn)委員會就對C語言進(jìn)行了標(biāo)準(zhǔn)化得管理,發(fā)布了第一個C語言的標(biāo)準(zhǔn)草案
16、,使得C語言廣發(fā)傳播。</p><p> 為了更好地適應(yīng)大規(guī)模軟件的生產(chǎn)和開發(fā),在C語言的基礎(chǔ)上,貝爾實驗室Biarne Stroustrup博士和她的同事們開始對它進(jìn)行改進(jìn)和擴(kuò)充。將“類”的概念引入到了C語言當(dāng)中。在1983年末構(gòu)成了最早期的C++語言。為了讓C++語言可以適應(yīng)軟件的大規(guī)模的開發(fā)需求,計算機(jī)技術(shù)的先驅(qū)們又為C++語言引入了多重繼承的概念,運算符的重載技術(shù)以及引用和虛函數(shù)方法等許多全新的特性。美
17、國國家標(biāo)準(zhǔn)委員會(ANSI)和國際標(biāo)準(zhǔn)化組織ISO一起進(jìn)行了標(biāo)準(zhǔn)化工作,并且于1998年正式頒布了C++語言的國際標(biāo)準(zhǔn)ISO/IEC:1998-14882,從此刻開始,軟件開發(fā)進(jìn)入到了一個快速發(fā)展的新階段。</p><p> 20世界90年代,美國的微軟公司(Microsoft)為了減少和降低Windows的應(yīng)用程序開發(fā)成本,提高應(yīng)用軟件在軟件市場當(dāng)中的地位和價值,在1992年發(fā)布了包含MFC2.0在內(nèi)的Vis
18、ual C++1.0。一個創(chuàng)時代的可視化C++集成的開發(fā)環(huán)境面世了。我們常說的MFC其實就是一個常用的軟件集合包(framework),就是利用面向?qū)ο蟮姆椒ò裌indows的應(yīng)用程序的接口封裝起來。提高了Windows的平臺上的軟件產(chǎn)品的開發(fā)效率。1998年,美國的微軟公司(Microsoft)推出了目前最流行也是最受歡迎的Visual C++ 6.0 版本。2002年末,又推出了Visual C++ 7.0,即是嵌入在Visual
19、Studio.NET框架中的Visual C++.NET2002。到目前為止,最新的版本已經(jīng)更新到了VS2012了。</p><p> 盡管軟件開發(fā)的環(huán)境一直在不換的更新發(fā)展,但是對于初學(xué)者來說,打好基礎(chǔ),學(xué)好C語言是非常重要的第一步。</p><p> 2.1.1 C語言是一門中級語言</p><p> C語言是一門非?;A(chǔ)的中級語言,也就是說C語言具有“
20、低級語言”的一些固有特征:允許程序自由訪問計算機(jī)物理地址,可以進(jìn)行位操作,能夠?qū)τ嬎銠C(jī)的硬件接口直接訪問,生成的目標(biāo)代碼的質(zhì)量非常高,程序的執(zhí)行效率也很高,同時它也兼顧了“高級語言”的固有特點:語句簡潔,緊湊,運算符特別靈活,數(shù)據(jù)類型豐富,具有結(jié)構(gòu)化的控制語句,邏輯嚴(yán)謹(jǐn),可移植性也很好。</p><p> 2.1.2 C語言是結(jié)構(gòu)式語言</p><p> C語言的一個特別突出的特點就
21、是C語言的代碼和數(shù)據(jù)是分隔開的,它把各個子程序之間除去必要的信息交流之外的程序部分相互獨立。這樣的結(jié)構(gòu)化程序設(shè)計方式可以使得程序代碼的層次更加清晰,便于用戶使用、程序的后期維護(hù)維護(hù)以及代碼編寫人員進(jìn)行調(diào)試。C 語言的很多方法是用函數(shù)的方式提供給用戶使用的,函數(shù)可以在C語言中非常方便的調(diào)用,并且C語言中具有多種循環(huán)控制結(jié)構(gòu)、條件判斷語句可以控制程序和數(shù)據(jù)信息的流向,從而使源代碼完全結(jié)構(gòu)化。</p><p> 2.
22、1.3 C語言的獨有特點</p><p> a ) C語言是包含結(jié)構(gòu)化程序設(shè)計以及遞歸功能的面向過程的中級語言。</p><p> b ) C語言的傳遞參數(shù)均是以值傳遞(pass by value),也可以傳遞指針(a pointer passed by value)。</p><p> c ) 在C語言中不同的變量類型能夠用結(jié)構(gòu)體(struct)組合在一
23、起。</p><p> d ) C語言只有32個保留字(reserved keywords),這樣變量、函數(shù)命名有更多的彈性。</p><p> e ) 在C語言中部份的變量類型可以轉(zhuǎn)換,比如整型和字符型變量。</p><p> f ) 使用指針的方式,C語言可以很輕松的對計算機(jī)中的存儲器進(jìn)行比較底層的控制。</p><p> g )
24、 預(yù)編譯處理(preprocessor)讓整個C語言的編譯過程更具有彈性。 </p><p> 2.2 C++簡介</p><p> C++語言一門非常流行而且受歡迎的高級語言,C語言算是C++語言當(dāng)中的一個子集,C++語言基本囊括了了C語言當(dāng)中的的絕大多數(shù)思想以及內(nèi)容。C++的應(yīng)用非常廣泛。C++可以支持多種編程范式 --面向?qū)ο缶幊獭⒎盒途幊桃约斑^程化編程。目前最新的正式標(biāo)準(zhǔn)C
25、++14將于2014年8月18日發(fā)布。 C++語言在程序設(shè)計的領(lǐng)域使用的非常廣泛,經(jīng)常將之運用在各種系統(tǒng)開發(fā)以及引擎開發(fā)等眾多的實際應(yīng)用當(dāng)中,是當(dāng)前社會上一門最受人們喜愛,使用人數(shù)最多的一門面向過程開發(fā)的程序設(shè)計語言之一。它可以支持類、封裝、重載等各種開發(fā)功能的實際應(yīng)用。</p><p> 2.2.1 C++語言支持?jǐn)?shù)據(jù)封裝</p><p> 支持?jǐn)?shù)據(jù)封裝可以說就是支持?jǐn)?shù)據(jù)
26、抽象。在C++語言中,類是支持?jǐn)?shù)據(jù)封裝的工具,對象是數(shù)據(jù)封裝的實現(xiàn)。面向?qū)ο蟮某绦蛟O(shè)計思想與面向過程的程序設(shè)計思想在運用函數(shù)以及數(shù)據(jù)信息的關(guān)系和方法上是完全不一樣的。在我們常用的面向過程的程序開發(fā)和設(shè)計的過程當(dāng)中,數(shù)據(jù)通常只是被當(dāng)做了一種靜態(tài)結(jié)構(gòu),只能夠等待程序的調(diào)用函數(shù)來對它們進(jìn)行計算和處理等操作,但是在我們面對對象方式的程序開發(fā)設(shè)計過程當(dāng)中,將要操作的數(shù)據(jù)以及可以對這個數(shù)據(jù)進(jìn)行操作的調(diào)用函數(shù)一起封裝,作為一個新生成的類的定義。另外,
27、封裝還可以提供一種對數(shù)據(jù)訪問的控制機(jī)制,保障程序的安全性。</p><p> 2.2.2 C++語言允許函數(shù)名和運算符重載</p><p> 函數(shù)名的重載和運算符的重載都是屬于高級語言的多態(tài)性表現(xiàn)。函數(shù)的多態(tài)性指的是完全一樣的語法結(jié)構(gòu)可以用來代表不相同的實體或者對不相同的實體類型進(jìn)行一些操作。C++語言可以支持函數(shù)的多態(tài)性,C++語言允許一個相同的運算符號或者一個完全一樣的函數(shù)名稱代
28、表多個不相同的實現(xiàn)的函數(shù),這個就被稱之為運算符或者說是表示符的重載。當(dāng)然,使用者可以根據(jù)自己的需要定義各種各樣的函數(shù)重載或者運算符重載。</p><p> 2.2.3 C++語言支持動態(tài)聯(lián)編</p><p> 在C++語言當(dāng)中可以定義虛函數(shù)。C++語言通過定義虛函數(shù)的方法可以完全支持程序設(shè)計中的動態(tài)聯(lián)編功能,動態(tài)聯(lián)編可以說是高級語言多態(tài)性表現(xiàn)出來的一個非常明顯的特點,多態(tài)性形成的由父
29、類以及子類來組成的一個簡單的樹形結(jié)構(gòu)。在這個樹形結(jié)構(gòu)中的每一個子類都可以接受一個或者數(shù)個具有相同名字的消息和信息。</p><p> 2.3 編譯原理介紹</p><p> 編譯程序算是計算機(jī)軟件系統(tǒng)的最基本的組成部分之一了,而且目前市場上流行的大多數(shù)計算機(jī)系統(tǒng)都配置了不止一種高級程序設(shè)計語言(例如JAVA和C++)的編譯程序,對一些使用率較高的高級語言甚至準(zhǔn)備了好幾個不同性能的編譯
30、程序。從功能上來說,一個編譯程序就好比一個語言的翻譯器,它可以把一種語言書寫的程序翻譯成用另外一種語言編寫的等價的程序。例如,匯編程序就是一個翻譯程序。它可以把匯編語言程序翻譯成由機(jī)器語言編寫的程序。如果這個程序的設(shè)計語言是像C語言或者C++語言這類的高級程序設(shè)計語言,而且程序的目標(biāo)語言是像機(jī)器語言這樣的低級程序設(shè)計語言,那么這個翻譯過程的程序就可以稱之為編譯程序,這種翻譯的原理則可以稱之為編譯原理。</p><p&
31、gt; 2.4 編譯器介紹</p><p> 編譯是從源程序代碼(一般是高級語言)到可以直接被計算機(jī)或者虛擬機(jī)執(zhí)行的目標(biāo)程序代碼(一般為低級語言或機(jī)器語言)的翻譯過程。但是,目前市場上也由把較低級的編程語言通過復(fù)雜的翻譯過程翻譯成較高級的編程語言的編譯程序,在這一類型的編譯器當(dāng)中,那部分用來把由高級編程語言生成的低級語言代碼重新翻譯成由高級編程語言代碼的程序被我們稱為反編譯器。也存在著從一種高級編程語言翻譯
32、為另外一種不同的高級編程語言的編譯程序,或者是生成另一種還需要計算機(jī)進(jìn)一步處理的的中間代碼的編譯程序(又稱之為級聯(lián))。</p><p> 3 程序設(shè)計前期分析</p><p><b> 3.1 初步調(diào)查</b></p><p> 在編譯原理的日常教學(xué)中,大部分學(xué)校的教學(xué)模式還是以課堂教學(xué)為主,學(xué)生主要通過在課堂上聽老師講解編譯原理教科
33、書上的理論知識來學(xué)習(xí)編譯原理這門課。但是,課堂講解的教學(xué)方式比較枯燥乏味,導(dǎo)致很多學(xué)生的學(xué)習(xí)積極性不高,學(xué)習(xí)效率較低,不能好好理解編譯原理的關(guān)鍵內(nèi)容。同時,學(xué)生在自己獨立編寫和設(shè)計編譯程序的時候往往無從下手,動手實踐能力很差,有的學(xué)生甚至連編譯器的運行環(huán)境和平臺都不了解,更不要說自己獨立設(shè)計和實現(xiàn)一個完整地編譯器程序了。所以,開展編譯原理的實踐教學(xué)課程勢在必行,在實踐課上,學(xué)生可以系統(tǒng)的學(xué)習(xí)編譯程序的開發(fā)過程,從一開始的需求分析,語法結(jié)
34、構(gòu)制定,詞法分析,語法分析,四元式到最后完整的編譯器。這些東西不自己動手練習(xí)是很難掌握的,通過實驗教學(xué),會使學(xué)生熟練地掌握編譯程序設(shè)計和實現(xiàn)的各個步驟,而且還能激發(fā)學(xué)生探究編譯原理更深層次的東西,增強(qiáng)學(xué)生對編譯原理的學(xué)習(xí)興趣。所以,一些好的、嚴(yán)謹(jǐn)?shù)膶嶒灣绦蚍独惋@得十分重要了。</p><p> 本課題的研究正是為了給編譯原理的日常教學(xué)提供一個更適合學(xué)生學(xué)習(xí)和成長的方式,給學(xué)生在動手實踐的過程中提供一個示范,使
35、越來越多的學(xué)生能對編譯原理產(chǎn)生興趣,希望繼續(xù)研究下去。這樣,老師們就再也不用擔(dān)心學(xué)生的成績啦。</p><p> 3.2 可行性分析</p><p> 通過對編譯原理課程實際教學(xué)的初步調(diào)查后,發(fā)現(xiàn)目前編譯原理課程的教學(xué)現(xiàn)狀非常需要一門實驗實踐課程來輔助教學(xué),并且現(xiàn)在掌握的資料和技術(shù)可以設(shè)計和編寫一些實驗實踐程序。在對現(xiàn)有的人力、物力、財力和技術(shù)條件進(jìn)行分析之后,確定開發(fā)該類高級語言編
36、譯系統(tǒng)的設(shè)計與實現(xiàn)課題,在經(jīng)濟(jì)上、技術(shù)上、時間上,法律上都已具備可行性。</p><p> a)經(jīng)濟(jì)可行性:即實現(xiàn)這個程序設(shè)計有沒有什么經(jīng)濟(jì)效益。這個編譯器程序設(shè)計是作為個人畢業(yè)設(shè)計而設(shè)計開發(fā)的,因為個人能力和設(shè)計水平不高,程序的結(jié)構(gòu)以及功能都不夠完整,正確性有待考究,內(nèi)容也許沒有太高的實際價值,該程序設(shè)計方案可能并沒有太高的經(jīng)濟(jì)效益,但是作為一個畢業(yè)設(shè)計,卻使得它具有了開發(fā)的價值。</p>&l
37、t;p> b)技術(shù)可行性:即現(xiàn)有的技術(shù)能否開發(fā)該程序,會有哪些困難。目前掌握的編譯原理和C語言知識還行,也有很多相關(guān)資料可以查閱,所以技術(shù)上應(yīng)該是可行的。</p><p> c)運行可行性:即該平臺規(guī)定的運行方式是否可行。本課題用的主要是VC++6.0,和學(xué)校機(jī)房的操作環(huán)境一致,所以肯定是可以再學(xué)校機(jī)房運行的。</p><p> d)法律可行性:該程序的開發(fā)和設(shè)計應(yīng)該不會觸犯法
38、律。因為該程序只是作為畢業(yè)設(shè)計與商業(yè)無關(guān),又因為是自主開發(fā)設(shè)計,因此不會構(gòu)成侵權(quán),在法律上是可行的。</p><p> 通過以上的可行性分析,決定設(shè)計和實現(xiàn)這個編譯器程序。</p><p> 在程序的設(shè)計和開發(fā)過程中,還應(yīng)注意一下兩點:</p><p> a)要注意程序的準(zhǔn)確性和注釋的完整性,只有完整的準(zhǔn)確的注釋才能為學(xué)生提供一個正確的參考和借鑒。</p
39、><p> b)從全局出發(fā)注意程序設(shè)計的整體優(yōu)化,還要注意程序的可擴(kuò)展性和延伸性。</p><p><b> 3.3 需求分析</b></p><p> 我們在軟件開發(fā)中經(jīng)常遇見關(guān)于需求分析的問題。需求分析,就是指對程序開發(fā)當(dāng)中需要解決的問題和障礙進(jìn)行的分析處理,搞明白要開發(fā)的程序軟件的實際需求,包括程序要輸入什么樣的類型和種類的數(shù)據(jù),期望
40、得到怎樣的運行結(jié)果,最終程序應(yīng)該輸出什么東西。可以說,在軟件工程當(dāng)中的“需求分析”就是確定程序要“做什么”。 </p><p> 在程序設(shè)計開發(fā)的過程中,我們所說的需求分析指的是在建立一個全新的或改變一個已經(jīng)存在的程序或者軟件的時候描述新的程序的設(shè)計方法、范圍、目的、功能和定義時所要進(jìn)行的所有的工作的總稱。需求分析是軟件工程中的一個關(guān)鍵過程。在需求分析的過程中,大部分的分析員以及高級軟件工程師都會明確顧客的實際
41、需要,只有在我們明確了顧客的這些實際需要以后,程序設(shè)計人員才能夠分析出新的系統(tǒng)的解決方案和方法。 </p><p> 基于需求分析的重要作用,查閱了圖書館書籍,瀏覽了網(wǎng)上的相關(guān)資料,從程序設(shè)計的方法,功能的實現(xiàn),技術(shù)的要求以及可行性等多方面進(jìn)行考慮完成了需求分析。</p><p> 現(xiàn)在很多高校都急需開展編譯原理課程的實驗實踐課程,肯定需要一些和實驗教學(xué)相關(guān)的參考程序,本課題設(shè)計的這些
42、程序可以很好地滿足他們的需求,所以本課題的這個設(shè)計還是很有市場的。</p><p><b> 4 總體設(shè)計</b></p><p><b> 4.1 設(shè)計思想</b></p><p> 本程序采用分步設(shè)計方式設(shè)計,把整個編譯器分解成三個主要部分,分別是詞法分析程序,語法分析程序,語義分析程序。各個部分的設(shè)計和實現(xiàn)
43、相對獨立,等把各個部分的功能設(shè)計和實現(xiàn)以后,在根據(jù)編譯器的運行原來來進(jìn)行組合,這樣,一個完整的,結(jié)構(gòu)層次清晰的編譯器就開發(fā)完成了。</p><p> 首先需要設(shè)計的程序是詞法分析器,</p><p> 第二步要做的是設(shè)計語法分析器,</p><p> 第三部要實現(xiàn)的部分是語義分析器。</p><p> 最后設(shè)計一個簡單地前端,把上面三
44、個功能組合一下,就實現(xiàn)了一個完整的編譯器程序了。</p><p><b> 4.2 主要任務(wù)</b></p><p> 由于本程序是用于給學(xué)校的實驗課教學(xué)提供一個參考,所以,程序的準(zhǔn)確性和平臺適用性需要保障,要確定程序能夠在學(xué)校機(jī)房的操作環(huán)境正常運行,同時要注意程序注釋部分的完整性,要使學(xué)生能夠清楚正確的理解程序的每個部分要實現(xiàn)的功能,要讓他們弄懂每一句代碼的含
45、義,由于編譯原理詞法分析,語法分析,語義分析的方法有很多,所以還要注意程序編譯分析方法的多樣性。由于每個學(xué)生擅長的編程語言各不相同,所以本課題選擇了C語言和C++語言作為程序開發(fā)的主要語言,因為C語言和C++語言是計算機(jī)科學(xué)與技術(shù)系的專業(yè)必修課,每個學(xué)生都應(yīng)該掌握的很不錯。同時,C語言和C++語言都很常用,二者代表了現(xiàn)如今最流行的兩種程序設(shè)計語言的類型,分別體現(xiàn)了面向?qū)ο笠约懊嫦蜻^程兩種完全不同的編程思想,對學(xué)生的日常學(xué)習(xí)有非常多的幫助
46、。</p><p><b> 4.3 模塊結(jié)構(gòu)</b></p><p> 4.3.1 詞法分析</p><p> 詞法分析的過程基本上是所有的編譯程序的第一個處理過程,編譯器一般可以使用兩種完全不相同的設(shè)計方法來構(gòu)造和設(shè)計開發(fā)詞法分析子程序。其中第一種就是根據(jù)對類高級程序設(shè)計語言中得各類單詞和符號的某種描述以及定義,使用手工構(gòu)造的策略
47、(例如可以用C語言或者C++語言)來開發(fā)詞法分析子程序。通常來說,程序可以根據(jù)已經(jīng)定義好的文法規(guī)則或者已經(jīng)畫好的狀態(tài)轉(zhuǎn)換圖來設(shè)計和開發(fā)相對應(yīng)的狀態(tài)矩陣,這個狀態(tài)矩陣所有的控制語句一起控制程序就可以組成一個簡單的編譯程序的詞法分析程序;也可以根據(jù)定義好的文法規(guī)則或狀態(tài)轉(zhuǎn)換圖來直接設(shè)計和編寫詞法分析程序。另外的一種構(gòu)造詞法分析程序的途徑就是所謂的詞法分析程序的自動生成,即先使用正規(guī)式對類高級語言中的各個種類的單詞和符號進(jìn)行詞型描述,并且分別
48、指出在識別單詞的過程中詞法分析程序應(yīng)該進(jìn)行的語義處理工作,然后用一個所謂的詞法分析程序的構(gòu)造程序?qū)ι鲜龅男畔⑦M(jìn)行進(jìn)一步的加工。例如著名的美國BELL科學(xué)實驗室研制和開發(fā)的LEX程序就是一個詞法分析子程序的自動化生成工具。</p><p> 一般的情況下,當(dāng)需要開發(fā)一種全新的編程語言的時候,因為該編程語言的單詞和運算符號在不停地開發(fā)和修改,所以利用LEX這一類的工具生成的我們需要的詞法分析程序相對于手工編寫的方式
49、來說會比較易于修改和維護(hù)。當(dāng)然,如果一旦這種語言確定了,那么采用手工設(shè)計和編寫詞法分析程序的效率會更高。</p><p> 4.3.2 語法分析</p><p> 語法分析的過程可以說是編譯程序中最核心的部分了,語法分析的目的就是識別由詞法分析程序所給出的單詞或者符號序列是不是事先給定文法的正確程序語句,目前大學(xué)教科書當(dāng)中涉及的語法分析方法有自頂向下分析方法以及自底向上這兩大類別,自
50、頂向下的分析方法又包括了確定分析和不確定分析兩個分支,自底向上分析方法則又包含了運算符優(yōu)先分析方法以及LR分析方法,這些分析方法都有各自的優(yōu)缺點。然而除去自頂向下的不確定分析方法外,都是目前編譯程序構(gòu)造的實用方法,本課題將在下一章著重介紹它們的實現(xiàn)原理和實踐技術(shù)。</p><p> 4.3.3 語義分析</p><p> 語法制導(dǎo)翻譯的模式是在語法分析的基礎(chǔ)上,增加語義操作來實現(xiàn)的。
51、對于要求中給定文法當(dāng)中的每一個產(chǎn)生式,開發(fā)和設(shè)計其相對應(yīng)的語義子程序。在程序的語法分析過程當(dāng)中,每當(dāng)程序使用一個產(chǎn)生式來進(jìn)行歸約或者是推導(dǎo)的時候,語法分析程序不但要執(zhí)行相對應(yīng)的語法分析動作,還需要調(diào)用相對應(yīng)的語義子程序,以便程序可以順利的完成包括生成中間代碼、查找和填寫相關(guān)的表格數(shù)據(jù)、檢查并且報告源程序中的錯誤在內(nèi)所有工作。每一個語義子程序都要求能夠指明相對應(yīng)的產(chǎn)生當(dāng)中的每一個字符和符號的具體含義和內(nèi)容,而且需要規(guī)定使用這個產(chǎn)生式來分析
52、程序的時候所應(yīng)該進(jìn)行的語義動作。</p><p><b> 5 詳細(xì)設(shè)計</b></p><p> 5.1 詞法分析器的設(shè)計和實現(xiàn)</p><p> 本程序用手工編碼方式構(gòu)造識別類高級語言的詞法分析程序。該類語言中具有的單詞包括了五個有代表性的關(guān)鍵字begin、end、if、then、else;標(biāo)識符;整型常數(shù);六種關(guān)系運算符;一個賦
53、值符和四個算術(shù)運算符。</p><p> 第一步,我們需要做的是對單詞進(jìn)行簡單的分類:設(shè)計好上面所說的類高級語言中的每一類單詞的符號以及分類碼表。</p><p><b> 如下表5.1所示:</b></p><p> 表5.1 類高級語言中的各類單詞符號及其分類碼表</p><p> 然后就是關(guān)于本程序詞法分
54、析的詳細(xì)處理過程了,在一個高級程序設(shè)計語言中,通常都會包含很多種類的單詞符號。所以,本程序采用先為每個種類的單詞符號建立一張完整的狀態(tài)轉(zhuǎn)換圖,然后再把這些狀態(tài)轉(zhuǎn)換圖一起組合成一張統(tǒng)一的狀態(tài)圖,就可以得到一個有限的自動機(jī),再進(jìn)行一些必要的確定化以及狀態(tài)數(shù)的最小化處理過程,最后再根據(jù)這個程序的狀態(tài)圖來設(shè)計和編寫詞法分析程序。在這里,為了使得本程序的詞法分析程序的設(shè)計結(jié)構(gòu)更加清晰,而且要盡量避免一些不必要問題的煩惱,可以假設(shè)要編譯的類高級語言
55、當(dāng)中,所有的關(guān)鍵字都算作是系統(tǒng)的保留字,開發(fā)人員不得將它們作為源程序中的標(biāo)識符;在要分析的源程序的輸入文本當(dāng)中,標(biāo)識符、關(guān)鍵字以及整常數(shù)之間,如果沒有出現(xiàn)關(guān)系運算符、算術(shù)運算符以及賦值符號,那么程序至少需要使用一個空白的字符來加以分隔。作了這些限制以后,就可以把關(guān)鍵字和標(biāo)識符的識別統(tǒng)一進(jìn)行處理。每當(dāng)程序開始識別一個單詞的時候,如果掃視到的第一個字符是字母的話,就會把后續(xù)輸入的字母或者數(shù)字字符按順序進(jìn)行一一拼接,直到程序掃視到不是字母或者
56、數(shù)字字符為止,以期望可以得到一個盡可能長的字母字符串或者數(shù)字字符串,然后程序再用這個字符串來查詢</p><p> 第三步,將表5.1里面的單詞集中的整常數(shù)全部改為無符號常數(shù),本程序把無符號常數(shù)的單詞分類碼助記符記為UCON;其值為無符號常數(shù)的機(jī)內(nèi)二進(jìn)制表示。</p><p> 描述無符號數(shù)的BNF定義和狀態(tài)轉(zhuǎn)換圖:</p><p> 無符號數(shù)的文法G如下:&
57、lt;/p><p> 〈無符號數(shù)〉→ d〈余留無符號數(shù)〉</p><p> 〈無符號數(shù)〉→ ·〈小數(shù)部分〉</p><p><b> 〈無符號數(shù)〉→ d</b></p><p> 〈余留無符號數(shù)〉→ d〈余留無符號數(shù)〉</p><p> 〈余留無符號數(shù)〉→ ·〈十進(jìn)小數(shù)〉
58、</p><p> 〈余留無符號數(shù)〉→ E〈指數(shù)部分〉</p><p> 〈余留無符號數(shù)〉→ d</p><p> 〈余留無符號數(shù)〉→ ·</p><p> 〈十進(jìn)小數(shù)〉→ E〈指數(shù)部分〉</p><p> 〈十進(jìn)小數(shù)〉→ d〈十進(jìn)小數(shù)〉</p><p><b>
59、 〈十進(jìn)小數(shù)〉→ d</b></p><p> 〈小數(shù)部分〉→ d〈十進(jìn)小數(shù)〉</p><p><b> 〈小數(shù)部分〉→ d</b></p><p> 〈指數(shù)部分〉→ d〈余留整指數(shù)〉</p><p> 〈指數(shù)部分〉→ +〈整指數(shù)〉</p><p> 〈指數(shù)部分〉→ -〈整指
60、數(shù)〉</p><p><b> 〈指數(shù)部分〉→ d</b></p><p> 〈整指數(shù)〉→ d〈余留整指數(shù)〉</p><p><b> 〈整指數(shù)〉→ d</b></p><p> 〈余留整指數(shù)〉→ d〈余留整指數(shù)〉</p><p> 〈余留整指數(shù)〉→ d</p
61、><p> 圖5.1所表示的就是上述文法的狀態(tài)轉(zhuǎn)換圖,下圖當(dāng)中編號0、1、2、3、4、5、6 分別代表了非終結(jié)符號中的<無符號數(shù)>、 <余留無符號數(shù)>、 <十進(jìn)小數(shù)>、 <小數(shù)部分>、<指數(shù)部分> 、<整指數(shù)> 以及<余留整指數(shù)>。</p><p> 圖5.1 文法G【<無符號數(shù)>】的狀態(tài)轉(zhuǎn)化
62、圖</p><p> 本程序用來實現(xiàn)無符號數(shù)識別的方法:在計算機(jī)系統(tǒng)內(nèi)部實現(xiàn)狀態(tài)轉(zhuǎn)換圖的常用方法方法之一就是把狀態(tài)圖當(dāng)中的各個狀態(tài)作為行,把有可能輸入的每一個輸入符號做為列,組成了一個狀態(tài)矩陣。在這個狀態(tài)矩陣當(dāng)中,矩陣內(nèi)的元素用來表示下一個狀態(tài)以及掃描器需要完成的語義動作(例如字符的拼接、數(shù)制之間的轉(zhuǎn)換、填充的符號表和程序輸出的單詞在計算機(jī)內(nèi)部的表示方法等)。下表5.2里面的第一列表示各狀態(tài)Si的編號,表中的第
63、二列則分別列出了在每一種狀態(tài)下程序有可能掃視到的輸入符號aj,表中的第三列指出了當(dāng)(Si,aj)出現(xiàn)的時候程序應(yīng)該執(zhí)行的語義動作,最后一欄的內(nèi)容用來指明下一個狀態(tài)的編號(假如其后是NULL或者是“結(jié)束”則表示沒有后繼狀態(tài))。</p><p> 表5.2 包含語義處理過程的識別無符號數(shù)的狀態(tài)矩陣</p><p> 圖5.2和詞法分析程序中所出現(xiàn)的語義變量及語義函數(shù)的含義和功能說明如下。
64、</p><p> 函數(shù)GETCHAR:程序每調(diào)用一次該函數(shù),就會把掃描指示器當(dāng)中的現(xiàn)在所指示的源程序中的字符傳送給字符變量ch,接著程序會把掃描指示器的位置前推一個字符位置。</p><p> 字符數(shù)組TOKEN:用來依次存放一個單詞詞文中的各個字符。</p><p> 函數(shù)CAT:程序每調(diào)用一次該函數(shù),就會把當(dāng)前ch中存儲的字符拼接到TOKEN中所存儲的字
65、符串的右邊。</p><p> 函數(shù)LOOKUP:程序每調(diào)用一次該函數(shù),就會用TOKEN中存儲的字符串來查詢保留字表,如果查到存在,該函數(shù)就會將相對應(yīng)關(guān)鍵字的類別碼賦值傳給整型變量c;如果沒有查到就將c置為零。</p><p> 函數(shù)RETRACT:程序每調(diào)用一次該函數(shù),就會把掃描指示器立刻回退一個字符的位置(就是退掉剛才多讀的那個字符)。</p><p>
66、函數(shù)OUT:程序一般只有在進(jìn)入終態(tài)的時候才會調(diào)用到這個函數(shù),調(diào)用的形式一般是OUT(c,VAL)。在這里面,實參c作為相對應(yīng)的單詞的類別碼或者是它的助記符。</p><p><b> 圖5.2如下所示:</b></p><p> 圖5.2 識別圖5.1所列語言中的部分單詞的DFA及相關(guān)的語義過程</p><p> 該詞法分析程序編譯運行結(jié)
67、果如圖5.3所示:</p><p> 圖5.3 詞法分析程序運行界面圖</p><p> 輸入字符串begin x:=10;if x>5 then x:=2*x+6/2; end #;</p><p> 其運行結(jié)果如下圖5.4所示:</p><p> 圖5.4 詞法分析程序測試結(jié)果一</p><p>
68、輸入字符串hello end#;</p><p> 其運行結(jié)果如下圖5.5所示:</p><p> 圖5.5 詞法分析程序測試結(jié)果一</p><p> 輸入字符串if x>5 ;y:=3; end #;</p><p> 其運行結(jié)果如下圖5.6所示:</p><p> 圖5.6 詞法分析程序測試結(jié)果三&
69、lt;/p><p><b> 源程序如下所示:</b></p><p> #include<stdio.h></p><p> #include<string.h></p><p> char prog[80],token[6];</p><p><b>
70、char ch;</b></p><p> int syn,p,m,n,sum;</p><p> char * rwtab[6]={"begin","if","then","while","do","end"}; //定義關(guān)鍵詞</p>
71、<p><b> main()</b></p><p><b> {</b></p><p><b> p=0;</b></p><p> printf("\n please intput string:");</p><p><b
72、> do</b></p><p><b> {</b></p><p> ch=getchar();</p><p> prog[p++]=ch;</p><p> }while(ch!='#');</p><p><b> p=0;<
73、/b></p><p><b> do</b></p><p><b> {</b></p><p><b> scaner();</b></p><p> switch(syn)</p><p><b> {</b>
74、;</p><p> case 11:printf("(%d,%d)\n",syn,sum);break;</p><p> case -1:printf("input error\n"); break;</p><p> default:printf("(%d,%s)\n",syn,token);
75、</p><p><b> }</b></p><p> }while(syn!=0);</p><p><b> getch();</b></p><p><b> }</b></p><p> /*詞法掃描程序:*/</p>
76、<p><b> scaner()</b></p><p><b> {</b></p><p> for(n=0;n<8;n++)</p><p> token[n]=NULL;</p><p><b> m=0;</b></p>&l
77、t;p> ch=prog[p++];</p><p> while(ch==' ')ch=prog[p++];</p><p> if((ch<='z'&&ch>='a')||(ch<='Z'&&ch>='A'))</p><
78、;p><b> {</b></p><p> while((ch<='z'&&ch>='a')||(ch<='Z'&&ch>='A')||(ch<='9'&&ch>='0'))</p>&l
79、t;p><b> {</b></p><p> token[m++]=ch;</p><p> ch=prog[p++];</p><p><b> }</b></p><p> token[m++]='\0';</p><p> ch=pr
80、og[--p];</p><p><b> syn=10;</b></p><p> for(n=0;n<6;n++)</p><p> if(strcmp(token,rwtab[n])==0)</p><p><b> {</b></p><p><b
81、> syn=n+1;</b></p><p><b> break;</b></p><p><b> }</b></p><p><b> }</b></p><p><b> else</b></p><
82、p> if((ch<='9'&&ch>='0'))</p><p><b> {</b></p><p><b> sum=0;</b></p><p> while((ch<='9'&&ch>='
83、0'))</p><p><b> {</b></p><p> sum=sum*10+ch-'0';</p><p> ch=prog[p++];</p><p><b> }</b></p><p> ch=prog[--p];</
84、p><p><b> syn=11;</b></p><p><b> }</b></p><p><b> else</b></p><p> switch(ch)</p><p><b> {</b></p>
85、<p> case '<':m=0;token[m++]=ch;</p><p> ch=prog[p++];</p><p> if(ch=='>')</p><p><b> {</b></p><p><b> syn=21;</b&
86、gt;</p><p> token[m++]=ch;</p><p><b> }</b></p><p><b> else</b></p><p> if(ch=='=')</p><p><b> {</b></p
87、><p><b> syn=22;</b></p><p> token[m++]=ch;</p><p><b> }</b></p><p><b> else</b></p><p><b> {</b></p&g
88、t;<p><b> syn=20;</b></p><p> ch=prog[--p];</p><p><b> }</b></p><p><b> break;</b></p><p> case '>':token[m++
89、]=ch;</p><p> ch=prog[p++];</p><p> if(ch=='=')</p><p><b> {</b></p><p><b> syn=24;</b></p><p> token[m++]=ch;</p&g
90、t;<p><b> }</b></p><p><b> else</b></p><p><b> {</b></p><p><b> syn=23;</b></p><p> ch=prog[--p];</p>
91、<p><b> }</b></p><p><b> break;</b></p><p> case ':':token[m++]=ch;</p><p> ch=prog[p++];</p><p> if(ch=='=')</p&g
92、t;<p><b> {</b></p><p><b> syn=18;</b></p><p> token[m++]=ch;</p><p><b> }</b></p><p><b> else</b></p>
93、<p><b> {</b></p><p><b> syn=17;</b></p><p> ch=prog[--p];</p><p><b> }</b></p><p><b> break;</b></p>
94、<p> case '+':syn=13;token[0]=ch;break; //判斷符號“+”。</p><p> case '-':syn=14;token[0]=ch;break; //判斷符號“-”。</p><p> case '*':syn=15;token[0]=ch;break; //判斷
95、符號“*”。</p><p> case '/':syn=16;token[0]=ch;break; //判斷符號“/”。</p><p> case ':=':syn=18;token[0]=ch;break; //判斷符號“:=”。</p><p> case '<>':syn=21;t
96、oken[0]=ch;break; //判斷符號“<>”。</p><p> case '<=':syn=22;token[0]=ch;break; //判斷符號“<=”。</p><p> case '>=':syn=24;token[0]=ch;break; //判斷符號“>=”。</p>
97、<p> case '=':syn=25;token[0]=ch;break; //判斷符號“=”。</p><p> case ';':syn=26;token[0]=ch;break; //判斷符號“;”。</p><p> case '(':syn=27;token[0]=ch;break; //
98、判斷符號“(”。</p><p> case ')':syn=28;token[0]=ch;break; //判斷符號“)”。</p><p> case '#':syn=0;token[0]=ch;break; //判斷符號“#”。</p><p> default:syn=-1;</p>&l
99、t;p><b> }</b></p><p><b> }</b></p><p> 5.2 語法分析器的設(shè)計和實現(xiàn)</p><p> 通過設(shè)計和開發(fā)一個典型的語法分析程序,使學(xué)生能夠?qū)崿F(xiàn)并且進(jìn)一步掌握日常教學(xué)中常常涉及的語法分析方法。本程序把下圖文法G1所定義的算術(shù)表達(dá)式的賦值語句作為分析對象,對輸入語
100、句的語法進(jìn)行分析。</p><p> G1[<賦值語句>]:</p><p> <賦值語句> → <變量>:=<算術(shù)表達(dá)式></p><p> <算術(shù)表達(dá)式> → <項> | <算術(shù)表達(dá)式>+<項> | <算術(shù)表達(dá)式>-<項></p&g
101、t;<p> <項> → <因式> | <項>*<因式> | <項>/<因式></p><p> <因式> → <變量> | <常數(shù)> | (<算術(shù)表達(dá)式>)</p><p> <變量> → <標(biāo)識符></p>&
102、lt;p> <標(biāo)識符> → <標(biāo)識符> <字母> | <標(biāo)識符> <數(shù)字> | <字母></p><p> <常數(shù)> → <整數(shù)> | <浮點數(shù)></p><p> <整數(shù)> → <數(shù)字> | <數(shù)字> <整數(shù)></p
103、><p> <浮點數(shù)> → ? <整數(shù)> | <整數(shù)> ? <整數(shù)></p><p> <字母> → A|B|C|…|X|Y|Z|a|b|c|…|x|y|z</p><p> <數(shù)字> → 0|1|2|…9</p><p> 該語法分析程序編譯運行結(jié)果如圖5.7所示:
104、</p><p> 圖5.7 語法分析程序運行結(jié)果界面</p><p> 輸入字符串begin x:=3*y end #;</p><p> 其運行結(jié)果如下圖5.8所示:</p><p> 圖5.8 語法分析程序測試結(jié)果一</p><p> 輸入字符串x:=3*5 end #;</p><
105、p> 其運行結(jié)果如下圖5.9所示:</p><p> 圖5.9 語法分析程序測試結(jié)果二</p><p> 輸入字符串begin x:=2*3(3+4 end #;</p><p> 其運行結(jié)果如下圖5.10所示:</p><p> 圖5.10 語法分析程序測試結(jié)果三</p><p><b>
106、 源程序如下所示:</b></p><p> #include<iostream></p><p> using namespace std;</p><p> #include<stdio.h></p><p> #include<string.h></p><p&
107、gt; #define MAX 150 //詞法分析表的最大容量</p><p> #define MAXBUF 255 //緩沖區(qū)的最大緩沖量</p><p> void term();</p><p> void lrparser();</p><p> void statement();</p>
108、<p> void yucu();</p><p> void expression();</p><p> void factor();</p><p> char prog[MAXBUF],token[MAX];</p><p><b> char ch;</b></p><p
109、> int syn,p,m,n,sum,kk;</p><p> char * rwtab[6]={"begin","if","then","while","do","end"}; //定義關(guān)鍵詞</p><p> void scaner()
110、 //詞法掃描程序</p><p><b> {</b></p><p> for(m=0;m<MAX;m++)</p><p> token[m]=NULL;</p><p> m=0;sum=0;</p><p>
111、ch=prog[p++];</p><p> while(ch==' ')ch=prog[p++];</p><p> if((ch<='z'&&ch>='a')||(ch<='Z'&&ch>='A'))</p><p><
112、;b> {</b></p><p> while((ch<='z'&&ch>='a')||(ch<='Z'&&ch>='A')||(ch<='9'&&ch>='0'))</p><p>&l
113、t;b> {</b></p><p> token[m++]=ch;</p><p> ch=prog[p++]; //讀取下一個字符</p><p><b> }</b></p><p> token[m++]='\0';</p&g
114、t;<p> ch=prog[--p];</p><p><b> syn=10;</b></p><p> for(n=0;n<6;n++)</p><p> if(strcmp(token,rwtab[n])==0)</p><p><b> {</b></p
115、><p> syn=n+1; //給出syn值</p><p><b> break;</b></p><p><b> }</b></p><p><b> }</b></p><p><b>
116、else</b></p><p> if((ch<='9'&&ch>='0'))</p><p><b> {</b></p><p><b> sum=0;</b></p><p> while((ch<=
117、9;9'&&ch>='0'))</p><p><b> {</b></p><p> sum=sum*10+ch-'0';</p><p> ch=prog[p++];</p><p><b> }</b></p>
118、<p> ch=prog[--p];</p><p><b> syn=11;</b></p><p><b> }</b></p><p><b> else</b></p><p> switch(ch)</p><p><
119、;b> {</b></p><p> case '<':m=0;token[m++]=ch;</p><p> ch=prog[p++];</p><p> if(ch=='>')</p><p><b> {</b></p><
120、p><b> syn=21;</b></p><p> token[m++]=ch;</p><p><b> }</b></p><p><b> else</b></p><p> if(ch=='=')</p><p&g
121、t;<b> {</b></p><p><b> syn=22;</b></p><p> token[m++]=ch;</p><p><b> }</b></p><p><b> else</b></p><p>
122、<b> {</b></p><p><b> syn=20;</b></p><p> ch=prog[--p];</p><p><b> }</b></p><p><b> break;</b></p><p>
123、 case '>':token[m++]=ch;</p><p> ch=prog[p++];</p><p> if(ch=='=')</p><p><b> {</b></p><p><b> syn=24;</b></p><
124、;p> token[m++]=ch;</p><p><b> }</b></p><p><b> else</b></p><p><b> {</b></p><p><b> syn=23;</b></p><p
125、> ch=prog[--p];</p><p><b> }</b></p><p><b> break;</b></p><p> case ':':token[m++]=ch;</p><p> ch=prog[p++];</p><p>
126、; if(ch=='=')</p><p><b> {</b></p><p><b> syn=18;</b></p><p> token[m++]=ch;</p><p><b> }</b></p><p><b
127、> else</b></p><p><b> {</b></p><p><b> syn=17;</b></p><p> ch=prog[--p];</p><p><b> }</b></p><p><b>
128、; break;</b></p><p> case '+':syn=13;token[0]=ch;break; //判斷符號“+”。</p><p> case '-':syn=14;token[0]=ch;break; //判斷符號“-”。</p><p> case '
溫馨提示
- 1. 本站所有資源如無特殊說明,都需要本地電腦安裝OFFICE2007和PDF閱讀器。圖紙軟件為CAD,CAXA,PROE,UG,SolidWorks等.壓縮文件請下載最新的WinRAR軟件解壓。
- 2. 本站的文檔不包含任何第三方提供的附件圖紙等,如果需要附件,請聯(lián)系上傳者。文件的所有權(quán)益歸上傳用戶所有。
- 3. 本站RAR壓縮包中若帶圖紙,網(wǎng)頁內(nèi)容里面會有圖紙預(yù)覽,若沒有圖紙預(yù)覽就沒有圖紙。
- 4. 未經(jīng)權(quán)益所有人同意不得將文件中的內(nèi)容挪作商業(yè)或盈利用途。
- 5. 眾賞文庫僅提供信息存儲空間,僅對用戶上傳內(nèi)容的表現(xiàn)方式做保護(hù)處理,對用戶上傳分享的文檔內(nèi)容本身不做任何修改或編輯,并不能對任何下載內(nèi)容負(fù)責(zé)。
- 6. 下載文件中如有侵權(quán)或不適當(dāng)內(nèi)容,請與我們聯(lián)系,我們立即糾正。
- 7. 本站不保證下載資源的準(zhǔn)確性、安全性和完整性, 同時也不承擔(dān)用戶因使用這些下載資源對自己和他人造成任何形式的傷害或損失。
最新文檔
- 本科畢業(yè)設(shè)計(論文)
- 本科畢業(yè)設(shè)計(論文)
- 本科畢業(yè)設(shè)計( 論文 )
- 采礦本科畢業(yè)設(shè)計
- 本科畢業(yè)設(shè)計.doc
- 制藥工程本科畢業(yè)設(shè)計
- 公路隧道本科畢業(yè)設(shè)計
- 本科畢業(yè)設(shè)計手冊.doc
- 本科畢業(yè)設(shè)計(論文)模板
- 軸承本科畢業(yè)設(shè)計論文
- 本科畢業(yè)設(shè)計(論文)模板
- 抗滑樁本科畢業(yè)設(shè)計計算書
- 本科畢業(yè)設(shè)計開題報告.doc
- 本科畢業(yè)設(shè)計開題報告.doc
- 本科畢業(yè)設(shè)計文檔管理系統(tǒng)設(shè)計
- 抗滑樁本科畢業(yè)設(shè)計計算書
- 本科畢業(yè)設(shè)計外文翻譯.doc
- 本科畢業(yè)設(shè)計開題報告.doc
- 邊緣檢測本科畢業(yè)設(shè)計論文
- 本科畢業(yè)設(shè)計開題報告.doc
評論
0/150
提交評論