⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 program4.pas

📁 PL/0语法分析程序原代码(TURBOL PASCAL)
💻 PAS
字号:
program PL0(input,output);
(* PL/0 compiler syntax analysis only *)
const norw = 11;        (* no. of reserved words *)
  txmax = 100;          (* length of identifier table *)
  nmax = 14;            (* max. no of digits in numbers *)
  al = 10;              (* length of identifiers *)
  chsetsize = 128;      (* for ASCII character set *)
type symbol =
  (nul,ident,number,plus,minus,times,slash,oddsym,
   eql,neq,lss,leq,gtr,geq,lparen,rparen,comma,semicolon,
   period,becomes,beginsym,endsym,ifsym,thensym,
   whilesym,dosym,callsym,constsym,varsym,procsym);
  alfa = packed array [1..al] of char;
  objekt = (constant,variable,prozedure);
var ch:char;            (* last character read *)
  sym: symbol;          (* last symbol read *)
  id: alfa;             (* last identifier read *)
  num: integer;         (* last number read *)
  cc: integer;          (* character count *)
  ll: integer;          (* line length *)
  kk: integer;
  line: array[1..81] of char;
  a: alfa;       
  word: array[1..norw] of alfa;
  wsym: array[1..norw] of symbol;
  ssym: array [char] of symbol;
  table: array [0..txmax] of
           record name: alfa;
                  kind: objekt;
           end;
  inf: text;

procedure error(n: integer);
begin writeln(' ':cc-1,'^',n:2);
  halt;
end(* error *);
procedure getsym;
  var i,j,k:integer;
  procedure getch;
  begin if cc = ll then
    begin if eof(inf) then
        begin write('program incomplete'); halt
        end;
      ll:= 0; cc:= 0;
      while not eoln(inf) do
        begin ll:=ll+1; read(inf,ch); write(ch); line[ll]:= ch
        end;
      writeln; ll:=ll+1; readln(inf); line[ll]:=' '
    end;
    cc:=cc+1; ch:=line[cc]
  end(* getch *);
begin (* getsym *)
  while ch=' ' do getch;
  if ch in ['a'..'z'] then
  begin (* identifier or reserved word *) k:=0;
    repeat if k < al then
      begin k:=k+1; a[k]:=ch
      end;
      getch
    until not (ch in ['a'..'z','0'..'9']);
    if k>=kk then kk:=k else
      repeat a[kk]:=' '; kk:=kk-1
      until kk=k;
    id:=a; i:=1; j:=norw;
    repeat k:=(i+j)div 2;
      if id<=word[k] then j:=k-1;
      if id>=word[k] then i:=k+1
    until i>j;
    if i-1>j then sym:=wsym[k] else sym:=ident
  end else
  if ch in ['0'..'9'] then
  begin (* number *) k:=0; num:=0; sym:=number;
    repeat num:=10*num+(ord(ch)-ord('0'));
      k:=k+1; getch
    until not (ch in ['0'..'9']);
    if k> nmax then error(30)
  end else
  if ch=':' then
  begin getch;
    if ch='=' then
    begin sym:=becomes; getch
    end else sym:=nul;
  end else
  if ch='<' then
  begin getch;
    if ch='=' then
    begin sym:=leq; getch
    end else sym:=lss
  end else
  if ch='>' then
  begin getch;
    if ch='=' then
    begin sym:=geq; getch
    end else sym:=gtr
  end else
  begin sym:=ssym[ch]; getch
  end
