日韩av黄I国产麻豆传媒I国产91av视频在线观看I日韩一区二区三区在线看I美女国产在线I麻豆视频国产在线观看I成人黄色短片

歡迎訪問 生活随笔!

生活随笔

當前位置: 首頁 > 编程资源 > 编程问答 >内容正文

编程问答

【编译原理】构建一个简单的解释器(Let’s Build A Simple Interpreter. Part 9.)(笔记)语法分析(未完,先搁置了!)

發布時間:2025/3/20 编程问答 39 豆豆
生活随笔 收集整理的這篇文章主要介紹了 【编译原理】构建一个简单的解释器(Let’s Build A Simple Interpreter. Part 9.)(笔记)语法分析(未完,先搁置了!) 小編覺得挺不錯的,現在分享給大家,幫大家做個參考.

【編譯原理】讓我們來構建一個簡單的解釋器(Let’s Build A Simple Interpreter. Part 9.)

文章目錄

    • spi.py
    • spi_lexer

我記得當我在大學(很久以前)學習系統編程時,我相信唯一“真正”的語言是匯編和 C。而 Pascal 是——怎么說好一點——一種非常高級的語言不想知道幕后發生了什么的應用程序開發人員。

那時我幾乎不知道我會用 Python 編寫幾乎所有東西(并且喜歡它的每一點)來支付我的賬單,而且我還會因為我在第一篇文章中提到的原因為 Pascal 編寫解釋器和編譯器該系列的。

這些天,我認為自己是一個編程語言愛好者,我對所有語言及其獨特的功能都很著迷。話雖如此,我必須指出,我比其他語言更喜歡使用某些語言。我有偏見,我將是第一個承認這一點的人。😃

這是我之前:


好的,讓我們進入正題。以下是您今天要學習的內容:

如何解析和解釋 Pascal 程序定義。
1、如何解析和解釋復合語句(compound statements)。
2、如何解析和解釋賦值語句(assignment statements),包括變量(variables)。
3、關于符號表(symbol tables)以及如何存儲和查找變量的一些知識。

我將使用以下示例 Pascal-like 程序來介紹新概念:

BEGINBEGINnumber := 2;a := number;b := 10 * a + 10 * number / 4;c := a - - bEND;x := 11; END.

您可以說,按照本系列的前幾篇文章,到目前為止您編寫的命令行解釋器是一個很大的跳躍,但我希望這種跳躍會帶來興奮。它不再“只是”一個計算器,我們在這里變得認真了,Pascal 是認真的。😃

讓我們深入了解新語言結構的語法圖(syntax diagrams)及其相應的語法規則(grammar rules)。

在你的標記上:準備好了。放。出發!




1、我將從描述什么是 Pascal程序開始。Pascal程序由一個以點結尾的復合語句組成。下面是一個程序示例:

“BEGIN END.

我必須提示,這不是一個完整的程序定義,我們將在本系列的后面對其進行擴展。

2、什么是復合語句?復合語句(compound statement)是標有塊BEGIN和END 語句包括其它復合語句,它可以包含一個列表(list)(可能為空)。復合語句中的每一條語句,除了最后一條,都必須以分號(semicolon)結束。塊中的最后一條語句可能有也可能沒有終止分號。以下是一些有效復合語句的示例:

“BEGIN END” “BEGIN a := 5; x := 11 END” “BEGIN a := 5; x := 11; END” “BEGIN BEGIN a := 5 END; x := 11 END”

3、語句列表(statement list )是一個復合語句中的零條或多個語句的列表。有關一些示例,請參見上文。

4、語句可以是一個復合語句,一個賦值語句,或者它可以是一個空的 語句。

5、賦值語句(assignment statement)是一個變量后面跟著個ASSIGN標記(兩個字符,“:”和“=”),后面再跟著個表達式。

“a := 11” “b := a + 9 - 5 * 2

6、變量是一個標識符。我們為變量使用ID標記。標記的值將是變量的名稱,如“a”、“number”等。在以下代碼塊中,‘a’ 和 ‘b’ 是變量:

“BEGIN a := 11; b := a + 9 - 5 * 2 END”

7、一個空的聲明表示,沒有進一步的生成語法規則。我們使用empty_statement語法規則來指示 解析器中statement_list的結尾,并允許像“ BEGIN END ”中那樣的空復合語句。

8、factor規則被用來更新處理變量。

現在讓我們來看看我們完整的語法:

program : compound_statement DOTcompound_statement : BEGIN statement_list ENDstatement_list : statement| statement SEMI statement_liststatement : compound_statement| assignment_statement| emptyassignment_statement : variable ASSIGN exprempty :expr: term ((PLUS | MINUS) term)*term: factor ((MUL | DIV) factor)*factor : PLUS factor| MINUS factor| INTEGER| LPAREN expr RPAREN| variablevariable: ID

您可能已經注意到,我沒有在復合語句規則中使用星號“*” 來表示零次或多次重復,而是明確指定了statement_list規則。這是表示“零或多個”操作的另一種方式,當我們查看本系列后面的PLY等解析器生成器時,它會派上用場。我還將“( PLUS | MINUS ) factor ”子規則拆分為兩個單獨的規則。

為了支持更新的語法,我們需要對詞法分析器、解析器和解釋器進行一些更改。讓我們一一回顧這些變化。

以下是詞法分析器更改的摘要:


1、為了支持 Pascal 程序的定義、復合語句、賦值語句和變量,我們的詞法分析器需要返回新的標記:

  • BEGIN(標記復合語句的開始)
  • END(標記復合語句的結束)
  • DOT(Pascal 程序定義所需的點字符“.”的標記)
  • ASSIGN(兩個字符序列“:=”的標記)。在 Pascal 中,賦值運算符與 C、Python、Java、Rust 或 Go 等許多其他語言不同,在這些語言中,您可以使用單個字符“=”來表示賦值
  • SEMI(分號字符“;”的標記,用于標記復合語句內的語句結束)
  • ID(有效標識符的標記。標識符以字母字符開頭,后跟任意數量的字母數字字符)

2、有時,為了能夠區分以相同字符開頭的不同標記,(':' vs ':=' or '==' vs '=>' )我們需要查看輸入緩沖區而不實際消耗下一個字符。出于這個特殊目的,我引入了一個peek 方法,它將幫助我們標記賦值語句。該方法不是嚴格要求的,但我想我會在本系列的前面介紹它,它也會使get_next_token 方法更簡潔一些。它所做的只是從文本緩沖區返回下一個字符,而不增加self.pos變量。這是方法本身:

def peek(self):peek_pos = self.pos + 1if peek_pos > len(self.text) - 1:return Noneelse:return self.text[peek_pos]

3、因為 Pascal 變量和保留關鍵字都是標識符,所以我們將它們的處理合并到一個稱為_id 的方法中。它的工作方式是詞法分析器使用一系列字母數字字符,然后檢查該字符序列是否為保留字(reserved word)。如果是,則為該保留關鍵字返回一個預先構造的標記。如果它不是保留關鍵字,則返回一個新的ID令牌,其值為字符串(詞素【lexeme】)。我打賭此時你會想,“天哪,給我看看代碼。” :) 這里是:

RESERVED_KEYWORDS = {'BEGIN': Token('BEGIN', 'BEGIN'),'END': Token('END', 'END'), }def _id(self):"""Handle identifiers(標識符) and reserved keywords"""result = ''while self.current_char is not None and self.current_char.isalnum():result += self.current_charself.advance()token = RESERVED_KEYWORDS.get(result, Token(ID, result))return token

現在讓我們看看主詞法分析器方法get_next_token的變化:

def get_next_token(self):while self.current_char is not None:...if self.current_char.isalpha():return self._id()if self.current_char == ':' and self.peek() == '=':self.advance()self.advance()return Token(ASSIGN, ':=')if self.current_char == ';':self.advance()return Token(SEMI, ';')if self.current_char == '.':self.advance()return Token(DOT, '.')...

是時候看看我們閃亮的新詞法分析器的所有榮耀和動作了。從GitHub下載源代碼并從保存spi.py 文件的同一目錄啟動 Python shell :(spi.py代碼見文章目錄)也可直接見spi_lexer代碼,在pycharm中可直接運行或調試看到結果

>>> from spi import Lexer >>> lexer = Lexer('BEGIN a := 2; END.') >>> lexer.get_next_token() Token(BEGIN, 'BEGIN') >>> lexer.get_next_token() Token(ID, 'a') >>> lexer.get_next_token() Token(ASSIGN, ':=') >>> lexer.get_next_token() Token(INTEGER, 2) >>> lexer.get_next_token() Token(SEMI, ';') >>> lexer.get_next_token() Token(END, 'END') >>> lexer.get_next_token() Token(DOT, '.') >>> lexer.get_next_token() Token(EOF, None) >>>

繼續解析器更改。

以下是我們的解析器更改的摘要:

讓我們從新的AST 節點開始:

復合 AST節點表示復合語句。它在其children 變量中包含一個語句節點列表。

class Compound(AST):"""Represents a 'BEGIN ... END' block"""def __init__(self):self.children = []

Assign AST節點代表一個賦值語句。它的左變量用于存儲一個Var節點,它的右變量用于存儲由 expr 解析器方法返回的節點:

class Assign(AST):def __init__(self, left, op, right):self.left = leftself.token = self.op = opself.right = right

Var AST節點(你猜對了)代表一個變量。該self.value持有變量的名稱。

class Var(AST):"""The Var node is constructed out of ID token."""def __init__(self, token):self.token = tokenself.value = token.value

NoOp節點用于表示空語句。例如,’ BEGIN END ’ 是一個沒有語句的有效復合語句。

class NoOp(AST):pass

您還記得,語法中的每個規則在我們的遞歸下降解析器中都有一個對應的方法。這次我們添加了七個新方法。這些方法負責解析新的語言結構和構建新的AST節點。它們非常簡單:(ps. 我就喜歡作者這么說!)

def program(self):"""program : compound_statement DOT"""node = self.compound_statement()self.eat(DOT)return nodedef compound_statement(self):"""compound_statement: BEGIN statement_list END"""self.eat(BEGIN)nodes = self.statement_list()self.eat(END)root = Compound()for node in nodes:root.children.append(node)return rootdef statement_list(self):"""statement_list : statement| statement SEMI statement_list"""node = self.statement()results = [node]while self.current_token.type == SEMI:self.eat(SEMI)results.append(self.statement())if self.current_token.type == ID:self.error()return resultsdef statement(self):"""statement : compound_statement| assignment_statement| empty"""if self.current_token.type == BEGIN:node = self.compound_statement()elif self.current_token.type == ID:node = self.assignment_statement()else:node = self.empty()return nodedef assignment_statement(self):"""assignment_statement : variable ASSIGN expr"""left = self.variable()token = self.current_tokenself.eat(ASSIGN)right = self.expr()node = Assign(left, token, right)return nodedef variable(self):"""variable : ID"""node = Var(self.current_token)self.eat(ID)return nodedef empty(self):"""An empty production"""return NoOp()

我們還需要更新現有的factor方法來解析變量:

def factor(self):"""factor : PLUS factor| MINUS factor| INTEGER| LPAREN expr RPAREN| variable"""token = self.current_tokenif token.type == PLUS:self.eat(PLUS)node = UnaryOp(token, self.factor())return node...else:node = self.variable()return node

解析器的parse方法更新為通過解析程序定義來啟動解析過程:

def parse(self):node = self.program()if self.current_token.type != EOF:self.error()return node

這是我們的示例程序:

BEGINBEGINnumber := 2;a := number;b := 10 * a + 10 * number / 4;c := a - - bEND;x := 11; END.

讓我們用genastdot.py對其進行可視化(為簡潔起見,當顯示Var節點時,它只顯示節點的變量名稱,當顯示一個 Assign 節點時,它顯示 ‘:=’ 而不是顯示 ‘Assign’ 文本):

$ python genastdot.py assignments.txt > ast.dot && dot -Tpng -o ast.png ast.dot


最后,這里是所需的解釋器更改:

要解釋新的AST節點,我們需要向解釋器添加相應的訪問者方法。有四種新的訪問者方法:

訪問_Compound
訪問_分配
訪問_變量
訪問_NoOp

Compound和NoOp訪問者方法非常簡單。該visit_Compound方法遍歷它的孩子和參觀各一轉,和visit_NoOp方法不起作用。

def visit_Compound(self, node):for child in node.children:self.visit(child)def visit_NoOp(self, node):pass

在分配和瓦爾游客方法值得仔細研究。

當我們為變量賦值時,我們需要將該值存儲在某個地方以備日后需要時使用,這正是visit_Assign方法所做的:

def visit_Assign(self, node):var_name = node.left.valueself.GLOBAL_SCOPE[var_name] = self.visit(node.right)

該方法在符號表GLOBAL_SCOPE 中存儲鍵值對(變量名和與變量關聯的值)。什么是符號表?甲符號表是一個抽象數據類型(ADT用于在源代碼中追蹤各種符號)。我們現在唯一的符號類別是變量,我們使用 Python 字典來實現符號表ADT。現在我只想說,本文中符號表的使用方式非常“hacky”:它不是一個具有特殊方法的單獨類,而是一個簡單的 Python 字典,它還作為內存空間執行雙重任務。在以后的文章中,我將更詳細地討論符號表,我們還將一起刪除所有的黑客。

讓我們看一下語句“a := 3;”的AST 和visit_Assign方法之前和之后的符號表完成它的工作:

現在讓我們看一下語句“b := a + 7;”的AST

如您所見,賦值語句的右側 - “a + 7” - 引用了變量 ‘a’,因此在我們評估表達式“a + 7”之前,我們需要找出 ’ 的值a’ 是,這是visit_Var 方法的職責:

def visit_Var(self, node):var_name = node.valueval = self.GLOBAL_SCOPE.get(var_name)if val is None:raise NameError(repr(var_name))else:return val

當該方法訪問上圖AST 中的Var節點時,它首先獲取變量的名稱,然后使用該名稱作為GLOBAL_SCOPE字典中的鍵來獲取變量的值。如果它可以找到該值,則返回該值,否則會引發NameError異常。以下是在評估賦值語句“b := a + 7;”之前的符號表內容:

這些都是我們今天需要做的改變,以使我們的解釋器打勾。在主程序結束時,我們簡單地將符號表 GLOBAL_SCOPE 的內容打印到標準輸出。

讓我們從 Python 交互式 shell 和命令行中使用我們更新的解釋器作為驅動器。確保在測試之前下載了解釋器的源代碼和assignments.txt文件:

啟動你的 Python shell:

$ python >>> from spi import Lexer, Parser, Interpreter >>> text = """\ ... BEGIN ... ... BEGIN ... number := 2; ... a := number; ... b := 10 * a + 10 * number / 4; ... c := a - - b ... END; ... ... x := 11; ... END. ... """ >>> lexer = Lexer(text) >>> parser = Parser(lexer) >>> interpreter = Interpreter(parser) >>> interpreter.interpret() >>> print(interpreter.GLOBAL_SCOPE) {'a': 2, 'x': 11, 'c': 27, 'b': 25, 'number': 2}

從命令行,使用源文件作為我們解釋器的輸入:

$ python spi.py assignments.txt {'a': 2, 'x': 11, 'c': 27, 'b': 25, 'number': 2}

如果您還沒有嘗試過,現在就嘗試一下,親眼看看解釋器是否正確地完成了它的工作。

讓我們總結一下你在這篇文章中擴展 Pascal 解釋器需要做的事情:

向語法添加新規則
向詞法分析器添加新標記和支持方法并更新get_next_token 方法
向解析器添加新的AST節點以獲得新的語言結構
將與新語法規則相對應的新方法添加到我們的遞歸下降解析器中,并在必要時更新任何現有方法(因子方法,我在看著你。😃
向解釋器添加新的訪問者方法
添加用于存儲變量和查找變量的字典

在這一部分中,我不得不介紹一些“技巧”,我們將隨著系列的推進而將其刪除:

該程序的語法規則是不完整的。稍后我們將使用其他元素對其進行擴展。
Pascal 是一種靜態類型語言,您必須在使用它之前聲明一個變量及其類型。但是,正如您所看到的,本文中的情況并非如此。
到目前為止沒有類型檢查。在這一點上這沒什么大不了的,但我只是想明確地提到它。例如,一旦我們向解釋器添加更多類型,當您嘗試添加字符串和整數時,我們將需要報告錯誤。
這部分中的符號表是一個簡單的 Python 字典,它具有雙重存儲空間的功能。不用擔心:符號表是一個非常重要的主題,我將專門針對它們撰寫幾篇文章。內存空間(運行時管理)本身就是一個話題。
在我們之前文章中的簡單計算器中,我們使用正斜杠字符“/”來表示整數除法。但是,在 Pascal 中,您必須使用關鍵字div來指定整數除法(參見練習 1)。
我還特意引入了一個 hack,以便您可以在練習 2 中修復它:在 Pascal 中,所有保留關鍵字和標識符都不區分大小寫,但本文中的解釋器將它們視為區分大小寫。

為了讓你保持健康,這里有新的練習給你:

Pascal 變量和保留關鍵字不區分大小寫,這與許多其他編程語言不同,因此BEGIN、begin和BeGin它們都引用相同的保留關鍵字。更新解釋器,使變量和保留關鍵字不區分大小寫。使用以下程序對其進行測試:

BEGINBEGINnumber := 2;a := NumBer;B := 10 * a + 10 * NUMBER / 4;c := a - - bend;x := 11; END.

我之前在“hacks”部分提到我們的解釋器使用正斜杠字符“/”來表示整數除法,但它應該使用 Pascal 的保留關鍵字div進行整數除法。更新解釋器以使用div關鍵字進行整數除法,從而消除其中一種技巧。

更新解釋器,以便變量也可以以下劃線開頭,如 ‘_num := 5’。

spi.py

""" SPI - Simple Pascal Interpreter. Part 9."""############################################################################### # # # LEXER # # # ################################################################################ Token types # # EOF (end-of-file) token is used to indicate that # there is no more input left for lexical analysis (INTEGER, PLUS, MINUS, MUL, DIV, LPAREN, RPAREN, ID, ASSIGN,BEGIN, END, SEMI, DOT, EOF) = ('INTEGER', 'PLUS', 'MINUS', 'MUL', 'DIV', '(', ')', 'ID', 'ASSIGN','BEGIN', 'END', 'SEMI', 'DOT', 'EOF' )class Token(object):def __init__(self, type, value):self.type = typeself.value = valuedef __str__(self):"""String representation of the class instance.Examples:Token(INTEGER, 3)Token(PLUS, '+')Token(MUL, '*')"""return 'Token({type}, {value})'.format(type=self.type,value=repr(self.value))def __repr__(self):return self.__str__()RESERVED_KEYWORDS = {'BEGIN': Token('BEGIN', 'BEGIN'),'END': Token('END', 'END'), }class Lexer(object):def __init__(self, text):# client string input, e.g. "4 + 2 * 3 - 6 / 2"self.text = text# self.pos is an index into self.textself.pos = 0self.current_char = self.text[self.pos]def error(self):raise Exception('Invalid character')def advance(self):"""Advance the `pos` pointer and set the `current_char` variable."""self.pos += 1if self.pos > len(self.text) - 1:self.current_char = None # Indicates end of inputelse:self.current_char = self.text[self.pos]def peek(self):peek_pos = self.pos + 1if peek_pos > len(self.text) - 1:return Noneelse:return self.text[peek_pos]def skip_whitespace(self):while self.current_char is not None and self.current_char.isspace():self.advance()def integer(self):"""Return a (multidigit) integer consumed from the input."""result = ''while self.current_char is not None and self.current_char.isdigit():result += self.current_charself.advance()return int(result)def _id(self):"""Handle identifiers and reserved keywords"""result = ''while self.current_char is not None and self.current_char.isalnum():result += self.current_charself.advance()token = RESERVED_KEYWORDS.get(result, Token(ID, result))return tokendef get_next_token(self):"""Lexical analyzer (also known as scanner or tokenizer)This method is responsible for breaking a sentenceapart into tokens. One token at a time."""while self.current_char is not None:if self.current_char.isspace():self.skip_whitespace()continueif self.current_char.isalpha():return self._id()if self.current_char.isdigit():return Token(INTEGER, self.integer())if self.current_char == ':' and self.peek() == '=':self.advance()self.advance()return Token(ASSIGN, ':=')if self.current_char == ';':self.advance()return Token(SEMI, ';')if self.current_char == '+':self.advance()return Token(PLUS, '+')if self.current_char == '-':self.advance()return Token(MINUS, '-')if self.current_char == '*':self.advance()return Token(MUL, '*')if self.current_char == '/':self.advance()return Token(DIV, '/')if self.current_char == '(':self.advance()return Token(LPAREN, '(')if self.current_char == ')':self.advance()return Token(RPAREN, ')')if self.current_char == '.':self.advance()return Token(DOT, '.')self.error()return Token(EOF, None)############################################################################### # # # PARSER # # # ###############################################################################class AST(object):passclass BinOp(AST):def __init__(self, left, op, right):self.left = leftself.token = self.op = opself.right = rightclass Num(AST):def __init__(self, token):self.token = tokenself.value = token.valueclass UnaryOp(AST):def __init__(self, op, expr):self.token = self.op = opself.expr = exprclass Compound(AST):"""Represents a 'BEGIN ... END' block"""def __init__(self):self.children = []class Assign(AST):def __init__(self, left, op, right):self.left = leftself.token = self.op = opself.right = rightclass Var(AST):"""The Var node is constructed out of ID token."""def __init__(self, token):self.token = tokenself.value = token.valueclass NoOp(AST):passclass Parser(object):def __init__(self, lexer):self.lexer = lexer# set current token to the first token taken from the inputself.current_token = self.lexer.get_next_token()def error(self):raise Exception('Invalid syntax')def eat(self, token_type):# compare the current token type with the passed token# type and if they match then "eat" the current token# and assign the next token to the self.current_token,# otherwise raise an exception.if self.current_token.type == token_type:self.current_token = self.lexer.get_next_token()else:self.error()def program(self):"""program : compound_statement DOT"""node = self.compound_statement()self.eat(DOT)return nodedef compound_statement(self):"""compound_statement: BEGIN statement_list END"""self.eat(BEGIN)nodes = self.statement_list()self.eat(END)root = Compound()for node in nodes:root.children.append(node)return rootdef statement_list(self):"""statement_list : statement| statement SEMI statement_list"""node = self.statement()results = [node]while self.current_token.type == SEMI:self.eat(SEMI)results.append(self.statement())if self.current_token.type == ID:self.error()return resultsdef statement(self):"""statement : compound_statement| assignment_statement| empty"""if self.current_token.type == BEGIN:node = self.compound_statement()elif self.current_token.type == ID:node = self.assignment_statement()else:node = self.empty()return nodedef assignment_statement(self):"""assignment_statement : variable ASSIGN expr"""left = self.variable()token = self.current_tokenself.eat(ASSIGN)right = self.expr()node = Assign(left, token, right)return nodedef variable(self):"""variable : ID"""node = Var(self.current_token)self.eat(ID)return nodedef empty(self):"""An empty production"""return NoOp()def expr(self):"""expr : term ((PLUS | MINUS) term)*"""node = self.term()while self.current_token.type in (PLUS, MINUS):token = self.current_tokenif token.type == PLUS:self.eat(PLUS)elif token.type == MINUS:self.eat(MINUS)node = BinOp(left=node, op=token, right=self.term())return nodedef term(self):"""term : factor ((MUL | DIV) factor)*"""node = self.factor()while self.current_token.type in (MUL, DIV):token = self.current_tokenif token.type == MUL:self.eat(MUL)elif token.type == DIV:self.eat(DIV)node = BinOp(left=node, op=token, right=self.factor())return nodedef factor(self):"""factor : PLUS factor| MINUS factor| INTEGER| LPAREN expr RPAREN| variable"""token = self.current_tokenif token.type == PLUS:self.eat(PLUS)node = UnaryOp(token, self.factor())return nodeelif token.type == MINUS:self.eat(MINUS)node = UnaryOp(token, self.factor())return nodeelif token.type == INTEGER:self.eat(INTEGER)return Num(token)elif token.type == LPAREN:self.eat(LPAREN)node = self.expr()self.eat(RPAREN)return nodeelse:node = self.variable()return nodedef parse(self):"""program : compound_statement DOTcompound_statement : BEGIN statement_list ENDstatement_list : statement| statement SEMI statement_liststatement : compound_statement| assignment_statement| emptyassignment_statement : variable ASSIGN exprempty :expr: term ((PLUS | MINUS) term)*term: factor ((MUL | DIV) factor)*factor : PLUS factor| MINUS factor| INTEGER| LPAREN expr RPAREN| variablevariable: ID"""node = self.program()if self.current_token.type != EOF:self.error()return node############################################################################### # # # INTERPRETER # # # ###############################################################################class NodeVisitor(object):def visit(self, node):method_name = 'visit_' + type(node).__name__visitor = getattr(self, method_name, self.generic_visit)return visitor(node)def generic_visit(self, node):raise Exception('No visit_{} method'.format(type(node).__name__))class Interpreter(NodeVisitor):GLOBAL_SCOPE = {}def __init__(self, parser):self.parser = parserdef visit_BinOp(self, node):if node.op.type == PLUS:return self.visit(node.left) + self.visit(node.right)elif node.op.type == MINUS:return self.visit(node.left) - self.visit(node.right)elif node.op.type == MUL:return self.visit(node.left) * self.visit(node.right)elif node.op.type == DIV:return self.visit(node.left) // self.visit(node.right)def visit_Num(self, node):return node.valuedef visit_UnaryOp(self, node):op = node.op.typeif op == PLUS:return +self.visit(node.expr)elif op == MINUS:return -self.visit(node.expr)def visit_Compound(self, node):for child in node.children:self.visit(child)def visit_Assign(self, node):var_name = node.left.valueself.GLOBAL_SCOPE[var_name] = self.visit(node.right)def visit_Var(self, node):var_name = node.valueval = self.GLOBAL_SCOPE.get(var_name)if val is None:raise NameError(repr(var_name))else:return valdef visit_NoOp(self, node):passdef interpret(self):tree = self.parser.parse()if tree is None:return ''return self.visit(tree)def main():import systext = open(sys.argv[1], 'r').read()lexer = Lexer(text)parser = Parser(lexer)interpreter = Interpreter(parser)result = interpreter.interpret()print(interpreter.GLOBAL_SCOPE)if __name__ == '__main__':main()

在spi.py文件所在的文件夾打開控制臺,運行結果:

C:\Users\Administrator\Desktop\編譯原理\python>python Python 3.6.8 (tags/v3.6.8:3c6b436a57, Dec 24 2018, 00:16:47) [MSC v.1916 64 bit (AMD64)] on win32 Type "help", "copyright", "credits" or "license" for more information. >>> from spi import Lexer >>> lexer = Lexer('BEGIN a := 2; END.') >>> lexer.get_next_token() Token(BEGIN, 'BEGIN') >>> lexer.get_next_token() Token(ID, 'a') >>> lexer.get_next_token() Token(ASSIGN, ':=') >>> lexer.get_next_token() Token(INTEGER, 2) >>> lexer.get_next_token() Token(SEMI, ';') >>> lexer.get_next_token() Token(END, 'END') >>> lexer.get_next_token() Token(DOT, '.') >>> lexer.get_next_token() Token(EOF, None) >>>

spi_lexer

# -*- coding: utf-8 -*- """ @File : spi_lexer.py @Time : 2021/8/2 10:02 @Author : Dontla @Email : sxana@qq.com @Software: PyCharm """ ############################################################################### # # # LEXER # # # ################################################################################ Token types # # EOF (end-of-file) token is used to indicate that # there is no more input left for lexical analysis (INTEGER, PLUS, MINUS, MUL, DIV, LPAREN, RPAREN, ID, ASSIGN,BEGIN, END, SEMI, DOT, EOF) = ('INTEGER', 'PLUS', 'MINUS', 'MUL', 'DIV', '(', ')', 'ID', 'ASSIGN','BEGIN', 'END', 'SEMI', 'DOT', 'EOF' )class Token(object):def __init__(self, type, value):self.type = typeself.value = valuedef __str__(self):"""String representation of the class instance.Examples:Token(INTEGER, 3)Token(PLUS, '+')Token(MUL, '*')"""return 'Token({type}, {value})'.format(type=self.type,value=repr(self.value))def __repr__(self):return self.__str__()RESERVED_KEYWORDS = {'BEGIN': Token('BEGIN', 'BEGIN'),'END': Token('END', 'END'), }class Lexer(object):def __init__(self, text):# client string input, e.g. "4 + 2 * 3 - 6 / 2"self.text = text# self.pos is an index into self.textself.pos = 0self.current_char = self.text[self.pos]def error(self):raise Exception('Invalid character')def advance(self):"""Advance the `pos` pointer and set the `current_char` variable."""self.pos += 1if self.pos > len(self.text) - 1:self.current_char = None # Indicates end of inputelse:self.current_char = self.text[self.pos]def peek(self):peek_pos = self.pos + 1if peek_pos > len(self.text) - 1:return Noneelse:return self.text[peek_pos]def skip_whitespace(self):while self.current_char is not None and self.current_char.isspace():self.advance()def integer(self):"""Return a (multidigit) integer consumed from the input."""result = ''while self.current_char is not None and self.current_char.isdigit():result += self.current_charself.advance()return int(result)def _id(self):"""Handle identifiers and reserved keywords"""result = ''# isalnum() 函數檢測至少有一個字符并且所有字符都是字母或數字則返回 True,否則返回 Falsewhile self.current_char is not None and self.current_char.isalnum():result += self.current_charself.advance()# get()函數返回字典指定鍵的值,如果鍵不存在,則返回get第二個參數的值(可選)# 如果字母數字字符串在保留字內則返回保留字字典中的值,否則返回Token(ID, result)token = RESERVED_KEYWORDS.get(result, Token(ID, result))return tokendef get_next_token(self):"""Lexical analyzer (also known as scanner or tokenizer)This method is responsible for breaking a sentenceapart into tokens. One token at a time."""while self.current_char is not None:if self.current_char.isspace():self.skip_whitespace()continueif self.current_char.isalpha():return self._id()if self.current_char.isdigit():return Token(INTEGER, self.integer())# “:=”同時做判斷if self.current_char == ':' and self.peek() == '=':self.advance()self.advance()return Token(ASSIGN, ':=')if self.current_char == ';':self.advance()return Token(SEMI, ';')if self.current_char == '+':self.advance()return Token(PLUS, '+')if self.current_char == '-':self.advance()return Token(MINUS, '-')if self.current_char == '*':self.advance()return Token(MUL, '*')if self.current_char == '/':self.advance()return Token(DIV, '/')if self.current_char == '(':self.advance()return Token(LPAREN, '(')if self.current_char == ')':self.advance()return Token(RPAREN, ')')if self.current_char == '.':self.advance()return Token(DOT, '.')self.error()return Token(EOF, None)if __name__ == '__main__':lexer = Lexer('BEGIN a := 2; END.')while True:token = lexer.get_next_token()print(token)if token.type == EOF:break

運行結果:

D:\python_virtualenv\my_flask\Scripts\python.exe C:/Users/Administrator/Desktop/編譯原理/python/spi_lexer.py Token(BEGIN, 'BEGIN') Token(ID, 'a') Token(ASSIGN, ':=') Token(INTEGER, 2) Token(SEMI, ';') Token(END, 'END') Token(DOT, '.') Token(EOF, None)進程已結束,退出代碼0

總結

以上是生活随笔為你收集整理的【编译原理】构建一个简单的解释器(Let’s Build A Simple Interpreter. Part 9.)(笔记)语法分析(未完,先搁置了!)的全部內容,希望文章能夠幫你解決所遇到的問題。

如果覺得生活随笔網站內容還不錯,歡迎將生活随笔推薦給好友。

麻花豆传媒一二三产区 | 成人免费观看av | 99久久激情视频 | 久久国产精品电影 | 久久精品久久精品久久39 | 国产看片 色 | 免费看黄色小说的网站 | 麻豆91网站| 欧美在线观看视频 | 国产福利免费在线观看 | 亚洲精品乱码久久久久 | 久久久久99精品成人片三人毛片 | 亚洲一区精品二人人爽久久 | 国产一级免费视频 | 国产高清在线 | 免费观看mv大片高清 | 手机av资源 | 欧日韩在线视频 | 国产日本在线观看 | 三上悠亚在线免费 | 九色精品免费永久在线 | 久久论理 | 亚洲一级片在线观看 | 一区二区中文字幕在线 | 精品国产a | 在线天堂中文在线资源网 | 99精品视频免费看 | 久久精品99国产精品酒店日本 | 97日日碰人人模人人澡分享吧 | 国产精品日韩高清 | 亚洲精品免费在线视频 | 91av精品| 五月婷婷在线综合 | 婷婷网站天天婷婷网站 | av先锋影音少妇 | 中文字幕a∨在线乱码免费看 | 91麻豆高清视频 | 97精品视频在线 | 深爱婷婷久久综合 | 黄色小视频在线观看免费 | 久久精品成人热国产成 | 国产在线播放一区二区三区 | 亚洲视频axxx | 丁香5月婷婷| 伊人亚洲综合网 | 91xav| 91在线免费看片 | 免费三级黄 | av888.com| 日韩国产精品毛片 | 精品一区二区综合 | www.色爱 | 久久 国产一区 | 亚洲欧洲一区二区在线观看 | 精品在线一区二区三区 | 国内精品免费久久影院 | 欧美黑吊大战白妞欧美 | 天天插日日操 | 久久久五月天 | 欧洲一区二区在线观看 | 免费a一级 | 日韩一二三区不卡 | 日韩欧美在线免费观看 | 成人91av| 丁香婷婷激情国产高清秒播 | 九九九在线观看 | 成人免费观看av | 国产一卡二卡四卡国 | 亚洲精品视频在 | 国产无套视频 | 高清av免费看 | 日韩精品中文字幕有码 | 99一级片 | 欧美另类交人妖 | 国产中文在线字幕 | 国产成人资源 | www日韩欧美 | wwxxxx日本 | 人人精品久久 | 国产又粗又猛又色又黄视频 | 在线播放你懂 | 九九免费在线观看视频 | 欧美性色综合 | 视频三区 | 最近日本字幕mv免费观看在线 | 手机av在线网站 | 国产亚洲精品久久久久动 | 亚洲女裸体 | 97手机电影网 | 国产在线观看黄 | 国际精品久久久久 | 色婷婷成人网 | 国产91勾搭技师精品 | 日韩av中文字幕在线免费观看 | 久久久久久高潮国产精品视 | 午夜美女福利直播 | 国产精品18毛片一区二区 | 人人射 | 精品久久久久亚洲 | 精品国产电影一区二区 | 黄色av三级在线 | 五月婷婷伊人网 | 丁香六月欧美 | 波多野结衣一区二区三区中文字幕 | 999久久久久久久久 69av视频在线观看 | 国产精品麻豆99久久久久久 | 激情视频一区二区三区 | 97精产国品一二三产区在线 | 色播五月婷婷 | 水蜜桃亚洲一二三四在线 | 在线观看视频黄色 | 亚洲久草在线视频 | 9999精品 | 蜜臀av在线一区二区三区 | 色老板在线 | 99热 精品在线| 超碰午夜 | 欧美日韩国产一区二区三区 | 人人爱人人做人人爽 | 又爽又黄又无遮挡网站动态图 | 国产h在线播放 | 亚洲高清视频在线播放 | 国产精品视频全国免费观看 | av网址在线播放 | 99国产视频| 国产精品第二页 | 麻豆va一区二区三区久久浪 | 亚洲五月花 | 国产成人av网| 精品国产一二三 | 欧美视频在线观看免费网址 | www.夜夜操.com | 久久新| 精品在线你懂的 | 成人网页在线免费观看 | 韩日三级在线 | 91人人揉日日捏人人看 | 不卡中文字幕在线 | 一区二区精品在线观看 | 91久久精品一区二区三区 | 丁香综合网 | 国产成人一区二区三区在线观看 | 久久久国产日韩 | 久久婷婷色综合 | 免费视频一区二区 | 国产精品福利视频 | 6080yy午夜一二三区久久 | 久久国产手机看片 | av中文字幕不卡 | 欧美 高跟鞋交 xxxxhd | 天天射网站 | 在线 国产 亚洲 欧美 | 在线亚洲人成电影网站色www | 337p日本大胆噜噜噜噜 | 久草在线在线精品观看 | 麻豆视频免费版 | 国产一区免费在线观看 | 日韩精品中文字幕在线播放 | 免费av在线播放 | 国产精品欧美久久久久天天影视 | 国产精品毛片久久久久久久久久99999999 | 草 免费视频 | 五月天婷婷在线视频 | 麻豆精品91 | 日韩av影片在线观看 | 日韩在线播放av | 欧美日韩国产一区二区三区在线观看 | 欧美午夜激情网 | 免费看片网址 | 亚洲激情在线视频 | 91一区啪爱嗯打偷拍欧美 | 天天综合操 | 一本一道波多野毛片中文在线 | 91九色性视频 | 有码中文在线 | 一本一本久久a久久精品牛牛影视 | 97电影在线观看 | 国产亚洲人 | 欧美日韩精品在线观看视频 | av软件在线观看 | 日韩视频一二三区 | 久久精品直播 | 丁香婷婷久久久综合精品国产 | 在线 欧美 日韩 | 日韩免费视频网站 | 日韩av中文在线 | www.av免费观看 | 国产剧情一区二区 | 日本久久不卡视频 | 麻豆传媒在线免费看 | 亚洲 成人 欧美 | 天天干天天爽 | 久久国产亚洲精品 | 精品一区二区三区香蕉蜜桃 | 久久久久久久福利 | 国产福利中文字幕 | 国产精品中文 | 99久久久久国产精品免费 | 日本成人免费在线观看 | 国产视频1区2区3区 久久夜视频 | 国产一区不卡在线 | 天天操天天射天天插 | 天堂va在线高清一区 | 伊人色**天天综合婷婷 | 国产伦理一区二区 | 麻豆视频免费在线观看 | 亚洲精品动漫久久久久 | 免费看片黄色 | 亚洲精品久久久久中文字幕m男 | 亚洲精品久久久蜜臀下载官网 | 久久深爱网 | 在线国产中文 | 麻豆视频免费网站 | 视频成人免费 | 久久免费在线 | 日韩免费观看视频 | 成人av在线网 | 中文字幕制服丝袜av久久 | 六月丁香在线观看 | 女人魂免费观看 | 久久96国产精品久久99软件 | 欧美日韩网址 | 91精品久久久久久久久 | 97电院网手机版 | 国产亚洲永久域名 | 51精品国自产在线 | 久久a热6 | 一区 二区电影免费在线观看 | 久久久久综合精品福利啪啪 | 天天操夜夜看 | 国产不卡在线看 | 色丁香久久 | 日本在线中文 | 免费欧美 | 激情久久综合网 | 青青河边草免费直播 | 久久在线一区 | 91人人射 | 黄色小说免费在线观看 | 在线黄色av | 黄色一级大片在线免费看产 | 日韩专区在线 | 国内视频在线 | 人人看黄色 | 99久久影院 | 天天夜操 | 久久综合婷婷国产二区高清 | 亚洲欧美成人综合 | 国产丝袜在线 | 中文区中文字幕免费看 | 国产xvideos免费视频播放 | 人人爽人人爱 | 少妇精品久久久一区二区免费 | 91在线一区 | 在线观看av不卡 | 国模视频一区二区 | 日韩影视在线观看 | 国内久久精品视频 | 一级免费观看 | 午夜黄色一级片 | 在线精品一区二区 | 久久亚洲专区 | 国产亚洲视频中文字幕视频 | 特黄特色特刺激视频免费播放 | 在线观看视频一区二区三区 | 麻豆91精品视频 | av亚洲产国偷v产偷v自拍小说 | 黄色三级网站在线观看 | 日韩精品中文字幕在线观看 | 婷婷丁香色 | 国产免费午夜 | 国产91精品久久久久 | 国产精品99久久久久久小说 | 国产视频999| 五月天亚洲婷婷 | 激情综合啪啪 | 中文区中文字幕免费看 | 免费热情视频 | 日韩三级视频 | 黄色在线视频网址 | 亚洲电影院 | 久操免费视频 | 一区二区三区在线电影 | 天天做日日做天天爽视频免费 | 亚洲成a人片77777潘金莲 | av黄色免费看 | 国产日韩三级 | 一本—道久久a久久精品蜜桃 | 天天干夜夜干 | 国产明星视频三级a三级点| av片在线看 | av在线影视 | 国产一区二区高清不卡 | av网站手机在线观看 | 一区二区三区精品在线视频 | 国产一级二级在线播放 | 午夜精品久久久久久久久久 | 久久久久观看 | 99精品久久久久久久 | 国产精品一区二区久久精品爱微奶 | 欧美激情另类文学 | 午夜91在线| a√资源在线| 国产一级精品视频 | 狠狠色丁香婷婷综合久小说久 | 干干夜夜 | 久久久久国产免费免费 | avlulu久久精品 | 国产精品夜夜夜一区二区三区尤 | 日韩中文字幕第一页 | 99c视频高清免费观看 | 狠狠的操你 | 国产 欧美 日本 | 丝袜美腿在线播放 | 日日夜夜免费精品 | 精品欧美在线视频 | 国产色在线观看 | 欧美精品三级在线观看 | 精品日韩在线 | 国产手机av| 免费观看成人av | 激情久久综合 | 国内精品久久久久久久影视麻豆 | 久草在线视频免费资源观看 | 日韩中文字幕a | 国产热re99久久6国产精品 | 91精品视频免费观看 | 狠狠色免费 | 91黄色小视频 | 爱爱av网站| 国产亚州av | 日本不卡一区二区 | 欧美 日韩 成人 | 色综合五月 | 91tv国产成人福利 | 精品a在线 | 91精品国产91 | 激情综合网五月 | 亚洲劲爆av| 亚洲成av人影院 | 国产亚洲永久域名 | 最新日本中文字幕 | 五月网婷婷 | 一本一本久久a久久 | 夜夜骑天天操 | 丁香五婷 | 在线观看亚洲专区 | 色欧美日韩 | 中文国产在线观看 | 青青河边草免费观看完整版高清 | 狠狠网亚洲精品 | 免费视频黄色 | 日日夜夜91 | 日韩欧美精品在线观看视频 | 国产精品美女免费看 | 在线免费观看黄色 | 免费色视频网站 | 亚洲天堂毛片 | 天天射射天天 | 深夜福利视频在线观看 | 日韩国产精品久久 | 在线观看久 | 色综合久久五月 | 国产一区视频导航 | 久久久久草 | 亚洲精品字幕在线 | 中文在线a√在线 | 最新动作电影 | 国产丝袜高跟 | 片网站 | 国产 欧美 日韩 | 精品99在线 | 国产精品av免费 | 久草在线视频资源 | 久久精品伊人 | 九九九九热精品免费视频点播观看 | 揉bbb玩bbb少妇bbb | 97超碰在线人人 | 婷婷丁香激情综合 | 91一区啪爱嗯打偷拍欧美 | 国产视频久久 | 亚洲高清视频一区二区三区 | 五月天伊人 | 麻豆观看 | 欧美性极品xxxx娇小 | 九九九九精品 | 国产91免费在线观看 | 在线观看视频99 | 综合天天久久 | 在线网址你懂得 | 欧美aaa视频 | 中文字幕4 | 亚洲jizzjizz日本少妇 | 日韩精品五月天 | 五月婷婷综合在线视频 | av线上看 | 婷婷色六月天 | 国产精品美女免费 | 91精品国产高清 | 香蕉视频啪啪 | 亚洲欧美日韩精品久久久 | 91免费视频黄 | 色婷婷av国产精品 | 国产免码va在线观看免费 | 免费黄a大片 | 免费看成人a | 欧美在线视频二区 | 91探花系列在线播放 | 一级黄色在线视频 | 国产精品99久久久精品 | 国产一区二区三区免费在线 | 婷婷四房综合激情五月 | 国产人成免费视频 | 欧美久久成人 | 999国产精品视频 | 欧美一区二区伦理片 | 国产精品久久久久一区二区国产 | 欧美激情视频免费看 | 日韩簧片在线观看 | 一级免费看视频 | av免费电影在线观看 | 国产精品入口麻豆www | 四虎在线免费 | 天天色天天操综合 | 久久午夜色播影院免费高清 | 国产一级大片在线观看 | 在线看黄色的网站 | 日韩在线观看第一页 | 六月色丁| 97人人人人 | 精品在线你懂的 | 人人干天天干 | www.国产毛片| 国产成人亚洲在线电影 | 91免费视频黄 | 韩国三级在线一区 | 最新av在线免费观看 | 日韩欧美在线免费观看 | 最新av网址在线观看 | 国产精品ⅴa有声小说 | 国产精品久久影院 | 日韩在线播放av | 五月婷婷国产 | 精品91视频 | 99国产视频 | 久久国产剧场电影 | 444av| 97超碰人人澡人人爱学生 | 在线 视频 亚洲 | 国产精品久久久久久a | 日韩在线免费看 | 久久国产精品第一页 | 麻豆成人在线观看 | 国产色视频网站2 | 亚洲欧美日韩国产一区二区三区 | 亚洲成av人影院 | 成人免费视频网站在线观看 | av短片在线 | 夜夜夜草 | 国产不卡视频在线播放 | 国产一线二线三线在线观看 | 欧美三级免费 | 国语自产偷拍精品视频偷 | 成人av在线电影 | 91精彩视频在线观看 | 国产在线观看一 | 国产在线观看a | 亚洲色图27p | 日韩在线视频观看免费 | 天天操综合网站 | 日韩精品最新在线观看 | 在线观看91久久久久久 | h网站免费在线观看 | 精品字幕在线 | 精品一区二区电影 | 99国产精品免费网站 | 黄色大片国产 | 亚洲国产精品成人综合 | 一区 在线观看 | 欧美日韩色婷婷 | 国产裸体视频bbbbb | 成片免费观看视频 | 在线成人免费 | 天天射天天干天天 | 成人毛片100免费观看 | 69精品视频在线观看 | 色偷偷88888欧美精品久久 | av一级在线| 黄色的视频网站 | 最近中文字幕完整视频高清1 | 日本h视频在线观看 | 亚洲综合爱 | 欧美精品久久久久久久亚洲调教 | 久久视频网 | 国产视频美女 | 亚洲a资源| 成人国产精品一区二区 | www五月婷婷 | 91麻豆高清视频 | 国产剧情一区二区 | 国产.精品.日韩.另类.中文.在线.播放 | 国产伦理久久精品久久久久_ | 欧美日韩中文在线视频 | 久久系列 | 丁香午夜| 免费日韩电影 | 一区二区三区视频网站 | 高潮久久久久久久久 | 欧美在线不卡一区 | 精品亚洲午夜久久久久91 | 久久欧美综合 | 超碰在线人人 | 久草在线免费看视频 | 一二区精品| 色综合天 | 国产资源精品在线观看 | 亚洲乱码精品久久久久 | 波多野结衣小视频 | 91精品久久久久久久久 | 日韩免费视频一区二区 | 精品久久久久一区二区国产 | 91mv.cool在线观看 | 久久久久亚洲精品中文字幕 | a电影免费看| 中文字幕一区二区三区四区 | 色资源网在线观看 | 免费av一级电影 | 操操操夜夜操 | 国产在线高清视频 | 亚洲国产97在线精品一区 | 在线观看免费av网站 | 99免费在线观看 | 麻豆视频观看 | 人人澡人人干 | 久久久噜噜噜久久久 | 国产麻豆成人传媒免费观看 | 国产成人精品电影久久久 | av电影中文| 久热av| 日韩区欧美久久久无人区 | 色婷婷亚洲婷婷 | 久久国产精品一二三区 | 黄色午夜网站 | 超碰激情在线 | 日本久久精品视频 | 91精品入口 | 亚洲一二三在线 | 在线观看黄色的网站 | 操老逼免费视频 | av色图天堂网 | 97在线视频免费播放 | 99久久精品免费看国产麻豆 | 五月婷婷激情 | 天天玩天天操天天射 | www.啪啪.com| 在线国产精品一区 | 在线国产激情视频 | 伊人狠狠色丁香婷婷综合 | 日韩高清免费无专码区 | 国产精品视频免费在线观看 | 九九热只有精品 | 在线播放91 | 91高清免费观看 | 亚洲精品一区二区18漫画 | 夜夜躁日日躁狠狠久久av | 亚洲综合一区二区精品导航 | 久草在线一免费新视频 | 在线电影中文字幕 | 中文字幕精品视频 | 成片人卡1卡2卡3手机免费看 | 久免费| 在线观看欧美成人 | 91在线免费播放 | 精品欧美一区二区精品久久 | 久久国产精品免费一区二区三区 | 色999视频 | 在线视频18在线视频4k | 国产精品自产拍在线观看桃花 | 成人免费视频a | 美女网站视频免费黄 | 中文字幕欧美日韩va免费视频 | 日日干网 | 99超碰在线播放 | 久久精品视频2 | 国产亚洲片 | 国产亚洲精品成人av久久ww | 国产精品一区在线观看你懂的 | 97在线观看免费高清完整版在线观看 | 狠狠干电影 | 国产一级视屏 | 少妇性aaaaaaaaa视频 | 九九热1| 欧美精品一区二区免费 | 懂色av一区二区三区蜜臀 | 韩国精品视频在线观看 | 国产黄a三级三级三级三级三级 | 91chinese在线 | 成人免费影院 | 国产无遮挡猛进猛出免费软件 | 视频国产 | 激情中文在线 | 精品乱码一区二区三四区 | 日韩欧美一级二级 | 亚洲在线| 国产精品毛片一区二区在线 | 欧美激情奇米色 | 在线观看资源 | 中文字幕一区二区在线播放 | 日韩视频a| 久草电影在线 | 欧美乱淫视频 | 在线免费看片 | 色a在线观看 | 成人在线一区二区三区 | 中文字幕黄色 | 成人久久毛片 | 在线亚洲成人 | www五月天婷婷 | 亚洲一区欧美激情 | 视频直播国产精品 | 国产五月色婷婷六月丁香视频 | 日日夜夜噜 | 日本精品在线视频 | 成人a免费看 | 欧美在线久久 | 丁香综合 | 国产精品视频一二三 | 一区二区三区日韩在线观看 | 免费在线观看的av网站 | 在线观看免费色 | 免费 在线 中文 日本 | 午夜精品久久久久久 | 久久综合久久伊人 | 国产精品久久综合 | 亚洲精品麻豆 | 色婷婷激情四射 | 欧美日韩视频观看 | 久久国产美女 | 九九综合九九综合 | 国产999视频在线观看 | 久久新视频 | a级黄色片视频 | 国产成人精品免费在线观看 | 婷婷综合影院 | 96国产精品视频 | 99久久久国产精品美女 | a级片久久 | 97视频在线免费观看 | av在线最新 | av国产在线观看 | 国内精品久久久久久久影视麻豆 | 国产精品综合在线观看 | 色婷婷国产精品 | 日韩电影一区二区三区 | 最近中文字幕高清字幕在线视频 | 99在线免费观看视频 | 日韩精品91偷拍在线观看 | 亚洲在线免费视频 | 18国产精品福利片久久婷 | 有码中文在线 | 黄色精品视频 | 欧美激情精品久久久久 | 一区二区精品在线视频 | 国产精品国产三级国产不产一地 | 国产一级片在线播放 | 在线国产能看的 | 午夜影院一级 | 在线观看视频精品 | 色婷婷电影 | 天天综合网~永久入口 | 久草热视频 | 国产精品免费视频观看 | 日韩成人免费电影 | 96精品高清视频在线观看软件特色 | a级国产片| 亚洲欧美日韩精品久久奇米一区 | 五月婷婷欧美视频 | 欧美一级电影在线观看 | av黄网站 | 国产精品综合在线观看 | 婷婷六月天在线 | 日韩天天操 | 成人免费一区二区三区在线观看 | 不卡的av在线 | 99久久久久久久久 | 在线观看91网站 | www.超碰97.com | 91大神电影 | 久久一级电影 | 91看片淫黄大片在线播放 | 欧美日韩网站 | 毛片精品免费在线观看 | 久热av| 91亚州| 国产淫片免费看 | 日韩大片在线看 | 99精品一区二区 | 999久久久免费视频 午夜国产在线观看 | 精品毛片在线 | 欧美午夜a | 日韩在线观看网站 | 超碰在线94 | 久久久免费看视频 | 日韩区欠美精品av视频 | 欧美乱熟臀69xxxxxx | 亚洲精品一区二区网址 | 欧美激情精品久久久久 | 天天干天天操天天操 | 久久综合中文字幕 | 久99久久| 婷婷在线不卡 | 在线综合色 | 蜜臀aⅴ精品一区二区三区 久久视屏网 | 91福利视频久久久久 | 免费视频一级片 | 五月天综合激情网 | 国产超碰97 | 欧美性天天 | 视频一区二区精品 | 国产精品一区二区久久久 | 国产一区二区播放 | www.国产在线观看 | 激情视频在线观看网址 | 亚洲精品国产精品99久久 | 日韩欧美综合 | 国产成人精品一区二三区 | 精品国产一区二区三区免费 | 国产精品免费高清 | 在线观看色网 | 99精品黄色片免费大全 | 欧美一级片免费在线观看 | 精品国产人成亚洲区 | 日韩成人精品在线观看 | 香蕉视频在线免费 | 久久激情网站 | 91麻豆国产 | 亚洲一区动漫 | 欧美日韩视频在线观看一区二区 | www.久久精品视频 | 91原创在线观看 | 中文字幕在线观看1 | 在线影院中文字幕 | 欧美日韩中文国产 | 在线欧美小视频 | 免费黄色a级毛片 | 久热香蕉视频 | 久久久午夜视频 | 日韩精品久久久免费观看夜色 | 成年人在线免费看视频 | 久久久久综合精品福利啪啪 | av在线播放一区二区三区 | freejavvideo日本免费 | 在线观看免费观看在线91 | 国产精品第十页 | 日韩欧美一区二区三区视频 | 久久综合五月 | 国产免费资源 | 在线电影中文字幕 | 天天综合天天做 | 激情视频免费在线观看 | 玖玖综合网| 美女视频网 | 国产精品美女久久久久久久久久久 | 亚洲欧美日韩在线看 | 免费成人在线网站 | 国产成人性色生活片 | 天天躁天天躁天天躁婷 | 日韩亚洲在线观看 | 国产黄色精品在线 | 亚洲理论片在线观看 | 天天视频亚洲 | 色狠狠干 | 亚洲精品456在线播放第一页 | 日本三级不卡视频 | adn—256中文在线观看 | 日本精品视频免费观看 | 天堂av免费观看 | 欧美福利久久 | av免费在线观看1 | 成人久久网 | 精品国自产在线观看 | 国产午夜精品一区二区三区欧美 | 欧美精品久久99 | 国产亚洲视频在线免费观看 | 国产精品普通话 | 天天操福利视频 | 97国产大学生情侣酒店的特点 | 国产乱码精品一区二区三区介绍 | 黄www在线观看 | 亚洲免费婷婷 | 国产美女精品视频免费观看 | 天天射天天添 | 国产成人99av超碰超爽 | 国产精品一区免费观看 | 国产精品久久免费看 | 黄色福利网站 | 少妇bbbb搡bbbb搡bbbb | 日韩精品视频免费看 | 国内久久久久 | 91av在线不卡| 成人午夜在线观看 | 亚洲精选视频在线 | 97色国产| av在线成人 | av手机版 | 久久久久久久久久伊人 | 丁香久久激情 | 国产精品日韩欧美一区二区 | 亚洲婷婷伊人 | 人人草人人草 | 国产精品日韩在线播放 | 亚洲人成免费 | 亚洲国产欧美在线人成大黄瓜 | 欧美最新大片在线看 | 久久亚洲美女 | 亚洲精品xxxx | 亚洲免费一级电影 | 中文字幕在线视频一区二区 | 久草在线观看视频免费 | www.av在线播放 | 欧美精品一区二区三区四区在线 | 91大神视频网站 | 日日夜夜中文字幕 | 精品久久久久久久久久久久 | www.久草视频 | 成人免费ⅴa | 国产伦理剧 | av在线电影网站 | 奇米影视8888在线观看大全免费 | av丝袜美腿 | www日韩| 在线有码中文字幕 | 精精国产xxxx视频在线播放 | 狠狠操夜夜操 | 色婷婷精品 | 色在线亚洲| 色a网 | 久碰视频在线观看 | 久草在线视频网 | av爱干 | 国产123av| 中文字幕专区高清在线观看 | 中文字幕在线一区二区三区 | 国内精品久久久久久久影视麻豆 | 久久久久久久网站 | 在线观看资源 | 五月天综合婷婷 | aaa免费毛片 | 91精品对白一区国产伦 | 国产.精品.日韩.另类.中文.在线.播放 | 欧美了一区在线观看 | 在线欧美日韩 | 久久久久久久影院 | 人人澡人摸人人添学生av | 五月在线视频 | 91一区在线观看 | 国产美女被啪进深处喷白浆视频 | 日韩一二区在线 | 国产成人一区二区精品非洲 | 国产精品久久久一区二区三区网站 | 久久免费国产精品 | 午夜狠狠干 | 日日夜夜精品免费视频 | 99精品久久99久久久久 | 黄色a级片在线观看 | 国产一卡二卡在线 | 日韩高清国产精品 | 成人国产网址 | 国产精品激情偷乱一区二区∴ | 国产视频在 | 亚洲尺码电影av久久 | 午夜精品一区二区三区免费视频 | 91在线色 | 久久中文欧美 | 毛片区 | 亚洲精品在线观看av | 久久免费试看 | 国产一级电影在线 | 九九在线国产视频 | 亚洲人xxx | 久久免费高清 | 久久久久久视频 | av免费观看网址 | 亚洲成a人片77777kkkk1在线观看 | av日韩不卡 | 婷婷丁香激情 | 99精品视频在线观看播放 | 黄色av网站在线观看免费 | 欧美性色黄大片在线观看 | 99精品欧美一区二区蜜桃免费 | 国产中文字幕在线看 | 成人91在线 | 97精品国产手机 | 91成人午夜| 欧美国产日韩一区二区三区 | 日韩狠狠操| 91av色 | a天堂最新版中文在线地址 久久99久久精品国产 | 欧美日韩国产精品爽爽 | 日韩高清一区在线 | 国产五月 | 96精品视频| 色综合五月天 | 色婷婷av一区二 | 国产黄影院色大全免费 | 黄色激情网址 | 日日爽天天 | 久久永久免费视频 | 成人午夜网址 | 大型av综合网站 | 日韩黄色在线观看 | 91九色蝌蚪国产 | 国产精品美女久久 | 亚洲免费在线 | 亚洲视频在线观看免费 | 天天操夜夜操天天射 | 久热电影| 日韩精品一区二区三区电影 | 婷婷六月天天 | 亚洲视频综合 | 成年人视频免费在线 | 日韩精品免费专区 | www.神马久久 | 黄在线免费观看 | 久久草网站 | 久久精品视频国产 | www.天堂av| 粉嫩一区二区三区粉嫩91 | 欧美成人精品在线 | 超碰97人人爱 | 日日操夜夜操狠狠操 | 久久一区二区三区国产精品 | 激情欧美网 | 日韩成人精品在线观看 | 中国精品少妇 | 国产99久久精品 | 国产又粗又硬又爽视频 | 久久成人一区二区 | 日产乱码一二三区别免费 | 五月天色中色 | 四虎在线视频 | 激情欧美日韩一区二区 | 在线观看黄色免费视频 | www.国产在线 | 激情综合网婷婷 | 久艹视频在线免费观看 | 高清视频一区二区三区 | 欧美成a人片在线观看久 | 激情视频国产 | 亚洲在线视频免费 | 欧美日韩精品影院 | 中文视频在线 | 美女免费黄网站 | 中文字幕av全部资源www中文字幕在线观看 | 久草在线视频网站 | 国产精品原创 | 国产一区二区综合 | 国产精品欧美精品 | 99精品视频在线播放观看 | 国产一区二区在线播放视频 | 亚洲理论电影网 | 香蕉久久久久 | 麻豆手机在线 | 国产精品乱码久久久久久1区2区 | 日韩一三区 | 在线中文字幕播放 | 亚洲精品影视 | 午夜精品一区二区三区免费 | 亚洲精品乱码久久久久久高潮 | 九九九九色 | av久久久| 国产一级免费片 | 久久婷婷开心 | 蜜桃视频精品 | 九九视频精品在线 | 国产淫片 | 国产高清专区 | 久久婷婷国产 | 亚洲视频 视频在线 | 国产一级视屏 | 91麻豆精品国产91久久久久久 | 国产精品免费在线播放 | 国产精品毛片一区 | 嫩草av在线 | 欧美性生交大片免网 | 成人免费在线观看入口 | 国产精品久久久久久久久久免费看 | 欧美精品三级 | 亚洲国产日韩一区 | 99人成在线观看视频 | 91精品在线视频观看 | 国产在线中文 | 在线观看国产一区二区 | 特级毛片爽www免费版 | 玖玖在线观看视频 | 99精品国产在热久久下载 | 久免费视频 | 韩国一区二区三区在线观看 |