詞法分析和算符優(yōu)先分析課程設(shè)計(jì)_第1頁(yè)
已閱讀1頁(yè),還剩22頁(yè)未讀, 繼續(xù)免費(fèi)閱讀

下載本文檔

版權(quán)說(shuō)明:本文檔由用戶提供并上傳,收益歸屬內(nèi)容提供方,若內(nèi)容存在侵權(quán),請(qǐng)進(jìn)行舉報(bào)或認(rèn)領(lǐng)

文檔簡(jiǎn)介

1、<p>  計(jì)算機(jī)科學(xué)與技術(shù)學(xué)院</p><p><b>  課程設(shè)計(jì)</b></p><p><b>  課程名稱:編譯原理</b></p><p>  題 目:詞法分析和算符優(yōu)先分析</p><p><b>  班 級(jí): </b></p>

2、<p><b>  學(xué) 號(hào): </b></p><p><b>  姓 名: </b></p><p>  2012年5月24日</p><p><b>  設(shè)計(jì)一:詞法分析器</b></p><p>  1. 課程設(shè)計(jì)目的和要求1</p>

3、<p><b>  1.1實(shí)驗(yàn)?zāi)康?</b></p><p><b>  1.2實(shí)驗(yàn)要求1</b></p><p><b>  2.設(shè)計(jì)描述2</b></p><p><b>  3.函數(shù)模塊3</b></p><p>  5.測(cè)試樣

4、例與測(cè)試結(jié)果6</p><p><b>  6. 結(jié)論7</b></p><p>  設(shè)計(jì)二: 算符優(yōu)先語(yǔ)法分析</p><p>  1.課程設(shè)計(jì)的目的和要求8</p><p>  1.1 課程設(shè)計(jì)的目的8</p><p>  1.2 課程設(shè)計(jì)的要求8</p><p&

5、gt;<b>  2.設(shè)計(jì)描述8</b></p><p>  2.1 自底向上分析方法的描述:8</p><p>  2.2算符優(yōu)先文法的描述:9</p><p>  3. 概要設(shè)計(jì)和詳細(xì)設(shè)計(jì)10</p><p>  4.1功能結(jié)構(gòu)10</p><p>  4.2模塊的劃分10<

6、/p><p><b>  5.源代碼11</b></p><p>  5.測(cè)試樣例與測(cè)試結(jié)果21</p><p><b>  6. 結(jié)論22</b></p><p><b>  設(shè)計(jì)一:詞法分析器</b></p><p><b>  課程設(shè)計(jì)

7、目的和要求</b></p><p><b>  1.1實(shí)驗(yàn)?zāi)康?lt;/b></p><p>  通過(guò)完成詞法分析程序,了解詞法分析的過(guò)程。編制一個(gè)讀單詞過(guò)程,從輸入的源程序中,識(shí)別出各個(gè)具有獨(dú)立意義的單詞,即基本保留字、標(biāo)識(shí)符、常數(shù)、運(yùn)算符、分隔符五大類。并依次輸出各個(gè)單詞的內(nèi)部編碼及單詞符號(hào)自身值。</p><p><b>

8、  1.2實(shí)驗(yàn)要求</b></p><p>  通過(guò)詞法分析器能夠?qū)崿F(xiàn)以下五種類型如單詞等的識(shí)別。</p><p>  (1)關(guān)鍵字"begin","end","if","then","else","while","write","

9、;read"等,</p><p>  "do", "call","const","char","until","procedure","repeat"等</p><p>  (2)運(yùn)算符:"+","-"

10、,"*","/","="等</p><p>  (3)界符:"{","}","[","]",";",",",".","(",")",":"等</p&g

11、t;<p>  (4)標(biāo)識(shí)符 (5)常量</p><p><b>  2.設(shè)計(jì)描述</b></p><p>  詞法分析:逐個(gè)讀入源程序字符并按照構(gòu)詞規(guī)則切分成一系列單詞。單詞是語(yǔ)言中具有獨(dú)立意義的最小單位,包括保留字、標(biāo)識(shí)符、運(yùn)算符、標(biāo)點(diǎn)符號(hào)和常量等。詞法分析是編譯過(guò)程中的一個(gè)階段,在語(yǔ)法分析前進(jìn)行 。也可以和語(yǔ)法分析結(jié)合在一起作為一遍,由語(yǔ)法分析程

12、序調(diào)用詞法分析程序來(lái)獲得當(dāng)前單詞供語(yǔ)法分析使用。</p><p>  表1 各種單詞符號(hào)對(duì)應(yīng)類型表</p><p><b>  3.函數(shù)模塊</b></p><p>  LexAnalyz()函數(shù)</p><p><b>  實(shí)現(xiàn)整個(gè)分析的過(guò)程</b></p><p>  2

13、.main主函數(shù):</p><p>  主要實(shí)驗(yàn)將輸入的字符串存進(jìn)token中,和組織其他函數(shù)已完成功能。</p><p><b>  print()函數(shù)</b></p><p>  將識(shí)別的結(jié)果打印出來(lái)。</p><p><b>  4.設(shè)計(jì)源碼</b></p><p> 

14、 #include<stdio.h></p><p>  #include<string.h></p><p>  #include<ctype.h></p><p>  #include<cstdlib></p><p>  using namespace std;</p>&l

15、t;p>  const char*reserchar[7]={"int", "if", "else", "while", "for", "read", "write" }; //關(guān)鍵字 </p><p>  const char*rememchar[25]={"

16、;","$SYMBOL", "$CNSTANT", "$INT", "$IF", "$ELSE", "$WHILE", "$FOR", "$READ", "$WRITE", "$ADD", "$SUB&q

17、uot;, "$MUL", "$DIV", "$L", "$LE", "$G", "$GE", "$NE", "$E", "$ASSIGN", "$LPAR", "$RPAR", "$

18、COM", "$SEM" }; //助記符 </p><p>  void LexAnalyz();</p><p>  void Print();</p><p>  char prog[100];</p><p>  char token[10];</p><p>  int syn,

19、p;</p><p><b>  char ch;</b></p><p>  int main()</p><p><b>  {</b></p><p><b>  char sym;</b></p><p><b>  Print();&

20、lt;/b></p><p><b>  p=0;</b></p><p><b>  do {</b></p><p>  ch=getchar();</p><p>  prog[p++]=ch;</p><p>  }while( ch!='#'

21、);</p><p><b>  p=0; </b></p><p><b>  do{ </b></p><p>  LexAnalyz(); </p><p>  if( syn==-1 )</p><p>  printf("error\n");

22、 </p><p>  else if( syn!=0 ){ </p><p>  printf("%s %d %s\n", token, syn, rememchar[syn]); </p><p><b>  }</b></p><p>  // system("pause

23、") ; </p><p>  }while( syn!=0 );</p><p>  system("pause");</p><p><b>  return 0;</b></p><p><b>  }</b></p><p&g

24、t;  void LexAnalyz()</p><p><b>  {</b></p><p>  int j=0,i=0;</p><p><b>  syn=0;</b></p><p>  for( i=0; i<10; i++)</p><p>  token

25、[i]='\0';</p><p>  ch=prog[p++];</p><p>  while( ch ==' ' )</p><p>  ch= prog[p++];</p><p>  if(isalpha( ch )){</p><p>  while(isalpha( ch

26、) || isdigit( ch )){</p><p>  token[j++] = ch;</p><p>  ch= prog[p++] ; </p><p><b>  }</b></p><p>  ch=prog[p--];</p><p>  syn=1; </p>

27、<p>  for( i=0; i<7; i++ )</p><p>  if( strcmp( token, reserchar[i])==0 ){</p><p><b>  syn=i+3;</b></p><p><b>  break;</b></p><p><

28、b>  }</b></p><p><b>  }</b></p><p>  else if( isdigit( ch ) ){</p><p>  while( isdigit( ch ) ){</p><p>  token[j++] = ch;</p><p>  ch

29、= prog[p++]; </p><p><b>  }</b></p><p>  ch= prog[p--];</p><p><b>  syn=2;</b></p><p><b>  }</b></p><p><b>  else

30、{</b></p><p>  switch( ch ){</p><p>  case '<': token[j++] = ch; ch = prog[p++]; </p><p>  if( ch == '=' ){</p><p>  token[j++] = ch;</p>

31、;<p>  syn=15; } </p><p><b>  else{</b></p><p><b>  syn=14; </b></p><p>  ch= prog[p--];</p><p><b>  } </b></p>&

32、lt;p><b>  break;</b></p><p>  case '>': token[j++] = ch; ch = prog[p++]; </p><p>  if( ch == '=' ){</p><p>  token[j++] = ch;</p><p> 

33、 syn=17; } </p><p><b>  else{</b></p><p><b>  syn=16; </b></p><p>  ch= prog[p--];}</p><p><b>  break;</b></p><p>  ca

34、se '!': token[j++] = ch; ch = prog[p++];</p><p>  if(ch == '='){</p><p>  token[j++] = ch;</p><p>  syn=18; }</p><p><b>  else</b></p>

35、;<p>  ch= prog[p--];</p><p><b>  break;</b></p><p>  case '=': token[j++] = ch; ch = prog[p++]; </p><p>  if( ch == '=' ){</p><p>  

36、token[j++] = ch;</p><p>  syn=19; } </p><p><b>  else{</b></p><p>  ch= prog[p--];</p><p><b>  syn=20; </b></p><p><b>  }<

37、;/b></p><p><b>  break; </b></p><p>  case '+': token[j++]=ch; syn=10; break;</p><p>  case '-': token[j++]=ch; syn=11; break; &

38、lt;/p><p>  case '*': token[j++]=ch; syn=12; break;</p><p>  case '/': token[j++]=ch; syn=13; break;</p><p>  case '(': token[j++]=ch; syn=21; break;</p>

39、<p>  case ')': token[j++]=ch; syn=22; break;</p><p>  case ',': token[j++]=ch; syn=23; break;</p><p>  case ';': token[j++]=ch; syn=24; break;</p><p>

40、  case '#': syn=0; break;</p><p>  default: syn=-1;</p><p><b>  }</b></p><p><b>  }</b></p><p><b>  }</b></p><p&g

41、t;  void Print()</p><p><b>  {</b></p><p>  for(int i=0; i<=25; i++ )</p><p>  printf("*");</p><p>  printf("\n");</p><p&g

42、t;  for(int i=0; i<=8; i++ )</p><p>  printf(" ");</p><p>  printf("詞法分析器\n");</p><p>  for(int i=0; i<=25; i++ )</p><p>  printf("*"

43、;);</p><p>  printf("\n");</p><p>  printf("輸入一串句子以#+enter鍵結(jié)束:\n");</p><p><b>  }</b></p><p>  5.測(cè)試樣例與測(cè)試結(jié)果</p><p>  輸入的是字符串

44、;識(shí)別出的是單詞,且輸出單詞的類別編碼和助記符。</p><p><b>  結(jié)論</b></p><p>  通過(guò)本次課程設(shè)計(jì)的練習(xí),熟悉了用C++語(yǔ)言編寫詞法分析器的過(guò)程,掌握了詞法分析器的原理以及功能。詞法分析是編譯過(guò)程中的一個(gè)階段,在語(yǔ)法分析前進(jìn)行。也可以和語(yǔ)法分析結(jié)合在一起作為一遍,由語(yǔ)法分析程序調(diào)用詞法分析程序來(lái)獲得當(dāng)前單詞供語(yǔ)法分析使用。詞法分析程序的主

45、要任務(wù):讀源程序,產(chǎn)生單詞符號(hào)。</p><p>  詞法分析程序的其他任務(wù):濾掉空格,跳過(guò)注釋、換行符追蹤換行標(biāo)志,復(fù)制出錯(cuò)源程序,宏展開,等等等等。</p><p>  詞法分析工作從語(yǔ)法分析工作獨(dú)立出來(lái)的原因:簡(jiǎn)化設(shè)計(jì),改進(jìn)編譯效率,增加編譯系統(tǒng)的可移植性 。</p><p>  而且從劃分關(guān)鍵字,運(yùn)算符,界符,標(biāo)識(shí)符和常量,才發(fā)現(xiàn)數(shù)字,字母及符號(hào)組合有很多很

46、多,無(wú)法全部枚舉,所以在新建的文本文檔中只象征性的列出幾種符號(hào),但這并不影響此法分析結(jié)果的完成。</p><p>  總之,通過(guò)本次實(shí)驗(yàn),一點(diǎn)點(diǎn)分析詞法分析器的功能,并努力實(shí)現(xiàn)它,掌握了課程設(shè)計(jì)內(nèi)容的同時(shí)也鍛煉了自己分析解決問題的能力以及編程能力,收獲頗豐!</p><p>  設(shè)計(jì)二: 算符優(yōu)先語(yǔ)法分析</p><p>  1.課程設(shè)計(jì)的目的和要求</p&g

47、t;<p>  1.1 課程設(shè)計(jì)的目的</p><p>  本次設(shè)計(jì)的時(shí)間為一個(gè)月,目的是件通過(guò)使用高級(jí)語(yǔ)言實(shí)現(xiàn)部分加強(qiáng)對(duì)編譯技術(shù)和理論的理解。設(shè)計(jì)的題目要求具有一定的規(guī)模,應(yīng)蘊(yùn)含本課程內(nèi)容和世紀(jì)應(yīng)用相關(guān)的主要技術(shù)。</p><p>  1.2 課程設(shè)計(jì)的要求</p><p>  文法使用產(chǎn)生式來(lái)定義;</p><p>  用大

48、寫字母和小寫字母分別表示非終結(jié)符和終結(jié)符;產(chǎn)生式使用->;</p><p>  文法中的空字符串統(tǒng)一使用@表示;</p><p>  分別給出每一個(gè)終結(jié)符的FIRSTVT 集和LASTVT集;</p><p>  版定給定的文法是否是算符優(yōu)先文法;</p><p>  畫出算符優(yōu)先關(guān)系表;、</p><p>  

49、給定富豪穿是否是文法中的句子,分析過(guò)程用分析表格的方式打印出來(lái)。</p><p><b>  2.設(shè)計(jì)描述</b></p><p>  本次實(shí)驗(yàn)使用windows vista操作系統(tǒng)下visiual c++6.0平臺(tái),使用c語(yǔ)言,利用讀文件方式將待分析的文法讀入到程序中,通過(guò)定義數(shù)組和結(jié)構(gòu)體作為具有一定意義或關(guān)系的表或棧,存放FIRSTVT 、LASTVT、算符有限關(guān)

50、系表的元素。</p><p>  系統(tǒng)能夠?qū)τ晌募x入的文法進(jìn)行分析,構(gòu)造出FIRSTVT LASTVT表以及算符優(yōu)先關(guān)系表。可以根據(jù)構(gòu)造的有限關(guān)系表對(duì)輸入的任意字符串進(jìn)行分析,判斷是否為本文法的句子,若是則打印歸約過(guò)程。結(jié)果顯示到DOS界面上。</p><p>  2.1 自底向上分析方法的描述:</p><p>  對(duì)輸入的字符串自左向右進(jìn)行掃描,并將輸入字符逐

51、個(gè)移入棧中,邊移入邊分析,一旦棧頂富豪穿形成摸個(gè)矩形的句柄時(shí)(該句柄對(duì)應(yīng)某個(gè)產(chǎn)生式的右部),就用該產(chǎn)生式的非終結(jié)符代替郵編的文法字符串,這一過(guò)程成為歸約。重復(fù)這一過(guò)程,知道棧中只剩下文法的開始負(fù)責(zé)分析成功。</p><p>  2.2算符優(yōu)先文法的描述:</p><p>  之規(guī)定酸腐之間的優(yōu)先關(guān)系,也就是說(shuō)只考慮終結(jié)符之間的有限關(guān)系。由于算符優(yōu)先分析不考慮終結(jié)符之間的優(yōu)先關(guān)系,在歸約過(guò)程

52、中只要找到最左素短語(yǔ)就可以規(guī)約了。</p><p>  如給定一個(gè)文法G[S]:</p><p><b>  S->#E#</b></p><p><b>  E->E+T</b></p><p><b>  E->T</b></p><p

53、><b>  T->T*F</b></p><p><b>  T->F</b></p><p><b>  F->(E)</b></p><p><b>  F->i</b></p><p>  利用算符優(yōu)先文法分析過(guò)程處理

54、如下:</p><p>  計(jì)算給定文法中任意兩個(gè)終結(jié)符對(duì)(a,b)之間的優(yōu)先關(guān)系,首先計(jì)算兩個(gè)幾何</p><p>  FIRSTVT(B)={b|B->b.......或B->Cb.......}</p><p>  LASTVT(B)={a|B->....a或B->.......aC}</p><p>  2.構(gòu)

55、造firstvt和lastvt集</p><p>  下表是FIRSTVT集和LASTVT集</p><p>  根據(jù)firstvt 和lastvt集構(gòu)造算符優(yōu)先關(guān)系表</p><p>  下表是算符優(yōu)先關(guān)系表</p><p><b>  3.進(jìn)行移進(jìn)和歸約</b></p><p><b&g

56、t;  概要設(shè)計(jì)和詳細(xì)設(shè)計(jì)</b></p><p><b>  4.1功能結(jié)構(gòu)</b></p><p><b>  4.2模塊的劃分</b></p><p><b>  文件的導(dǎo)入:</b></p><p>  int ReadFile(char exp[][col

57、]); </p><p>  文法以文件的形式輸入而且保存在二維數(shù)組中。</p><p>  Firstvt和Lastvt集的構(gòu)造:</p><p>  void FirstVT(char exp[][col],char firstvt[][col],const int&exp_len,const int&first_len); </

58、p><p>  //構(gòu)造firstvt集</p><p>  void LastVT(char exp[][col],char lastvt[][col],const int&exp_len,const int&first_len); </p><p>  //構(gòu)造lastvt集</p><p>  算符優(yōu)先關(guān)系表的構(gòu)

59、造:</p><p>  bool OperPriTable(char exp[][col],char opertable[][col],char firstvt[][col],char lastvt[][col],int exp_len,int first_len,int&oper_len);</p><p>  //構(gòu)造算符優(yōu)先表如果不是算符優(yōu)先文法則返回false。</

60、p><p><b>  歸約的過(guò)程:</b></p><p>  bool GuiYue(char input[],char opertable[][col],char exp[][col],int oper_len,int exp_len); </p><p>  //在其他的小函數(shù)的幫助下歸約 </p><p><

61、;b>  主函數(shù)</b></p><p>  Main函數(shù),將模塊組合,完成整個(gè)設(shè)計(jì)的功能。</p><p><b>  5.源代碼</b></p><p>  #include<iostream></p><p>  #include<fstream></p>&

62、lt;p>  #include<stack></p><p>  #define col 50</p><p>  #define row 50 </p><p>  using namespace std; </p><p>  struct Element</p><p><b>  

63、{</b></p><p>  char nont; //非終結(jié)符 </p><p>  char ternal; //終結(jié)符 </p><p><b>  };</b></p><p>  void Init(char exp[][col],char firstvt[][col],char last

64、vt[][col],int&exp_len,int&first_len);</p><p>  //first和lastvt的非終結(jié)符</p><p>  bool FindRecord(Element record[],Element &a, int &r); </p><p>  //在構(gòu)造first集時(shí),判斷是否已經(jīng)添加了A

65、->a </p><p>  bool TerminalJug(char ch); //判斷是否是終結(jié)符 </p><p>  void FirstVT(char exp[][col],char firstvt[][col],const int&exp_len,const int&first_len); //構(gòu)造firstvt集&l

66、t;/p><p>  void LastVT(char exp[][col],char lastvt[][col],const int&exp_len,const int&first_len); //構(gòu)造lastvt集</p><p>  int ReadFile(char exp[][col]); //文法以文件的形式輸入</p><p&g

67、t;  bool InsertTable(char opertable[][col],int&i,int&j,char oper); </p><p>  int FindItem( char firstvt[][col],int first_len,char ch); </p><p>  //判斷此終結(jié)符是否在firsrvt集 0列中</p>&l

68、t;p>  int FindPosition(char term[],const char&c1,const int&term_len) ;</p><p>  bool OperPriTable(char exp[][col],char opertable[][col],char firstvt[][col],char lastvt[][col],int exp_len,int first

69、_len,int&oper_len); //構(gòu)造算符優(yōu)先表</p><p>  void Print(char array[][col],int r); //輸出查看 </p><p>  bool GuiYue(char input[],char opertable[][col],char exp[][col],int oper_len,int exp_len);

70、 //歸約函數(shù) </p><p>  char Match(char sk[],int s,int e,char exp[][col],int&exp_len); </p><p>  //在產(chǎn)生式中查找匹配式子并返回規(guī)約后的非終結(jié)符 </p><p>  int main()</p><p><b>  {</b&

71、gt;</p><p>  char firstvt[row][col]={'\0'},lastvt[row][col]={'\0'};</p><p>  char exp[row][col]={'\0'},opertable[row][col]={'\0'};</p><p>  int first

72、_len,exp_len,oper_len; </p><p>  exp_len=ReadFile(exp);</p><p>  Init(exp,firstvt,lastvt,exp_len,first_len);</p><p>  FirstVT(exp,firstvt,exp_len,first_len);</p><p>  c

73、out<<"\n\t文法中非終結(jié)符的firstvt集: "<<endl;</p><p>  cout<<"**************************************"<<endl; </p><p>  Print(firstvt,first_len);</p>

74、<p>  LastVT(exp,lastvt,exp_len,first_len);</p><p>  cout<<"\n\t文法中非終結(jié)符的lastvt集: "<<endl;</p><p>  cout<<"**************************************"&

75、lt;<endl;</p><p>  Print(lastvt,first_len);</p><p>  if( OperPriTable(exp,opertable,firstvt,lastvt,exp_len,first_len,oper_len) ){</p><p>  cout<<"\n\t算符優(yōu)先分析表:

76、"<<endl;</p><p>  cout<<"**************************************"<<endl;</p><p>  Print(opertable,oper_len);</p><p>  cout<<endl;</p>&l

77、t;p>  char input[col]={'\0'};</p><p>  cout<<"輸入要分析的串 (以#+enter鍵結(jié)束):"<<endl;</p><p>  cin>>input;</p><p>  cout<<"\t\t分析過(guò)程描述

78、 "<<endl;</p><p>  if( GuiYue(input,opertable,exp,oper_len,exp_len) ){</p><p>  cout<<"該句型符合算符優(yōu)先文法"<<endl;</p><p><b>  }</b></p>

79、;<p><b>  else</b></p><p>  cout<<"該句型不符合算符優(yōu)先文法,是個(gè)錯(cuò)誤的句子"<<endl;</p><p><b>  }</b></p><p>  //C:\\Users\\peanut\\Desktop\\編譯原理\\文

80、法.txt </p><p><b>  return 0;</b></p><p><b>  }</b></p><p>  void FirstVT(char exp[][col],char firstvt[][col],const int&exp_len,const int&first_len)

81、 //構(gòu)造firstvt集 </p><p><b>  {</b></p><p>  int i,recor_len;</p><p>  stack<Element> expstack; </p><p>  Element temp;</p><p>  Element r

82、ecord[col];</p><p>  recor_len=0;</p><p>  for( i=0; i<exp_len; i++){ //第一次掃描得出的A->a入棧 </p><p>  for( int j=3; exp[i][j]!='\0'; j++){</p><p&g

83、t;  if( TerminalJug( exp[i][j] ) ){ //判斷是否是終結(jié)符 </p><p>  temp.nont=exp[i][0];</p><p>  temp.ternal=exp[i][j];</p><p>  if( recor_len==0 || !FindRecord(record,temp, rec

84、or_len) ){ //沒記錄就添加到記錄 </p><p>  expstack.push(temp);</p><p>  record[recor_len].nont=temp.nont;</p><p>  record[recor_len].ternal=temp.ternal;</p><p>  recor_len++;<

85、;/p><p><b>  }</b></p><p><b>  break; </b></p><p><b>  }</b></p><p><b>  }</b></p><p><b>  }</b>

86、</p><p>  Element DL,DE; </p><p>  int location[col]; //用來(lái)記錄firstvt集中每行的長(zhǎng)度 </p><p>  for( i=0; i<first_len; i++)</p><p>  location[i]=1;</