end(* getsym *);
procedure block(tx: integer);
  procedure enter(k:objekt);
  begin (*enter objekt into table*)
    tx:=tx+1;
    with table[tx] do
      begin name:=id;
            kind:=k;
      end
  end(*enter*);
  function position(id:alfa):integer;
    var i:integer;
  begin(* find identifier id in table *)
    table[0].name:=id; i:=tx;
    while table[i].name<>id do i:=i-1;
    position:=i
  end(* position *);
  procedure constdeclaration;
  begin if sym=ident then
    begin getsym;
        if sym = eql then
          begin  getsym;
            if sym=number then
                begin enter(constant); getsym
          end
        else error(2)
      end else error(3)
    end else error(4)
  end (* constdeclaration *);
  procedure vardeclaration;
  begin if sym=ident then
        begin enter(variable); getsym
        end else error(4)
  end(* vardeclaration *);
  procedure statement;
    var i:integer;
    procedure expression;
      procedure term;
        procedure factor;
          var i:integer;
        begin 
          if sym=ident then
          begin i:=position(id);
            if i=0 then error(11) else
            if table[i].kind = prozedure then error(21);
            getsym
          end else
          if sym=number then
          begin getsym
          end else
          if sym=lparen then
          begin getsym; expression;
            if sym=rparen then getsym else error(22)
          end
          else error(23)
      end(*factor*);
      begin(* term *) factor;
        while sym in [times,slash] do
          begin getsym; factor
          end
      end(* term *);
    begin (* expression *)
      if sym in [plus,minus] then
        begin getsym; term;
        end else term;
      while sym in [plus,minus] do
        begin getsym; term;
        end
    end(*expression*);
    procedure condition;
    begin
      if sym=oddsym then
      begin getsym; expression
      end else
      begin expression;
        if not(sym in [eql,neq,lss,leq,gtr,geq]) then
          error(20) else
        begin getsym; expression;
        end
      end
    end(*condition*);
  begin(*statement*)
    if sym=ident then
    begin i:=position(id);
      if i=0 then error(11) else
      if table[i].kind<>variable then error(12);
      getsym; if sym=becomes then getsym else error(13);
      expression;
    end else
    if sym=callsym then
    begin getsym;
      if sym<>ident then error(14) else
        begin i:=position(id);
          if i=0 then error(11) else
          if table[i].kind <> prozedure then error(15);
          getsym
        end
    end else
    if sym=ifsym then
    begin getsym; condition;
      if sym=thensym then getsym else error(16);
      statement;
    end else
    if sym=beginsym then
    begin getsym; statement;
      while sym= semicolon do
      begin
        getsym;
        statement
      end;
      if sym=endsym then getsym else error(17)
    end else
    if sym=whilesym then
    begin getsym; condition;
      if sym=dosym then getsym else error(18);
      statement
    end;
  end(*statement*);
begin(*block*)
    if sym=constsym then
    begin getsym;
      constdeclaration;
        while sym=comma do
          begin getsym; constdeclaration
          end;
        if sym=semicolon then getsym else error(5)
    end;
    if sym=varsym then
    begin getsym;
      vardeclaration;
        while sym=comma do
          begin getsym; vardeclaration
          end;
        if sym=semicolon then getsym else error(5)
    end;
    while sym=procsym do
    begin getsym;
      if sym=ident then
        begin enter(prozedure); getsym
        end
      else error(4);
      if sym=semicolon then getsym else error(5);
      block(tx);
      if sym=semicolon then getsym else error(5)
    end;
  statement
end(*block*);
begin(* main program *)
  assign(inf,'testin.pl0');
  reset(inf);
  for ch:=chr(0) to chr(chsetsize-1) do ssym[ch]:=nul;
  word[ 1]:='begin     '; word[ 2]:='call      ';
  word[ 3]:='const     '; word[ 4]:='do        ';
  word[ 5]:='end       '; word[ 6]:='if        ';
  word[ 7]:='odd       '; word[ 8]:='procedure ';
  word[ 9]:='then      '; word[10]:='var       ';
  word[11]:='while     ';
  wsym[ 1]:=beginsym; wsym[ 2]:=callsym;
  wsym[ 3]:=constsym; wsym[ 4]:=dosym;
  wsym[ 5]:=endsym;   wsym[ 6]:=ifsym;
  wsym[ 7]:=oddsym;   wsym[ 8]:=procsym;
  wsym[ 9]:=thensym;  wsym[10]:=varsym;
  wsym[11]:=whilesym;
  ssym['+']:=plus;   ssym['-']:=minus;
  ssym['*']:=times;  ssym['/']:=slash;
  ssym['(']:=lparen; ssym[')']:=rparen;
  ssym['=']:=eql;    ssym[',']:=comma;
  ssym['.']:=period; ssym['#']:=neq;
  ssym['<']:=lss;    ssym['>']:=gtr;
  ssym[';']:=semicolon;
  cc:=0; ll:=0; ch:=' '; kk:=al; getsym;
  block(0);
  if sym<>period then error(9);
end. 

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -