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

歡迎訪問 生活随笔!

生活随笔

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

编程问答

【编译原理】构建一个简单的解释器(Let’s Build A Simple Interpreter. Part 7.)(笔记)解释器 interpreter 解析器 parser 抽象语法树AST

發布時間:2025/3/20 编程问答 25 豆豆

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

文章目錄

    • python代碼
    • 插——后序遍歷
    • C語言代碼(有錯誤)
    • C語言代碼(修改優化后)
    • 總結

正如我上次向您保證的那樣,今天我將討論我們將在本系列的其余部分中使用的中心數據結構之一,所以系好安全帶,讓我們開始吧。

到目前為止,我們將解釋器(interpreter )和解析器(parser )代碼混合在一起,一旦解析器識別出某種語言結構,如加法、減法、乘法或除法,解釋器就會對表達式求值。這種解釋器被稱為語法導向解釋器。它們通常對輸入進行一次傳遞,適用于基本語言應用程序。為了分析更復雜的 Pascal 編程語言結構,我們需要構建一個中間表示( IR )。我們的解析器將負責構建一個 IR,我們的解釋器將使用它來解釋表示為IR的輸入。

事實證明,是非常適合IR 的數據結構。


讓我們快速討論一下樹術語。

  • 一個樹是一種數據結構,由組織成一個層次中的一個或多個節點。
  • 這棵樹有一個根,它是頂部節點。
  • 除了根節點之外的所有節點都有一個唯一的parent。
  • 下圖中標有*的節點是父節點。標記為2和7 的節點是它的子節點;孩子們是從左到右排列的。
    沒有子 節點的節點稱為葉節點
  • 具有一個或多個子節點且不是根的 節點稱為內部節點。
  • 子樹也可以是完整的子樹。在下圖中,+節點的左子節點(標記為*) 是一個完整的子樹,帶有自己的子節點。
  • 在計算機科學中,我們從頂部的根節點和向下生長的分支開始倒置繪制樹。
  • 這是表達式 2 * 7 + 3 的樹,并附有說明:


我們將在整個系列中使用的IR稱為抽象語法樹( AST )。但在我們深入研究 AST 之前,讓我們先簡單地談談解析樹。盡管我們不會在解釋器和編譯器中使用解析樹,但它們可以通過可視化解析器的執行跟蹤來幫助您了解解析器如何解釋輸入。我們還將它們與 AST 進行比較,以了解為什么 AST 比解析樹更適合用于中間表示。

那么,什么是解析樹?一個解析樹(有時稱為具體語法樹)是表示根據我們的語法定義語言結構的句法結構樹。它基本上顯示了您的解析器如何識別語言結構,或者換句話說,它顯示了您的語法的起始符號如何派生出編程語言中的某個字符串。

解析器的調用堆棧隱式地表示一個解析樹,它會在您的解析器嘗試識別特定語言結構時自動構建在內存中。

讓我們看一下表達式 2 * 7 + 3 的解析樹:

在上圖中,您可以看到:(expr加減 -> term乘除 -> factor數)

  • 解析樹記錄了解析器用于識別輸入的一系列規則。
  • 解析樹的根用文法開??始符號標記。
  • 每個內部節點代表一個非終結符,即它代表一個文法規則應用,如我們的例子中的expr、term或factor。
  • 每個葉節點代表一個標記。
    正如我已經提到的,我們不會手動構建解析器樹并將它們用于我們的解釋器,但是解析樹可以通過可視化解析器調用序列來幫助您理解解析器如何解釋輸入。

您可以通過試用一個名為genptdot.py的小實用程序來了解不同算術表達式的解析樹的外觀,我很快編寫了該實用程序來幫助您將它們可視化。要使用該實用程序,您首先需要安裝Graphviz包,在運行以下命令后,您可以打開生成的圖像文件 parsetree.png 并查看作為命令行參數傳遞的表達式的解析樹:(用不了不知咋回事,導不了包)

$ python genptdot.py "14 + 2 * 3 - 6 / 2" > \parsetree.dot && dot -Tpng -o parsetree.png parsetree.dot

這是為表達式 14 + 2 * 3 - 6 / 2 生成的圖像 parsetree.png:


通過向它傳遞不同的算術表達式來稍微玩一下該實用程序,并查看特定表達式的解析樹是什么樣的。

現在,讓我們談談抽象語法樹( AST )。這是我們將在本系列的其余部分大量使用的中間表示( IR )。它是我們解釋器和未來編譯器項目的核心數據結構之一。

讓我們通過查看表達式 2 * 7 + 3的AST和解析樹來開始我們的討論:

從上圖可以看出,AST在變小的同時捕捉到了輸入的本質。

以下是 AST 和解析樹之間的主要區別:

  • AST 使用操作符/操作作為根節點和內部節點,并使用操作數作為它們的子節點。
  • 與解析樹不同,AST 不使用內部節點來表示語法規則。
  • AST 并不代表真實語法中的每一個細節(這就是為什么它們被稱為abstract)——例如,沒有規則節點和括號。
  • 與相同語言構造的解析樹相比,AST 是密集的。

那么,什么是抽象語法樹呢?一個抽象語法樹(AST)是表示一個語言結構,其中每個內部節點和根節點表示操作者的抽象句法結構的樹,并且節點的子節點表示操作者的操作數

我已經提到 AST 比解析樹更緊湊。讓我們看一下表達式 7 + ((2 + 3))的AST和解析樹。可以看到下面的AST比解析樹小很多,但還是抓住了輸入的本質:

到目前為止一切順利,但是您如何在AST 中編碼運算符優先級?為了在AST 中對運算符優先級進行編碼,即表示“X 發生在 Y 之前”,您只需將 X 在樹中放在比 Y 的位置。您已經在前面的圖片中看到了這一點。

讓我們再看一些例子。

在下圖中的左側,您可以看到表達式 2 * 7 + 3的AST。讓我們通過將 7 + 3 放在括號內來更改優先級。您可以在右側看到修改后的表達式 2 * (7 + 3)的AST是什么樣的:

這是表達式 1 + 2 + 3 + 4 + 5的AST:

從上面的圖片中,您可以看到優先級較高的運算符在樹中的位置較低。

好的,讓我們編寫一些代碼來實現不同的AST節點類型并修改我們的解析器以生成由這些節點組成的AST樹。

首先,我們將創建一個名為AST的基節點類,其他類將從該類繼承:

class AST(object):pass

實際上沒有多少。回想一下,AST 表示操作符-操作數模型。到目前為止,我們有四個運算符和整數操作數。運算符是加法、減法、乘法和除法。我們可以創建一個單獨的類來表示每個運算符,例如 AddNode、SubNode、MulNode 和 DivNode,但是我們將只用一個BinOp類來表示所有四個二元運算符(二元運算符是對兩個運算符進行運算的運算符)操作數):

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

構造函數的參數是left、op和right,其中left和right 分別指向左操作數的節點和右操作數的節點。Op持有運算符本身的標記: Token( PLUS , ‘+’) 表示加號運算符, Token( MINUS , ‘-’) 表示減號運算符,依此類推。

為了在我們的AST 中表示整數,我們將定義一個Num類,該類將保存一個INTEGER標記和標記的值:

class Num(AST):def __init__(self, token):self.token = tokenself.value = token.value

正如您所注意到的,所有節點都存儲用于創建節點的標記。這主要是為了方便,將來會派上用場。

回憶一下表達式 2 * 7 + 3的AST。我們將在該表達式的代碼中手動創建它:

>>> from spi import Token, MUL, PLUS, INTEGER, Num, BinOp >>> >>> mul_token = Token(MUL, '*') >>> plus_token = Token(PLUS, '+') >>> mul_node = BinOp( ... left=Num(Token(INTEGER, 2)), ... op=mul_token, ... right=Num(Token(INTEGER, 7)) ... ) >>> add_node = BinOp( ... left=mul_node, ... op=plus_token, ... right=Num(Token(INTEGER, 3)) ... )

以下是定義了我們的新節點類后AST 的外觀。下圖也沿用了上面的手工構建過程:


這是我們修改后的解析器代碼,它構建并返回一個AST作為識別輸入(算術表達式)的結果:

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 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 factor(self):"""factor : INTEGER | LPAREN expr RPAREN"""token = self.current_tokenif token.type == INTEGER:self.eat(INTEGER)return Num(token)elif token.type == LPAREN:self.eat(LPAREN)node = self.expr()self.eat(RPAREN)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 expr(self):"""expr : term ((PLUS | MINUS) term)*term : factor ((MUL | DIV) factor)*factor : INTEGER | LPAREN expr RPAREN"""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 parse(self):return self.expr()

讓我們回顧一下一些算術表達式的AST構造過程。

如果你看一下解析器代碼上面你可以看到它的方式構建了一個節點AST是每個BinOp節點采用的當前值節點變量作為它的左子和呼叫到一個結果項或因素作為其右孩子,所以它有效地將節點向左下推,下面的表達式 1 +2 + 3 + 4 + 5 的樹就是一個很好的例子。這是解析器如何逐漸為表達式 1 + 2 + 3 + 4 + 5構建AST的直觀表示:


為了幫助您可視化不同算術表達式的 AST,我編寫了一個小實用程序,它將算術表達式作為其第一個參數,并生成一個DOT文件,然后由dot實用程序處理該文件以實際為您繪制AST(dot是運行dot命令需要安裝的Graphviz包)。這是一個命令和為表達式 7 + 3 * (10 / (12 / (3 + 1) - 1))生成的AST圖像:

$ python genastdot.py "7 + 3 * (10 / (12 / (3 + 1) - 1))" > \ ast.dot && dot -Tpng -o ast.png ast.dot


編寫一些算術表達式,手動繪制表達式的 AST,然后通過使用genastdot.py工具為相同的表達式生成AST圖像來驗證它們是值得的。這將幫助您更好地理解解析器如何為不同的算術表達式構造 AST。

好的,這是表達式 2 * 7 + 3的AST:

您如何遍歷樹以正確評估該樹表示的表達式?您可以通過使用后序遍歷(深度優先遍歷的一種特殊情況)來實現這一點,它從根節點開始并從左到右遞歸訪問每個節點的子節點。后序遍歷盡可能快地訪問遠離根的節點

這是后序遍歷的偽代碼,其中<>是BinOp節點的加法、減法、乘法或除法等操作的占位符,或者是返回Num 節點的整數值等更簡單的操作:

我們要為解釋器使用后序遍歷的原因是,首先,我們需要評估樹中較低的內部節點,因為它們代表具有更高優先級的運算符;其次,我們需要在應用運算符之前評估運算符的操作數到那些操作數。在下圖中,您可以看到,通過后序遍歷,我們首先評估表達式 2 * 7,然后才評估 14 + 3,這給出了正確的結果,17:

為了完整起見,我將提到深度優先遍歷的三種類型:前序遍歷、中序遍歷和后序遍歷。遍歷方法的名字來自你在訪問代碼中放置動作的地方:

有時您可能必須在所有這些點(前序、中序和后序)執行某些操作。您將在本文的源代碼存儲庫中看到一些示例。

好的,讓我們編寫一些代碼來訪問和解釋由我們的解析器構建的抽象語法樹,好嗎?

這是實現訪問者模式的源代碼:

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__))

這是我們的Interpreter類的源代碼,它繼承自NodeVisitor類并實現了不同的方法,這些方法的形式為visit_NodeType,其中NodeType被替換為節點的類名,如BinOp、Num等:

class Interpreter(NodeVisitor):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.value

這段代碼有兩個有趣的地方值得一提:首先,操作AST節點的訪問者代碼與AST節點本身是解耦的。您可以看到任何AST節點類(BinOp 和 Num)都沒有提供任何代碼來操作存儲在這些節點中的數據。該邏輯封裝在實現NodeVisitor 類的Interpreter類中。

其次,而不是像這樣在 NodeVisitor 的訪問方法中使用一個巨大的if語句:

def visit(node):node_type = type(node).__name__if node_type == 'BinOp':return self.visit_BinOp(node)elif node_type == 'Num':return self.visit_Num(node)elif ...# ...

或者像這樣:

def visit(node):if isinstance(node, BinOp):return self.visit_BinOp(node)elif isinstance(node, Num):return self.visit_Num(node)elif ...

NodeVisitor 的訪問方法非常通用,它根據傳遞給它的節點類型將調用分派到適當的方法。正如我之前提到的,為了使用它,我們的解釋器從NodeVisitor類繼承并實現了必要的方法。所以如果傳遞給visit方法的節點類型是BinOp,那么visit方法會將調用分派到visit_BinOp方法;如果節點類型是Num,那么visit方法會將調用分派到visit_Num方法, 等等。

花一些時間研究這種方法(標準 Python 模塊ast使用相同的節點遍歷機制),因為將來我們將使用許多新的visit_NodeType方法擴展我們的解釋器。

該generic_visit方法是拋出一個異常,表明它遇到的實現類沒有相應的節點的后備visit_NodeType方法。

現在,讓我們為表達式 2 * 7 + 3手動構建一個AST,并將其傳遞給我們的解釋器,以查看訪問方法的作用,以評估表達式。以下是從 Python shell 執行此操作的方法:

>>> from spi import Token, MUL, PLUS, INTEGER, Num, BinOp >>> >>> mul_token = Token(MUL, '*') >>> plus_token = Token(PLUS, '+') >>> mul_node = BinOp( ... left=Num(Token(INTEGER, 2)), ... op=mul_token, ... right=Num(Token(INTEGER, 7)) ... ) >>> add_node = BinOp( ... left=mul_node, ... op=plus_token, ... right=Num(Token(INTEGER, 3)) ... ) >>> from spi import Interpreter >>> inter = Interpreter(None) >>> inter.visit(add_node) 17

如您所見,我將表達式樹的根傳遞給了訪問方法,并通過將調用分派到解釋器類的正確方法(visit_BinOp和visit_Num)并生成結果來觸發樹的遍歷。

好的,為了您的方便,這是我們新解釋器的完整代碼:

python代碼

運行結果:

""" SPI - Simple Pascal Interpreter """############################################################################### # # # 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, EOF = ('INTEGER', 'PLUS', 'MINUS', 'MUL', 'DIV', '(', ')', '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__()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 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 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.isdigit():return Token(INTEGER, self.integer())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, ')')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 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 factor(self):"""factor : INTEGER | LPAREN expr RPAREN"""token = self.current_tokenif token.type == INTEGER:self.eat(INTEGER)return Num(token)elif token.type == LPAREN:self.eat(LPAREN)node = self.expr()self.eat(RPAREN)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 expr(self):"""expr : term ((PLUS | MINUS) term)*term : factor ((MUL | DIV) factor)*factor : INTEGER | LPAREN expr RPAREN"""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 parse(self):return self.expr()############################################################################### # # # 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):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 interpret(self):tree = self.parser.parse()return self.visit(tree)def main():while True:try:try:# text = raw_input('spi> ')text = input('spi> ')except NameError: # Python3text = input('spi> ')except EOFError:breakif not text:continuelexer = Lexer(text)parser = Parser(lexer)interpreter = Interpreter(parser)result = interpreter.interpret()print(result)if __name__ == '__main__':main() C:\Users\Administrator.YC-20180625SBPF\AppData\Local\Programs\Python\Python39\python.exe C:/Users/Administrator.YC-20180625SBPF/Desktop/構建編譯器/calc7.py spi> 3+5*(3-3 )/3+3 6.0 spi>

插——后序遍歷

C語言代碼(有錯誤)

#include <stdio.h> #include <stdlib.h> #include <memory.h> #include <string.h> #include<math.h>#define flag_integer 0 #define flag_plus 1 #define flag_minus 2 #define flag_multiply 3 #define flag_divide 4 #define flag_LPAREN 5 #define flag_RPAREN 6#define flag_EOF 7void error() {printf("程序不對勁!\n");exit(-1); }//給字符數組賦值 void StrAssign(char* T, const char* chars) {int i;for (i = 0; i < strlen(chars); i++)T[i] = *(chars + i);T[i] = '\0'; }struct Token {int type;int value; };//Token初始化函數 struct Token* mallocToken(int type, int value) {struct Token* token = (Token*)malloc(sizeof(Token));if (NULL != token) {token->type = type;token->value = value;return token;}elseerror(); }//樹的節點 struct BinOpOrNum {struct BinOpOrNum* left;struct Token* op_or_num;struct BinOpOrNum* right; };//樹結點初始化函數 struct BinOpOrNum* mallocOpOrNum(struct BinOpOrNum* left, struct Token* op_or_num, struct BinOpOrNum* right) {struct BinOpOrNum* node = (BinOpOrNum*)malloc(sizeof(BinOpOrNum));if (NULL != node) {node->left = left;node->op_or_num = op_or_num;node->right = right;return node;}elseerror(); }struct Lexer {char* text;int pos; };struct Parser {struct Lexer* lexer;struct Token current_token; };struct Interpreter {struct Parser* parser; };void skip_whitespace(struct Lexer* le) {while (le->text[le->pos] == ' ') {le->pos++;} }//判斷Interpreter中當前pos是不是數字 int is_digital(char c) {if (c >= '0' && c <= '9')return 1;elsereturn 0; }void advance(struct Lexer* le) {le->pos++; }char current_char(struct Lexer* le) {return(le->text[le->pos]); }//獲取數字token的數值(把數字字符數組轉換為數字) int integer(struct Lexer* le) {char temp[20];int i = 0;while (is_digital(le->text[le->pos])) {temp[i] = le->text[le->pos];i++;advance(le);}int result = 0;int j = 0;int len = i;while (j < len) {result += (temp[j] - '0') * pow(10, len - j - 1);j++;}return result; }void get_next_token(struct Parser* par) {//先跳空格,再判斷有沒有結束符if (current_char(par->lexer) == ' ')skip_whitespace(par->lexer);if (par->lexer->pos > (strlen(par->lexer->text) - 1)) {par->current_token = { flag_EOF, NULL };return;}char current = current_char(par->lexer);if (is_digital(current)) {par->current_token = { flag_integer, integer(par->lexer) };return;}if (current == '+') {par->current_token = { flag_plus, NULL };advance(par->lexer);return;}if (current == '-') {par->current_token = { flag_minus, NULL };advance(par->lexer);;return;}if (current == '*') {par->current_token = { flag_multiply, NULL };advance(par->lexer);;return;}if (current == '/') {par->current_token = { flag_divide, NULL };advance(par->lexer);;return;}if (current == '(') {par->current_token = { flag_LPAREN, NULL };advance(par->lexer);;return;}if (current == ')') {par->current_token = { flag_RPAREN, NULL };advance(par->lexer);;return;}error();//如果都不是以上的字符,則報錯并退出程序 }int eat(struct Parser* par, int type) {int current_token_value = par->current_token.value;if (par->current_token.type == type) {get_next_token(par);return current_token_value;}else {error();} }//遍歷樹 int visit_BinOp(struct BinOpOrNum* node) {if (node->op_or_num->type == flag_plus)return visit_BinOp(node->left) + visit_BinOp(node->right);else if (node->op_or_num->type == flag_minus)return visit_BinOp(node->left) - visit_BinOp(node->right);else if (node->op_or_num->type == flag_multiply)return visit_BinOp(node->left) * visit_BinOp(node->right);else if (node->op_or_num->type == flag_divide)return visit_BinOp(node->left) / visit_BinOp(node->right);else if (node->op_or_num->type == flag_integer)return node->op_or_num->value; }struct BinOpOrNum* expr(struct Parser* par);//expr定義在后面,在這里要聲明才能使用 //判斷數字或括號 struct BinOpOrNum* factor(struct Parser* par) {int type = par->current_token.type;if (type == flag_integer) {struct Token* token = mallocToken(par->current_token.type, par->current_token.value);eat(par, flag_integer);struct BinOpOrNum* node = mallocOpOrNum(NULL, token, NULL);return node;}else if (type == flag_LPAREN) {eat(par, flag_LPAREN);//遇到括號先吃掉,然后回到exprstruct BinOpOrNum* node = expr(par);eat(par, flag_RPAREN);return node;} }//判斷乘除 struct BinOpOrNum* term(struct Parser* par) {struct BinOpOrNum* node;node = factor(par);int type = par->current_token.type;if (type == flag_multiply or type == flag_divide) {struct Token* token = mallocToken(par->current_token.type, par->current_token.value);eat(par, type);struct BinOpOrNum* node_ = mallocOpOrNum(node, token, factor(par));return node_;}else {return node;} }//判斷加減 struct BinOpOrNum* expr(struct Parser* par) {struct BinOpOrNum* node;node = term(par);int type = par->current_token.type;if (type == flag_plus or type == flag_minus) {struct Token* token = mallocToken(par->current_token.type, par->current_token.value);eat(par, type);struct BinOpOrNum* node_ = mallocOpOrNum(node, token, term(par));return node_;}else {return node;} }struct BinOpOrNum* parser(struct Parser* par) {return expr(par); }int interpreter(struct Interpreter* ipt) {struct BinOpOrNum* tree = parser(ipt->parser);return visit_BinOp(tree); }int main() {char text[50];//StrAssign(text, "(3+2) *7");StrAssign(text, " 3+ 2* ((3+7) + 5 )");struct Lexer le = { text, 0 };struct Parser par = { &le };get_next_token(&par);struct Interpreter ipt = { &par };int result = interpreter(&ipt);printf("%s = %d\n\n", text, result);return 0; }

運行結果:

3+ 2* ((3+7) + 5 ) = 33

但是程序貌似有bug啊,不知道是建立樹那里問題還是訪問樹那里問題,明天再調試了!

原來是expr和term函數不能缺少while循環,如果缺少的話就不能判斷多個加減號和乘除號了

同時我發現我里面有重定義的情況,找了半天才發現錯誤

修改后如下

//判斷乘除 struct BinOpOrNum* term(struct Parser* par) {struct BinOpOrNum* node = factor(par);while (par->current_token.type == flag_multiply or par->current_token.type == flag_divide) {struct Token* token = mallocToken(par->current_token.type, par->current_token.value);eat(par, par->current_token.type);node = mallocOpOrNum(node, token, factor(par));}return node; }//判斷加減 struct BinOpOrNum* expr(struct Parser* par) {struct BinOpOrNum* node = term(par);while (par->current_token.type == flag_plus or par->current_token.type == flag_minus) {struct Token* token = mallocToken(par->current_token.type, par->current_token.value);eat(par, par->current_token.type);//struct BinOpOrNum* node = mallocOpOrNum(node, token, term(par));//重定義了,還報錯使用未經初始化的指針,找了半天才發現問題node = mallocOpOrNum(node, token, term(par));}return node; }

C語言代碼(修改優化后)