87、p><p>  while( !expstack.empty() ){</p><p>  DE=expstack.top();</p><p>  // cout<<DE.nont<<"->"<<DE.ternal<<endl; </p><p>  expstack.p

88、op();</p><p>  for( i=0; i<first_len; i++){ //將終結(jié)符加到相應(yīng)firstvt集中 </p><p>  if( firstvt[i][0] == DE.nont ){</p><p>  int len=location

89、[i]; </p><p>  firstvt[i][len]=DE.ternal;</p><p>  location[i]++;</p><p><b>  break;</b></p><p><b>  }</b></p><p><b>  }</

90、b></p><p>  for( i=0; i<exp_len; i++){ //找出能推出相應(yīng)非終結(jié)符的產(chǎn)生式 規(guī)律二: A->B </p><p>  if( exp[i][3]==DE.nont ){</p><p>  DL.nont=exp[i][0];</p>

91、;<p>  DL.ternal=DE.ternal;</p><p>  if( !FindRecord(record, DL, recor_len) ){//沒記錄就添加到記錄 </p><p>  expstack.push(DL);</p><p>  record[recor_len].nont=DL.nont;</p><

92、;p>  record[recor_len].ternal=DL.ternal;</p><p>  recor_len++;</p><p><b>  }</b></p><p><b>  }</b></p><p><b>  }</b></p>&

93、lt;p><b>  }</b></p><p><b>  }</b></p><p>  void LastVT(char exp[][col],char lastvt[][col],const int&exp_len,const int&first_len) //構(gòu)造lastvt集 </p>&l

94、t;p><b>  {</b></p><p>  int i,j,k;</p><p><b>  char ch;</b></p><p>  char temp[row][col]={'\0'};</p><p>  for( i=0; i<exp_len; i++

95、){</p><p>  for( j=0; exp[i][j]!='\0'; j++)</p><p>  temp[i][j]=exp[i][j]; </p><p><b>  j--;</b></p><p>  for( k=3; k<j; k++,j--){</p><

96、;p>  ch=temp[i][k];</p><p>  temp[i][k]=temp[i][j];</p><p>  temp[i][j]=ch;</p><p><b>  }</b></p><p><b>  }</b></p><p>  FirstVT

97、(temp,lastvt,exp_len,first_len); </p><p><b>  }</b></p><p>  void Init(char exp[][col],char firstvt[][col],char lastvt[][col],int&exp_len,int&first_len)</p><p>

98、  //first和lastvt的非終結(jié)符</p><p><b>  {</b></p><p>  first_len=0;</p><p>  for( int i=0; i<exp_len; i++){</p><p>  if( TerminalJug(exp[i][0]) == false &&