#include <stdio.h> #include <stdlib.h> #include <memory.h> #include <string.h> #include<math.h>#define flag_integer 0 #define flag_plus 1 #define flag_minus 2 #define flag_multiply 3 #define flag_divide 4 #define flag_LPAREN 5 #define flag_RPAREN 6#define flag_EOF 7void error() {printf("程序不對勁!\n");exit(-1); }//給字符數組賦值 void StrAssign(char* T, const char* chars) {int i;for (i = 0; i < strlen(chars); i++)T[i] = *(chars + i);T[i] = '\0'; }struct Token {int type;int value; };//Token初始化函數 struct Token* mallocToken(int type, int value) {struct Token* token = (Token*)malloc(sizeof(Token));if (NULL != token) {token->type = type;token->value = value;return token;}elseerror(); }//樹的節點 struct BinOpOrNum {struct BinOpOrNum* left;struct Token* op_or_num;struct BinOpOrNum* right; };//樹結點初始化函數 struct BinOpOrNum* mallocOpOrNum(struct BinOpOrNum* left, struct Token* op_or_num, struct BinOpOrNum* right) {struct BinOpOrNum* node = (BinOpOrNum*)malloc(sizeof(BinOpOrNum));if (NULL != node) {node->left = left;node->op_or_num = op_or_num;node->right = right;return node;}elseerror(); }struct Lexer {char* text;int pos; };struct Parser {struct Lexer* lexer;struct Token current_token; };struct Interpreter {struct Parser* parser; };void skip_whitespace(struct Lexer* le) {while (le->text[le->pos] == ' ') {le->pos++;} }//判斷Interpreter中當前pos是不是數字 int is_digital(char c) {if (c >= '0' && c <= '9')return 1;elsereturn 0; }void advance(struct Lexer* le) {le->pos++; }char current_char(struct Lexer* le) {return(le->text[le->pos]); }//獲取數字token的數值(把數字字符數組轉換為數字) int integer(struct Lexer* le) {char temp[20];int i = 0;while (is_digital(le->text[le->pos])) {temp[i] = le->text[le->pos];i++;advance(le);}int result = 0;int j = 0;int len = i;while (j < len) {result += (temp[j] - '0') * pow(10, len - j - 1);j++;}return result; }void get_next_token(struct Parser* par) {//先跳空格,再判斷有沒有結束符if (current_char(par->lexer) == ' ')skip_whitespace(par->lexer);if (par->lexer->pos > (strlen(par->lexer->text) - 1)) {par->current_token = { flag_EOF, NULL };return;}char current = current_char(par->lexer);if (is_digital(current)) {par->current_token = { flag_integer, integer(par->lexer) };return;}if (current == '+') {par->current_token = { flag_plus, NULL };advance(par->lexer);return;}if (current == '-') {par->current_token = { flag_minus, NULL };advance(par->lexer);;return;}if (current == '*') {par->current_token = { flag_multiply, NULL };advance(par->lexer);;return;}if (current == '/') {par->current_token = { flag_divide, NULL };advance(par->lexer);;return;}if (current == '(') {par->current_token = { flag_LPAREN, NULL };advance(par->lexer);;return;}if (current == ')') {par->current_token = { flag_RPAREN, NULL };advance(par->lexer);;return;}error();//如果都不是以上的字符,則報錯并退出程序 }int eat(struct Parser* par, int type) {int current_token_value = par->current_token.value;if (par->current_token.type == type) {get_next_token(par);return current_token_value;}else {error();} }//遍歷樹 int visit_BinOp(struct BinOpOrNum* node) {if (node->op_or_num->type == flag_plus)return visit_BinOp(node->left) + visit_BinOp(node->right);else if (node->op_or_num->type == flag_minus)return visit_BinOp(node->left) - visit_BinOp(node->right);else if (node->op_or_num->type == flag_multiply)return visit_BinOp(node->left) * visit_BinOp(node->right);else if (node->op_or_num->type == flag_divide)return visit_BinOp(node->left) / visit_BinOp(node->right);else if (node->op_or_num->type == flag_integer)return node->op_or_num->value; }struct BinOpOrNum* expr(struct Parser* par);//expr定義在后面,在這里要聲明才能使用 //判斷數字或括號 struct BinOpOrNum* factor(struct Parser* par) {if (par->current_token.type == flag_integer) {struct Token* token = mallocToken(par->current_token.type, par->current_token.value);eat(par, flag_integer);struct BinOpOrNum* node = mallocOpOrNum(NULL, token, NULL);return node;}else if (par->current_token.type == flag_LPAREN) {eat(par, flag_LPAREN);//遇到括號先吃掉,然后回到exprstruct BinOpOrNum* node = expr(par);eat(par, flag_RPAREN);return node;} }//判斷乘除 struct BinOpOrNum* term(struct Parser* par) {struct BinOpOrNum* node = factor(par);while (par->current_token.type == flag_multiply or par->current_token.type == flag_divide) {struct Token* token = mallocToken(par->current_token.type, par->current_token.value);eat(par, par->current_token.type);node = mallocOpOrNum(node, token, factor(par));}return node; }//判斷加減 struct BinOpOrNum* expr(struct Parser* par) {struct BinOpOrNum* node = term(par);while (par->current_token.type == flag_plus or par->current_token.type == flag_minus) {struct Token* token = mallocToken(par->current_token.type, par->current_token.value);eat(par, par->current_token.type);//struct BinOpOrNum* node = mallocOpOrNum(node, token, term(par));//重定義了,還報錯使用未經初始化的指針,找了半天才發現問題node = mallocOpOrNum(node, token, term(par));}return node; }struct BinOpOrNum* parser(struct Parser* par) {return expr(par); }int interpreter(struct Interpreter* ipt) {struct BinOpOrNum* tree = parser(ipt->parser);return visit_BinOp(tree); }int main() {char text[50];//StrAssign(text, "(3+2) *7");StrAssign(text, " 3+2*(3+7)+(1 - (3+7)/5)* (2/2+3 )* 1");struct Lexer le = { text, 0 };struct Parser par = { &le };get_next_token(&par);struct Interpreter ipt = { &par };int result = interpreter(&ipt);printf("%s = %d\n\n", text, result);return 0; }

運行結果:

3+2*(3+7)+(1 - (3+7)/5)* (2/2+3 )* 1 = 19

總結

今天,您已經了解了解析樹、AST、如何構造 AST 以及如何遍歷它們以解釋由這些 AST 表示的輸入。您還修改了解析器和解釋器并將它們分開。詞法分析器、解析器和解釋器之間的當前接口現在看起來像這樣:

您可以將其理解為“解析器從詞法分析器中獲取標記,然后返回生成的AST以供解釋器遍歷和解釋輸入。”

今天就到此為止,但在結束之前,我想簡單地談談遞歸下降解析器,即給它們一個定義,因為我上次承諾會更詳細地討論它們。所以你開始了:遞歸下降解析器是一個自頂向下的解析器,它使用一組遞歸過程來處理輸入。自頂向下反映了解析器從構造解析樹的頂部節點開始,然后逐漸構造較低節點的事實。

現在是練習的時候了:)

  • 編寫一個翻譯器(提示:節點訪問者),將算術表達式作為輸入并以后綴表示法打印出來,也稱為反向波蘭表示法 ( RPN )。例如,如果翻譯器的輸入是表達式 (5 + 3) * 12 / 3,那么輸出應該是 5 3 + 12 * 3 /。請在此處查看答案,但請先嘗試自己解決。
  • 編寫一個翻譯器(節點訪問者),將算術表達式作為輸入并以LISP風格的符號打印出來,即 2 + 3 將變為 (+ 2 3) 并且 (2 + 3 * 5) 將變為 (+ 2 (* 3 5))。您可以在此處找到答案,但在查看提供的解決方案之前再次嘗試先解決它。

在下一篇文章中,我們將向不斷增長的 Pascal 解釋器添加賦值和一元運算符。在那之前,玩得開心,很快就會見到你。

PS我還提供了解釋器的 Rust 實現,您可以在GitHub上找到。這是我學習Rust 的一種方式,所以請記住,代碼可能還不是“慣用的”。關于如何改進代碼的意見和建議總是受歡迎的。

總結

以上是生活随笔為你收集整理的【编译原理】构建一个简单的解释器(Let’s Build A Simple Interpreter. Part 7.)(笔记)解释器 interpreter 解析器 parser 抽象语法树AST的全部內容,希望文章能夠幫你解決所遇到的問題。

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

久草资源在线 | 免费在线黄色av | 国产成人精品一区二区三区在线 | 超级碰碰碰碰 | 91精品一区二区三区蜜臀 | 欧美午夜精品久久久久久孕妇 | 亚洲 欧美 国产 va在线影院 | 九九热99视频 | 欧美一级性生活 | 一区二区三区视频网站 | 亚洲精品欧美专区 | 香蕉视频免费看 | 九色视频网址 | 成人黄色电影免费观看 | 91mv.cool在线观看 | 波多野结衣在线视频一区 | 在线观看国产一区 | 国产 视频 高清 免费 | 日韩欧美aaa| 五月天综合色 | 久久视频免费看 | 精品自拍sae8—视频 | 少妇性aaaaaaaaa视频 | 国产精品一区二区你懂的 | 国产不卡精品视频 | 在线观看va | 国产精品一级在线 | 91一区二区三区在线观看 | 最新免费中文字幕 | 亚洲精品女人 | 国产精品mv| 亚洲最大免费成人网 | 中文字幕成人网 | 91精品国产成人www | 深夜福利视频一区二区 | 精精国产xxxx视频在线播放 | 欧美色图30p| 欧美视频在线观看免费网址 | 丁香六月综合网 | 国内精品视频在线播放 | 高清一区二区三区 | 亚洲精品在线观看网站 | 天天天天天天天天操 | 少妇啪啪av入口 | 精品国产伦一区二区三区观看说明 | 国产在线视频一区二区三区 | 久热国产视频 | 日韩久久久 | 欧美成人在线免费 | 亚洲一区二区精品视频 | 久久精品99国产精品日本 | 国产午夜免费视频 | 精品国产一区二区三区免费 | 国产视频日韩 | 日本色小说视频 | 欧美性久久久久久 | 最新中文字幕 | 免费合欢视频成人app | 久久亚洲欧美日韩精品专区 | 久久久久久久久精 | 西西www444 | 日本h视频在线观看 | 欧美成人在线免费 | 日日夜夜精品视频 | 国产情侣一区 | 热久久免费视频精品 | 国产高清视频在线观看 | 在线精品亚洲一区二区 | 久久综合欧美 | 国产一区二区三区视频在线 | 免费观看一级特黄欧美大片 | 四虎国产精品成人免费影视 | 国产精品久久久久av免费 | 精品国产视频一区 | 毛片美女网站 | 久久久久久久久久网 | av黄色av| 亚洲欧美成人网 | 91精品国产91p65 | av成人在线观看 | 最新av在线播放 | 丁香五婷 | 欧美日本日韩aⅴ在线视频 插插插色综合 | 麻豆国产在线视频 | 免费观看91 | 韩国三级av在线 | 99r在线精品 | 精品久久久久久国产 | 日韩有色| 久久精品99国产精品酒店日本 | 亚洲国产精品成人综合 | 91视频高清免费 | 久久96国产精品久久99软件 | 婷婷在线网 | 在线精品视频免费播放 | 国产精品九九视频 | 日韩视频在线一区 | 国偷自产中文字幕亚洲手机在线 | 免费看片成人 | 成人精品视频久久久久 | 91视频高清 | 久青草影院 | 精品国产一区二区三区在线观看 | 国产中文字幕一区二区三区 | 激情av网| 91av视频 | 色噜噜在线观看视频 | 天天天天天操 | 亚洲精品国产综合久久 | 五月天久久婷 | 最新色站| 国产精品综合在线观看 | 成人免费观看在线视频 | 一区二区三区免费在线观看视频 | 又湿又紧又大又爽a视频国产 | 国产中的精品av小宝探花 | 日本在线观看中文字幕 | 激情婷婷久久 | 亚洲国产wwwccc36天堂 | 成年人免费观看在线视频 | 国产视频在线一区二区 | 在线观看黄污 | 久久精品这里热有精品 | 美女在线免费观看视频 | 在线免费亚洲 | 狠狠做深爱婷婷综合一区 | 日韩高清一区 | 久久精选 | 免费日韩视| 日韩高清精品一区二区 | 91黄色在线视频 | 亚洲黄色激情小说 | 中文字幕日韩有码 | 精品福利在线 | 久久人人添人人爽添人人88v | 日韩成人免费在线 | 成人9ⅰ免费影视网站 | 久久艹欧美 | 特级西西www44高清大胆图片 | 97超碰成人 | 午夜精品一区二区三区在线播放 | 色播五月婷婷 | 亚州精品在线视频 | 婷婷网站天天婷婷网站 | 久久久午夜影院 | 久久人人精 | 人人要人人澡人人爽人人dvd | 国产视频999 | 8x成人免费视频 | 国产一区在线免费观看 | 国产精品中文久久久久久久 | 久久久免费观看完整版 | 日本精油按摩3 | 亚洲精品动漫在线 | 在线亚洲激情 | 伊人天堂久久 | 91成品人影院 | 又黄又爽又刺激 | 日日操天天操狠狠操 | 久久久久国 | 91专区在线观看 | 国产99久久久精品视频 | 色综合天天狠狠 | 91男人影院 | 韩国三级在线一区 | 日韩视频在线不卡 | 国产精品免费av | 国产麻豆精品久久一二三 | 天天干,天天射,天天操,天天摸 | 欧美精品在线观看一区 | 国产在线色视频 | 国产一级特黄毛片在线毛片 | 99精品电影 | 欧美成人精品三级在线观看播放 | 日本成人黄色片 | 日韩在线第一 | 亚洲精品中文在线 | 欧美日韩一区二区免费在线观看 | 午夜性生活片 | 四月婷婷在线观看 | 久久热首页 | 成人 国产 在线 | 91九色成人 | 国产精品一区二区在线观看免费 | 天天插狠狠干 | av在线看网站| 狠狠88综合久久久久综合网 | 亚洲精品视频在线免费播放 | 日日操天天爽 | www.天天操.com | 免费韩国av| 91成年视频 | 日韩欧美在线一区二区 | 中文字幕精品久久 | 视频在线观看国产 | 久久久麻豆视频 | 日本精品久久久久中文字幕 | 欧美久久电影 | 在线影院av | 97精品国产97久久久久久久久久久久 | 国产成人精品一二三区 | 国语自产偷拍精品视频偷 | 天天干天天操天天搞 | 日韩免费av网址 | 国产一区成人在线 | 国产免费观看高清完整版 | 四虎影视成人永久免费观看视频 | 中文字幕视频播放 | 伊人激情网| 国产精品三级视频 | 久久99久久99精品免视看婷婷 | 久久观看免费视频 | 国产乱对白刺激视频不卡 | 亚洲国产小视频在线观看 | 久久精品国产久精国产 | 日韩av免费大片 | 在线观看免费中文字幕 | 99c视频在线| 日韩大片在线看 | 日韩精品一区二区免费视频 | 国产一级片网站 | 国产中文在线播放 | 97成人精品视频在线观看 | 最近中文字幕在线 | 婷婷丁香七月 | 色视频成人在线观看免 | 久久五月天综合 | 国产福利中文字幕 | 国产成人精品免高潮在线观看 | 在线视频日韩精品 | 狠狠躁夜夜躁人人爽超碰97香蕉 | 国产 一区二区三区 在线 | 亚洲国产成人高清精品 | 四虎国产精品免费 | 美女精品网站 | 欧美日韩国产二区三区 | 国产精品美女免费 | 狠狠操狠狠干天天操 | 中国老女人日b | 中文字幕高清有码 | 91亚洲欧美| 九九九九九国产 | 午夜久久网站 | 久久一区91 | 欧美一级乱黄 | 97视频一区 | 色午夜影院 | 日韩在线视频在线观看 | 精品天堂av | 精品视频在线播放 | 免费网站看av片 | 在线亚洲观看 | 天天鲁天天干天天射 | 精品国产一区二区三区久久影院 | 亚洲精品国产精品99久久 | 亚洲精品在线视频观看 | 日日天天狠狠 | 91原创在线观看 | 在线看日韩| 九九精品久久久 | 亚洲无吗av | 久草剧场 | 欧美ⅹxxxxxx | 久久精品亚洲一区二区三区观看模式 | 国产精品久久久久久久久久久久午夜片 | 成人av免费| 国产99re| 成人久久久久久久久久 | 久久视频99 | 日韩久久精品 | 久久精品视频99 | 狠狠干 狠狠操 | 97伊人网| 久久久久免费精品国产 | 欧美日韩大片在线观看 | 中文国产成人精品久久一 | www.久热 | 国产精品伦一区二区三区视频 | 国产一区久久久 | 久久理伦片 | 韩国精品视频在线观看 | 一级片免费在线 | 激情综合国产 | 久久久精品免费看 | 亚洲精品一区二区三区四区高清 | 国产精品黄色av | 黄色日本免费 | 免费在线黄网 | 日韩欧美综合 | 国产精品午夜8888 | 亚洲欧美日韩一级 | 日本h在线播放 | 在线看国产 | 五月开心网 | 男女免费av | 日韩欧美在线一区 | 国产在线精品二区 | 国产精品久久二区 | 福利视频午夜 | 日韩欧美在线第一页 | 美女福利视频一区二区 | 午夜美女网站 | 欧美日本三级 | 久草在线在线视频 | 91手机视频| 久久精品小视频 | 三级黄色片在线观看 | 久久久影院一区二区三区 | 亚州av网站 | 91成人区| 午夜精品一区二区三区可下载 | 狠狠干夜夜爽 | 国产免费黄色 | 日日精品| 国产精品免费视频观看 | 99精品视频一区二区 | 丰满少妇高潮在线观看 | 欧美天天综合 | av 一区二区三区 | 天堂在线视频免费观看 | 99人久久精品视频最新地址 | av色图天堂网 | 五月婷婷综合在线视频 | 亚洲三区在线 | 亚洲2019精品 | 国产精品久久久久久一二三四五 | 久久天天躁狠狠躁夜夜不卡公司 | 91成年人在线观看 | 精品一区二区免费 | 日韩中文字幕国产精品 | 免费日韩一区二区三区 | 日韩精品中文字幕在线不卡尤物 | 波多野结衣在线播放一区 | 亚洲黄色片一级 | 免费在线播放黄色 | 久艹视频在线观看 | av免费网站在线观看 | 一区二区三区免费在线播放 | 久久精品视频网 | 国产日韩欧美综合在线 | 98精品国产自产在线观看 | 一区二区三区高清 | 日韩中文字幕国产精品 | 婷婷色在线 | 久久综合成人网 | 免费欧美精品 | 日韩精品视频在线免费观看 | 99国产精品一区二区 | 五月天视频网站 | 四虎国产 | 国内精品久久久久久久影视麻豆 | 美女视频免费一区二区 | 色欧美成人精品a∨在线观看 | 亚洲国产视频a | 亚洲欧美日韩国产一区二区三区 | 日韩在线影视 | 大型av综合网站 | 久草在线视频精品 | 国产精品久久久久久久免费观看 | 免费精品在线视频 | 欧美伦理一区 | av在线一 | 国产久草在线观看 | 99在线视频播放 | 亚洲国产mv | 亚洲国产精品成人精品 | 国产美女视频免费 | 四虎影视成人精品国库在线观看 | 久久er99热精品一区二区 | 在线精品视频在线观看高清 | 久久不卡免费视频 | 精品成人久久 | 亚洲精品乱码久久 | 毛片网在线播放 | 91插插视频| 中文字幕一区二区三区在线观看 | 国产亚洲精品久久久久久网站 | 国产97在线观看 | 婷婷久操| 精品久久网站 | 天天操天天操天天操 | 成人免费观看在线视频 | 日日干天天射 | 综合色影院 | 免费国产亚洲视频 | 日韩视频一区二区在线观看 | 91人人爽久久涩噜噜噜 | 日日干夜夜干 | 欧美精品久久久 | 久久99国产精品二区护士 | 色婷婷色 | 国产成人三级在线观看 | 亚洲一区日韩精品 | 国产黄网在线 | 欧美日韩成人 | 天天夜夜操| 日韩激情免费视频 | 精品一区二区综合 | 青草视频网 | 国产精品一区二区你懂的 | 国产精品毛片久久 | 精品国内自产拍在线观看视频 | av片一区| 亚洲成人精品久久 | 久久久精品免费看 | 懂色av一区二区三区蜜臀 | 麻豆视频国产在线观看 | 98久9在线 | 免费 | 久久国产亚洲精品 | 97电影在线观看 | 免费影视大全推荐 | 国产又粗又猛又黄又爽的视频 | 久久综合久久综合这里只有精品 | 中文字幕在线资源 | 欧美视频二区 | 美女免费视频一区二区 | 天天射天天操天天 | 97免费| 日韩在线电影一区 | 人人爽人人爽人人爽人人爽 | 视频在线观看入口黄最新永久免费国产 | 最近免费在线观看 | 日韩精品一区二区三区外面 | 免费观看完整版无人区 | 国产亚洲日本 | 91精品国产麻豆国产自产影视 | www.狠狠插.com | 日韩二区三区在线观看 | 视频一区二区在线观看 | 最新久久久 | 久久婷婷国产色一区二区三区 | 91在线欧美 | av三级在线免费观看 | 亚洲va欧美va人人爽春色影视 | 九九九热 | 日韩成人免费在线观看 | 视频国产精品 | 亚洲精品黄色在线观看 | 欧美日韩高清在线一区 | 久久视频网| 日韩欧美精品一区二区三区经典 | 日本3级在线观看 | 久福利 | 国产精品免费不 | 免费www视频 | 免费黄a | 97电影在线看视频 | 国产精品18久久久久久vr | 中文字幕免费高清 | 97涩涩视频| 日本深夜福利视频 | 激情视频免费在线观看 | 亚洲视屏在线播放 | 国产精品久久久久永久免费观看 | 91精品无人成人www | 99国产精品视频免费观看一公开 | 国产一区二区不卡在线 | 中文在线免费一区三区 | 色在线中文字幕 | 亚洲国产精品久久久久婷婷884 | 丰满少妇久久久 | 中文字幕国产视频 | 精品国内自产拍在线观看视频 | 日韩欧美高清 | 黄色电影网站在线观看 | 悠悠av资源片 | 国产精品黄色影片导航在线观看 | 狠狠综合网 | 国产视频丨精品|在线观看 国产精品久久久久久久久久久久午夜 | 欧美黄网站 | 99久久网站 | 久青草视频 | 天天看天天操 | 中文字幕观看av | 亚洲资源在线观看 | 国产视频一二区 | 久久午夜视频 | 国内精品久久久久久久影视简单 | 久久国产精品99久久人人澡 | 日韩精品2区 | 4p变态网欧美系列 | 日韩精品一区二区三区在线视频 | 欧美日韩中文字幕在线视频 | 国产精品免费久久 | 亚洲草视频 | 久久色视频 | 在线日韩av | 国产黄色特级片 | 国产999久久久 | 亚洲精品中文在线 | 国产在线精品国自产拍影院 | 狠狠躁夜夜躁人人爽视频 | 国产精品理论视频 | www.久艹| 亚洲综合欧美日韩狠狠色 | 在线v片免费观看视频 | 亚洲精品在线一区二区 | 伊人精品在线 | 日韩欧美在线一区二区 | 韩日三级在线 | 天海冀一区二区三区 | 国产在线视频导航 | 久久亚洲免费 | 91桃色在线播放 | 中文字幕在线免费看 | 97高清视频 | 久久无码精品一区二区三区 | 中文字幕三区 | 日韩伦理片一区二区三区 | 欧美日韩国产在线观看 | 美女网色| 99精品国产亚洲 | 日日操夜| 中文字幕在线观看国产 | 人人看看人人 | 婷婷色婷婷 | 精品久久久久久久久久岛国gif | 日韩av片无码一区二区不卡电影 | 久久99精品久久久久久清纯直播 | 国产一区私人高清影院 | www.婷婷com | 国产精品短视频 | 久久福利国产 | 亚洲国产欧洲综合997久久, | 国产亚洲精品久久久久久久久久 | 国产精品露脸在线 | 在线视频 区 | 美女在线黄 | 久久综合久久综合这里只有精品 | 久久精品导航 | 国产精品国产亚洲精品看不卡15 | av黄色成人 | 麻豆传媒一区二区 | 亚洲综合色丁香婷婷六月图片 | 欧洲av在线| 中文字幕免费看 | 国产99一区二区 | 久久精品亚洲 | 久久综合久久综合九色 | 免费看黄电影 | 91在线精品播放 | 日韩午夜剧场 | 六月色丁 | 99re热精品视频 | 奇米网网址| 亚洲视频久久 | 精品福利视频在线 | 亚洲免费av片 | 丁香六月综合网 | 久久久久区 | jizz18欧美18| 日本久久精品视频 | 精精国产xxxx视频在线播放 | 中文在线a∨在线 | 亚洲www天堂com | 亚洲国产精品电影在线观看 | 黄色成年 | 激情丁香综合五月 | 婷婷五情天综123 | 日韩精品一区二区在线观看 | 人人藻人人澡人人爽 | 久久国产精品久久精品 | 日韩免费在线 | 久久久久久网址 | 你操综合 | 国产一区福利在线 | 在线你懂 | 免费看片成人 | 日韩专区av | 亚洲精品视频在线免费 | 免费三级av | 国产手机在线精品 | 久久精品综合视频 | 久久综合久久综合久久 | 欧美一级免费高清 | 日韩国产精品一区 | 久操97| 国产永久免费高清在线观看视频 | 国产在线高清精品 | 亚洲成av人片在线观看无 | 在线观看久久久久久 | 久一久久 | 国产视频不卡一区 | 俺要去色综合狠狠 | 久久人人爽人人人人片 | 免费在线观看日韩欧美 | 99久热 | 国产高清视频免费最新在线 | 日韩av不卡在线播放 | 国产精品久久久999 国产91九色视频 | 亚洲视频在线免费观看 | 一区二区三区久久 | 韩国精品在线观看 | 新版资源中文在线观看 | 国产精品免费一区二区三区 | 亚洲第一色 | 天天爱综合 | 免费亚洲视频在线观看 | 久久男人免费视频 | 欧美成人亚洲 | 国产91欧美 | 欧美激情片在线观看 | 免费看一级特黄a大片 | 91九色在线视频观看 | 亚洲午夜小视频 | 99r精品视频在线观看 | 99色婷婷 | 九精品 | 在线免费视频你懂的 | 五月天.com | 久久久亚洲网站 | 久久久男人的天堂 | 婷婷在线精品视频 | 国产精品一区二区三区免费看 | 欧美精品少妇xxxxx喷水 | 一级α片免费看 | 六月色播 | 99国产成+人+综合+亚洲 欧美 | 999精品在线 | 香蕉视频免费看 | 国产精品密入口果冻 | 色婷婷天天干 | 在线视频中文字幕一区 | 国产精品久久电影网 | 夜夜躁日日躁狠狠久久88av | 麻豆免费精品视频 | 91桃色国产在线播放 | 91九色精品女同系列 | 久久99中文字幕 | 深爱综合网 | 亚洲激情 欧美激情 | 久草网站在线观看 | 激情av网址 | 九色精品免费永久在线 | 成人一级 | 色婷婷综合久久久中文字幕 | 成人黄色片在线播放 | 激情婷婷在线 | 久久国产精品99久久久久久老狼 | 激情欧美日韩一区二区 | 久久精选 | 99久久精品国产亚洲 | 人人爽久久涩噜噜噜网站 | 中文字幕视频观看 | 久久久久伊人 | 狠狠色丁香九九婷婷综合五月 | 国产一级免费在线观看 | h动漫中文字幕 | 亚洲91在线 | 亚洲va欧美va国产va黑人 | 黄色在线免费观看网站 | 国产福利一区二区三区在线观看 | 成人一区影院 | 激情av在线播放 | 九九热在线视频 | 国产精品原创在线 | 日韩在线免费观看视频 | 4438全国亚洲精品观看视频 | 国产爽妇网 | 娇妻呻吟一区二区三区 | 国产亚洲精品成人av久久ww | 国产精品久久久免费看 | 色综合激情网 | 91九色视频观看 | 久久久久久久久久久网 | 波多野结衣综合网 | 免费在线播放av电影 | 国产日韩在线看 | 永久av免费在线观看 | 丝袜美腿亚洲综合 | 香蕉久久久久久av成人 | 狠狠插狠狠干 | 精品国产一区二区三区在线观看 | 天天综合网 天天 | 国产不卡视频在线播放 | 日韩欧美在线中文字幕 | av在线永久免费观看 | 日批视频| 国产精品黄色影片导航在线观看 | 在线视频观看你懂的 | 亚洲美女视频在线 | 97爱爱爱 | 日韩欧美精品在线观看视频 | 51久久成人国产精品麻豆 | 亚洲国产精品一区二区久久,亚洲午夜 | 麻豆视屏 | 91精品久久久久久久91蜜桃 | 97超碰福利久久精品 | 欧美一级在线 | 在线观看完整版 | 国产中文欧美日韩在线 | av免费成人 | 久久久久国产一区二区三区四区 | 99久久综合国产精品二区 | 全黄色一级片 | 91最新在线观看 | 日韩毛片在线一区二区毛片 | 日韩中字在线观看 | 日韩专区在线播放 | 黄色av电影在线 | av观看在线观看 | 国内精品久久久久影院优 | 狠狠干干 | 一级免费片 | 国产精品免费一区二区三区 | 久久极品 | 国产高清在线观看av | 综合色播| 国产69精品久久久久久 | 亚洲午夜精品在线观看 | 在线观看一区二区精品 | 中文字幕中文字幕中文字幕 | 精品久操 | 成 人 黄 色 片 在线播放 | 亚洲天堂网在线播放 | 亚洲一区精品人人爽人人躁 | 成人在线免费观看网站 | 国产麻豆剧传媒免费观看 | 天天干天天拍天天操 | 国产精品91一区 | 欧洲在线免费视频 | 蜜桃久久久| 成人黄大片视频在线观看 | 欧洲亚洲女同hd | 色综合亚洲精品激情狠狠 | 在线电影日韩 | 永久免费精品视频网站 | 在线观看爱爱视频 | 九九久久视频 | ww亚洲ww亚在线观看 | 久久成人久久 | 日本大片免费观看在线 | 久久激情电影 | 永久免费视频国产 | 久久免费福利 | 在线观看视频h | 中文字幕婷婷 | 最近字幕在线观看第一季 | 色香com.| 91久久国产露脸精品国产闺蜜 | 国内精品久久久久久久影视简单 | 亚洲码国产日韩欧美高潮在线播放 | 国产一区二区在线影院 | 国产精品一区二区精品视频免费看 | 久久视频精品 | 国产乱对白刺激视频不卡 | 国产69精品久久久久久久久久 | 91av视频在线播放 | 精品影院一区二区久久久 | 日韩大片在线免费观看 | 中文字幕亚洲精品在线观看 | 天天拍天天干 | 九色精品 | 日韩在线视 | 欧美日本日韩aⅴ在线视频 插插插色综合 | 97看片| 日本女人b | 久久久av电影 | 亚洲精品毛片一级91精品 | www.亚洲精品在线 | 欧美午夜视频在线 | 欧美色图视频一区 | 免费观看一区二区三区视频 | 特级a老妇做爰全过程 | 国产成人精品av | 69国产精品视频 | 日韩综合在线观看 | 深夜视频久久 | 国产一级久久 | 91精品一区二区三区蜜臀 | 国产v在线 | 九九久久成人 | 九九热免费在线视频 | 免费在线观看成年人视频 | 欧美在线观看视频免费 | 久久久污 | 99免费国产| 久久婷婷色综合 | 丝袜一区在线 | av片一区 | 国产精品久99 | 亚洲人毛片 | 久久爱资源网 | 久久久久麻豆 | 国产精品高潮久久av | 欧美极品少妇xbxb性爽爽视频 | 热久久免费视频精品 | 五月亚洲婷婷 | 午夜色大片在线观看 | av在线网站免费观看 | 一区二区三区免费在线播放 | 91污在线观看 | 中文国产在线观看 | 国内外成人在线视频 | 亚洲 综合 精品 | 国产精品久久久久av | 中国美女一级看片 | 欧美日韩精品在线播放 | 亚洲国产精品成人女人久久 | 99视频偷窥在线精品国自产拍 | 国产一二三区av | 日韩在线在线 | 久久欧美视频 | 亚洲精品999 | 永久免费av在线播放 | 久久精品成人热国产成 | 久久久久亚洲精品成人网小说 | 一区二区中文字幕在线 | 亚洲区另类春色综合小说校园片 | www·22com天天操 | 啪啪免费试看 | 久久久久免费网站 | 久久99久久99精品 | 500部大龄熟乱视频 欧美日本三级 | 综合黄色网 | 91精品国产91p65 | 日韩免费网址 | 又黄又爽的免费高潮视频 | 最新一区二区三区 | 日韩大片在线免费观看 | 天天操夜夜叫 | 免费网站色 | 国产拍在线 | 久久丁香 | 精品视频免费看 | 精品国产伦一区二区三区 | 国产麻豆电影 | 成人国产精品av | 人人爽人人干 | 日韩美女久久 | 色多多在线观看 | 日韩激情网| 成人中文字幕在线观看 | 超碰人在线 | 国产无遮挡猛进猛出免费软件 | 久久久久国产一区二区三区 | 日本动漫做毛片一区二区 | 香蕉视频啪啪 | 午夜免费福利视频 | 四虎影视成人永久免费观看亚洲欧美 | 91精品亚洲影视在线观看 | 天天透天天插 | 中文字幕中文中文字幕 | 免费a视频在线 | 99操视频| 高清不卡毛片 | 日韩一级片网址 | 99精品在线观看视频 | 国产一级大片免费看 | 国产一级一片免费播放放 | 一区二区三区在线看 | 久久久精品二区 | 日批视频在线观看免费 | 中文字幕一区二区三区在线播放 | 91av在线国产 | 国产精品18久久久久久不卡孕妇 | 日韩av资源站 | 久久成人国产 | 精品亚洲成a人在线观看 | 天天av资源 | 中文字幕亚洲高清 | 国产亚洲高清视频 | 91香蕉视频720p | 婷婷激情网站 | 超碰97人人在线 | 成人免费视频在线观看 | 日韩av在线资源 | 五月天综合色 | 女人18精品一区二区三区 | 免费又黄又爽的视频 | 人人爱爱 | 三级午夜片 | 久久久久久久久久久久国产精品 | 日韩欧美精品在线观看 | 欧美日韩一级久久久久久免费看 | 日韩视频欧美视频 | 国产日产在线观看 | 亚洲综合激情小说 | 国产一区二区三区视频在线 | 免费福利在线视频 | 久久久久久高潮国产精品视 | 免费人成在线观看网站 | 国产精品 视频 | 91伊人久久大香线蕉蜜芽人口 | 69av国产 | 国内外成人免费在线视频 | 久久久久久不卡 | jizz18欧美18| 青青啪| 中文字幕第一页在线视频 | 欧美极品少妇xxxxⅹ欧美极品少妇xxxx亚洲精品 | 国产拍揄自揄精品视频麻豆 | 狠狠躁日日躁狂躁夜夜躁 | 精品久久久网 | 91九色成人蝌蚪首页 | 91精品国自产在线观看 | 99久久精品国产免费看不卡 | 国产成人精品av在线 | 久久99网| 2020天天干天天操 | 国产专区精品视频 | 免费看毛片网站 | 激情五月激情综合网 | 99精品久久久久久久久久综合 | 国产精品嫩草影视久久久 | 日本精油按摩3 | 人人爽人人插 | 国产精品嫩草55av | 欧亚久久| 久久久久久久免费观看 | 日韩欧美成人网 | 久久网页| 日韩视频中文字幕在线观看 | 摸bbb搡bbb搡bbbb | 四虎在线免费观看 | 尤物97国产精品久久精品国产 | 激情图片区| 在线色网站 | 免费观看9x视频网站在线观看 | 国产精品久久久久久欧美 | 国产超碰在线观看 | 亚洲精品日韩一区二区电影 | 成 人 黄 色 视频播放1 | 久久人人爽 | 精品一区 在线 | 国产乱对白刺激视频不卡 | 波多野结衣在线播放一区 | 久青草视频 | 久久久久成人免费 | 狠狠干综合网 | 久久一二区 | 99国产精品一区 | 亚洲女欲精品久久久久久久18 | 成人av免费在线观看 | 欧美性久久久久久 | 欧美不卡视频在线 | 婷婷av色综合 | 91天天操| 91av在线视频免费观看 | 欧日韩在线 | av黄色影院 | 国产一区二区三区免费观看视频 | 亚洲 中文 欧美 日韩vr 在线 | 精品一区二区免费视频 | 国产精品一区二区视频 | 国产精品嫩草影视久久久 | 国产福利网站 | 亚洲精品乱码久久久一二三 | 欧美大片第1页 | 国产91全国探花系列在线播放 | www.色国产| 久久99久久精品 | 免费一级黄色 | 久久久久观看 | 国内精品久久久 | 久久草在线免费 | 免费看国产曰批40分钟 | .精品久久久麻豆国产精品 亚洲va欧美 | 亚洲精品久久激情国产片 | 香蕉视频在线免费 | 欧美一级免费片 | 国产乱码精品一区二区蜜臀 | 久久综合九色综合欧美就去吻 | 国产精品3 | 91日韩精品视频 | 手机在线中文字幕 | 91精品国产91久久久久久三级 | 亚洲精品午夜久久久 | 亚洲午夜精品在线观看 | 国产亚洲精品成人 | 欧美黑人性猛交 | 日本成人中文字幕在线观看 | 国产天天爽 | 免费av观看网站 | 美女久久久久 | 成人av一区二区在线观看 | 精品国产乱码久久久久久三级人 | 97精产国品一二三产区在线 | av中文字幕网站 | 亚洲黄色一级大片 | 97超碰在线播放 | 国产电影黄色av | 国产精品日韩在线 | 青青河边草免费视频 | 伊人午夜| 在线免费av观看 | 国产91丝袜在线播放动漫 | 久久久久免费网 | 国产视频在线播放 | 日韩 国产 | 久久精品中文字幕 | 香蕉在线视频观看 | 特黄一级毛片 | 国产美女主播精品一区二区三区 | 九九热精品视频在线观看 | 欧美极品少妇xxxxⅹ欧美极品少妇xxxx亚洲精品 | 久青草视频在线观看 | 成年人在线观看网站 |