99、amp; FindItem(firstvt,first_len,exp[i][0]) == -1 ){ </p><p>  firstvt[first_len][0]=exp[i][0];</p><p>  lastvt[first_len][0]=exp[i][0];</p><p>  first_len++;</p><p&g

100、t;<b>  }</b></p><p><b>  }</b></p><p><b>  }</b></p><p>  int ReadFile(char exp[][col]) //文法以文件的形式輸入 </p><p><b>  {</b>

101、;</p><p>  ifstream fin;</p><p><b>  int i;</b></p><p>  char address[30];</p><p>  cout<<"輸入文件的路徑:"<<endl; </p><p>  cin

102、>>address;</p><p>  cout<<endl;</p><p>  fin.open(address,ios::in);</p><p><b>  if(!fin)</b></p><p>  cout<<"file can not be opened!&

103、quot;<<endl;</p><p><b>  else{</b></p><p>  cout<<"\t文法的中表達(dá)式:"<<endl;</p><p>  cout<<"*****************************"<<e

104、ndl; </p><p>  for( i=0; !fin.eof(); i++){ //表達(dá)式的輸入 </p><p>  fin>>exp[i];</p><p>  cout<<exp[i]<<endl;</p><p><b>  }</b>&l

105、t;/p><p>  fin.close();</p><p><b>  }</b></p><p>  return i; //返回表達(dá)式的行數(shù) </p><p><b>  }</b></p><p>  int FindItem(char first

106、vt[][col],int first_len,char ch) //判斷此終結(jié)符是否在firsrvt集 0列中 </p><p><b>  {</b></p><p>  for(int i=0; i<first_len; i++)</p><p>  if( firstvt[i][0]==ch )</p><p&

107、gt;<b>  return i;</b></p><p>  return -1; </p><p><b>  }</b></p><p>  bool FindRecord(Element record[],Element &a, int &recor_len) //判斷是否有記錄

108、 </p><p><b>  {</b></p><p>  for( int k=0; k<recor_len; k++)</p><p>  if( record[k].nont==a.nont && record[k].ternal==a.ternal ) </p><p>  retur

109、n true;</p><p>  return false; </p><p><b>  }</b></p><p>  bool TerminalJug(char ch)</p><p><b>  {</b></p><p>  if( ch>='A&#

110、39; && ch<='Z') </p><p>  return false;</p><p>  return true;</p><p><b>  }</b></p><p>  bool OperPriTable(char exp[][col],char opertable

111、[][col],char firstvt[][col],char lastvt[][col],int exp_len,int first_len,int&oper_len) </p><p>  //構(gòu)造算符優(yōu)先表 </p><p><b>  {</b></p><p>  char term[col]={'\0'}

112、;</p><p>  int term_len=0,i,j,k;</p><p>  int r,c,p;</p><p>  for( i=0; i<exp_len; i++) //得到所有終結(jié)符 </p><p>  for( j=3; exp[i][j]!='\0'; j++)</p><

113、;p>  if( TerminalJug(exp[i][j]) && !FindPosition(term,exp[i][j],term_len) )</p><p>  term[term_len++]=exp[i][j];</p><p>  oper_len=term_len+1; </p><p>  f

114、or( i=0; i<term_len+1; i++)</p><p>  for( j=0; j<term_len+1; j++)</p><p>  opertable[i][j]=' ';</p><p>  for( i=1; i<term_len+1; i++){ //初始化表頭 </p><p>

115、  opertable[i][0]=term[i-1]; </p><p>  opertable[0][i]=term[i-1];</p><p><b>  } </b></p><p>  for( i=0; i<exp_len; i++){ </p><p>  for( j=5; exp[i][

116、j]!='\0'; j++){</p><p><b>  // 等于關(guān)系</b></p><p>  if(TerminalJug(exp[i][j-2])&&!TerminalJug(exp[i][j-1])&& TerminalJug(exp[i][j]) ){</p><p>  //

117、cout<<exp[i][j-2]<<' '<<exp[i][j-1]<<' '<<exp[i][j]<<endl; </p><p>  r=FindPosition(term,exp[i][j-2],term_len); </p><p>  c=FindPosition(term,

118、exp[i][j],term_len);</p><p>  if( !InsertTable(opertable, r, c, '=') ){</p><p>  cout<<"不是算符優(yōu)先文法!"<<endl;</p><p>  return false;</p><p>&l

119、t;b>  } </b></p><p><b>  }</b></p><p><b>  }</b></p><p><b>  }</b></p><p>  for( i=0; i<exp_len; i++){ </p><

120、p>  for( j=4; exp[i][j]!='\0'; j++){</p><p>  if( TerminalJug(exp[i][j-1]) && TerminalJug(exp[i][j]) ){ //相鄰兩個(gè)終結(jié)符的等于關(guān)系</p><p>  r=FindPosition(term,exp[i][j-1],term_len); &l

121、t;/p><p>  c=FindPosition(term,exp[i][j],term_len);</p><p>  if( !InsertTable(opertable, r, c, '=') ){</p><p>  cout<<"不是算符優(yōu)先文法!"<<endl;</p><p&

122、gt;  return false;</p><p><b>  } </b></p><p><b>  }</b></p><p>  // Print(opertable,term_len+1);</p><p>  if( TerminalJug(exp[i][j-1]) &&

123、amp; !TerminalJug(exp[i][j]) ){ //小于關(guān)系</p><p>  // cout<<exp[i][j-1]<<' '<<exp[i][j]<<endl;</p><p>  p=FindItem( firstvt,first_len,exp[i][j] ); //在firstvt集中

124、的行標(biāo)。 </p><p>  r=FindPosition(term,exp[i][j-1],term_len);</p><p>  for( k=1; firstvt[p][k]!='\0'; k++ ){</p><p>  // cout<<firstvt[p][k]<<' ';</p>

125、<p>  c=FindPosition(term,firstvt[p][k],term_len);</p><p>  //cout<<r<<' '<<c<<endl; </p><p>  if( !InsertTable(opertable, r, c, '<') ){ &

126、lt;/p><p>  cout<<"不是算符優(yōu)先文法!"<<endl;</p><p>  return false;</p><p><b>  } </b></p><p><b>  } </b></p><p>

127、<b>  } </b></p><p>  // Print(opertable,term_len+1);</p><p>  if( !TerminalJug(exp[i][j-1]) && TerminalJug(exp[i][j]) ){ //大于關(guān)系</p><p>  p=FindItem( lastvt,firs

128、t_len,exp[i][j-1] ); </p><p>  //在lastvt集中的行標(biāo)。</p><p>  c=FindPosition(term,exp[i][j],term_len);</p><p>  for( k=1; lastvt[p][k]!='\0'; k++ ){</p><p>

129、  r=FindPosition(term,lastvt[p][k],term_len);</p><p>  if( !InsertTable(opertable, r, c , '>') ){ </p><p>  cout<<"不是算符優(yōu)先文法!"<<endl;</p><p>  

130、return false;</p><p><b>  }</b></p><p><b>  }</b></p><p><b>  }</b></p><p>  // Print(opertable,term_len+1);</p><p><

131、;b>  }</b></p><p><b>  } </b></p><p>  return true; </p><p><b>  }</b></p><p>  int FindPosition(char term[],const char&am

132、p;c1,const int&term_len) </p><p>  //查找字符在算符優(yōu)先表中的下標(biāo) </p><p><b>  {</b></p><p><b>  int i;</b></p><p>  for( i=0; i<term_len; i++ )<

133、/p><p>  if( term[i]==c1 )</p><p>  return i+1;</p><p><b>  return 0;</b></p><p><b>  }</b></p><p>  bool InsertTable(char opertable[]

134、[col],int&i,int&j,char oper)</p><p><b>  {</b></p><p>  if( opertable[i][j]!=' ' )</p><p>  return false; </p><p>  opertable[i][j]=oper;&

135、lt;/p><p>  return true;</p><p><b>  }</b></p><p>  void Print(char array[][col],int r)</p><p><b>  {</b></p><p><b>  int i,j;&l

136、t;/b></p><p>  for( i=0; i<r; i++){</p><p>  cout<<array[i][0]<<' ';</p><p>  for( j=1; array[i][j]!='\0'; j++)</p><p>  cout<<a

137、rray[i][j]<<' '; </p><p>  cout<<endl;</p><p><b>  } </b></p><p><b>  }</b></p><p>  bool GuiYue(char input[],char opertable

138、[][col],char exp[][col],int oper_len,int exp_len)</p><p><b>  {</b></p><p>  char sk[col]={'\0'};</p><p>  char*str=input;</p><p>  int k,j,pa,pb,i,

139、p;</p><p>  char a,b,op;</p><p>  cout.setf(ios::left);</p><p>  cout.width(15);</p><p>  cout<<"下推棧";</p><p>  cout.unsetf(ios::left);<

140、/p><p>  cout.width(15);</p><p>  cout<<"輸入串";</p><p>  cout.width(10);</p><p>  cout<<"動(dòng)作"<<endl; </p><p>  sk[0]='

141、;#'; </p><p><b>  k=0; </b></p><p>  for( i=0; input[i]!='\0'; ){ </p><p>  b=input[i];</p><p>  if( TerminalJug(sk[k]) )</p><p>&

溫馨提示

  • 1. 本站所有資源如無(wú)特殊說(shuō)明,都需要本地電腦安裝OFFICE2007和PDF閱讀器。圖紙軟件為CAD,CAXA,PROE,UG,SolidWorks等.壓縮文件請(qǐng)下載最新的WinRAR軟件解壓。
  • 2. 本站的文檔不包含任何第三方提供的附件圖紙等,如果需要附件,請(qǐng)聯(lián)系上傳者。文件的所有權(quán)益歸上傳用戶所有。
  • 3. 本站RAR壓縮包中若帶圖紙,網(wǎng)頁(yè)內(nèi)容里面會(huì)有圖紙預(yù)覽,若沒有圖紙預(yù)覽就沒有圖紙。
  • 4. 未經(jīng)權(quán)益所有人同意不得將文件中的內(nèi)容挪作商業(yè)或盈利用途。
  • 5. 眾賞文庫(kù)僅提供信息存儲(chǔ)空間,僅對(duì)用戶上傳內(nèi)容的表現(xiàn)方式做保護(hù)處理,對(duì)用戶上傳分享的文檔內(nèi)容本身不做任何修改或編輯,并不能對(duì)任何下載內(nèi)容負(fù)責(zé)。
  • 6. 下載文件中如有侵權(quán)或不適當(dāng)內(nèi)容,請(qǐng)與我們聯(lián)系,我們立即糾正。
  • 7. 本站不保證下載資源的準(zhǔn)確性、安全性和完整性, 同時(shí)也不承擔(dān)用戶因使用這些下載資源對(duì)自己和他人造成任何形式的傷害或損失。

評(píng)論

0/150

提交評(píng)論