国产无遮挡裸体免费直播视频,久久精品国产蜜臀av,动漫在线视频一区二区,欧亚日韩一区二区三区,久艹在线 免费视频,国产精品美女网站免费,正在播放 97超级视频在线观看,斗破苍穹年番在线观看免费,51最新乱码中文字幕

Go手寫數(shù)據(jù)庫ZiyiDB的實現(xiàn)

 更新時間:2025年05月25日 10:31:24   作者:NPE~  
本文主要介紹了Go手寫數(shù)據(jù)庫ZiyiDB的實現(xiàn),文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧

ZiyiDB是一個簡單的內(nèi)存數(shù)據(jù)庫實現(xiàn),支持基本的SQL操作,包含create、insert、delete、select、update、drop。目前一期暫支持int類型以及字符類型數(shù)據(jù),后續(xù)會支持更多數(shù)據(jù)結(jié)構(gòu)以及能力。本項目基于https://github.com/eatonphil/gosql靈感進行開發(fā)。

項目結(jié)構(gòu)

// 項目創(chuàng)建
mkdir ZiyiDB
cd ZiyiDB/
go mod init ziyi.db.com
ZiyiDB/
├── cmd/
│   └── main.go           # 主程序入口
├── internal/
│   ├── ast/
│   │   └── ast.go        # 抽象語法樹定義
│   ├── lexer/
│   │   ├── lexer.go      # 詞法分析器實現(xiàn)
│   │   └── token.go      # 詞法單元定義
│   ├── parser/
│   │   └── parser.go     # 語法分析器實現(xiàn)
│   └── storage/
│       └── memory.go     # 內(nèi)存存儲引擎實現(xiàn)
├── go.mod                # Go模塊定義
└── go.sum                # 依賴版本鎖定

原理介紹

流程圖:

在這里插入圖片描述

主要包含幾大模塊:

  • cmd/main.go:
  • 程序入口點
  • 實現(xiàn)交互式命令行界面
  • 處理用戶輸入
  • 顯示執(zhí)行結(jié)果
  • internal/ast/ast.go:
  • 定義抽象語法樹節(jié)點
  • 定義 SQL 語句結(jié)構(gòu)
  • 定義表達(dá)式結(jié)構(gòu)
  • internal/lexer/token.go:
  • 定義詞法單元類型
  • 定義 SQL 關(guān)鍵字
  • 定義運算符和分隔符
  • internal/lexer/lexer.go:
  • 實現(xiàn)詞法分析器
  • 將輸入文本轉(zhuǎn)換為標(biāo)記序列
  • 處理標(biāo)識符和字面量
  • internal/parser/parser.go:
  • 實現(xiàn)語法分析器
  • 將標(biāo)記序列轉(zhuǎn)換為抽象語法樹
  • 處理各種 SQL 語句
  • internal/storage/memory.go:
  • 實現(xiàn)內(nèi)存存儲引擎
  • 處理數(shù)據(jù)存儲和檢索
  • 實現(xiàn)索引和約束

具體實現(xiàn)

模塊一: 詞法分析器 (Lexer)

詞法分析器 (Lexer):SQL轉(zhuǎn)token序列

①定義標(biāo)記類型token.go

思路

新建ziyi-db/internal/lexer/token.go文件,完成詞法分析器(Lexer)中的標(biāo)記(Token)定義部分,用于將 SQL 語句分解成基本的語法單元。

定義詞法單元以及關(guān)鍵字:

  • 包含常見的SQL關(guān)鍵字,如:select、update等
  • 包含符號關(guān)鍵字:=、>、<
  • 包含字段類型:INT、字符型(TEXT)
  • 包含標(biāo)識符:INDENT,解析出來的SQL列名、表名
type TokenType string

const (
    SELECT  TokenType = "SELECT"
    FROM    TokenType = "FROM"
    IDENT   TokenType = "IDENT"  // 標(biāo)識符(如列名、表名)
    INT_LIT TokenType = "INT"    // 整數(shù)字面量
    STRING  TokenType = "STRING" // 字符串字面量
    EQ TokenType = "=" // 等于
    GT TokenType = ">" // 大于
    LT TokenType = "<" // 小于
    ....
)

// Token 詞法單元
// Type:標(biāo)記的類型(如 SELECT、IDENT 等)
// Literal:標(biāo)記的實際值(如具體的列名、數(shù)字等)
type Token struct {
    Type    TokenType // 標(biāo)記類型
    Literal string    // 標(biāo)記的實際值
}

示例:

SELECT id, name FROM users WHERE age > 18;

該SQL 語句會被下面的詞法分析器lexer.go分解成以下標(biāo)記序列:
{Type: SELECT, Literal: "SELECT"}
{Type: IDENT, Literal: "id"}
{Type: COMMA, Literal: ","}
{Type: IDENT, Literal: "name"}
{Type: FROM, Literal: "FROM"}
{Type: IDENT, Literal: "users"}
{Type: WHERE, Literal: "WHERE"}
{Type: IDENT, Literal: "age"}
{Type: GT, Literal: ">"}
{Type: INT_LIT, Literal: "18"}
{Type: SEMI, Literal: ";"}
解析后的標(biāo)記隨后會被傳遞給語法分析器(Parser)進行進一步處理,構(gòu)建抽象語法樹(AST)。

全部代碼

// internal/lexer/token.go
package lexer

// TokenType 表示詞法單元類型
type TokenType string

const (
    // 特殊標(biāo)記
    EOF   TokenType = "EOF"   // 文件結(jié)束標(biāo)記
    ERROR TokenType = "ERROR" // 錯誤標(biāo)記

    // 關(guān)鍵字
    SELECT  TokenType = "SELECT"
    FROM    TokenType = "FROM"
    WHERE   TokenType = "WHERE"
    CREATE  TokenType = "CREATE"
    TABLE   TokenType = "TABLE"
    INSERT  TokenType = "INSERT"
    INTO    TokenType = "INTO"
    VALUES  TokenType = "VALUES"
    UPDATE  TokenType = "UPDATE"
    SET     TokenType = "SET"
    DELETE  TokenType = "DELETE"
    DROP    TokenType = "DROP"
    PRIMARY TokenType = "PRIMARY"
    KEY     TokenType = "KEY"
    INT     TokenType = "INT"
    TEXT    TokenType = "TEXT"
    LIKE    TokenType = "LIKE"

    // 標(biāo)識符和字面量
    IDENT   TokenType = "IDENT"  // 標(biāo)識符(如列名、表名)
    INT_LIT TokenType = "INT"    // 整數(shù)字面量
    STRING  TokenType = "STRING" // 字符串字面量

    // 運算符
    EQ TokenType = "="
    GT TokenType = ">"
    LT TokenType = "<"

    // 標(biāo)識符
    COMMA    TokenType = ","
    SEMI     TokenType = ";"
    LPAREN   TokenType = "("
    RPAREN   TokenType = ")"
    ASTERISK TokenType = "*"
)

// Token 詞法單元
// Type:標(biāo)記的類型(如 SELECT、IDENT 等)
// Literal:標(biāo)記的實際值(如具體的列名、數(shù)字等)
type Token struct {
    Type    TokenType // 標(biāo)記類型
    Literal string    // 標(biāo)記的實際值
}

② 實現(xiàn)詞法分析器lexer.go

思路

新建ziyi-db/internal/lexer/lexer.go文件,這是詞法分析器(Lexer)的核心實現(xiàn),負(fù)責(zé)將輸入的 SQL 語句分解成標(biāo)記(Token)序列。

詞法分析器lexer.go:讀取SQL到內(nèi)存中并進行解析,將字符轉(zhuǎn)換為對應(yīng)關(guān)鍵字

示例:

SELECT id, name FROM users WHERE age > 18;

處理過程:
跳過空白字符
讀取 "SELECT" 并識別為關(guān)鍵字
讀取 "id" 并識別為標(biāo)識符
讀取 "," 并識別為分隔符
讀取 "name" 并識別為標(biāo)識符
讀取 "FROM" 并識別為關(guān)鍵字
讀取 "users" 并識別為標(biāo)識符
讀取 "WHERE" 并識別為關(guān)鍵字
讀取 "age" 并識別為標(biāo)識符
讀取 ">" 并識別為運算符
讀取 "18" 并識別為數(shù)字
讀取 ";" 并識別為分隔符
這個詞法分析器是 SQL 解析器的第一步,它將輸入的 SQL 語句分解成標(biāo)記序列,為后續(xù)的語法分析提供基礎(chǔ)



該SQL 語句會被詞法分析器分解成以下標(biāo)記序列:
{Type: SELECT, Literal: "SELECT"}
{Type: IDENT, Literal: "id"}
{Type: COMMA, Literal: ","}
{Type: IDENT, Literal: "name"}
{Type: FROM, Literal: "FROM"}
{Type: IDENT, Literal: "users"}
{Type: WHERE, Literal: "WHERE"}
{Type: IDENT, Literal: "age"}
{Type: GT, Literal: ">"}
{Type: INT_LIT, Literal: "18"}
{Type: SEMI, Literal: ";"}
解析后的標(biāo)記隨后會被傳遞給語法分析器(Parser)進行進一步處理,構(gòu)建抽象語法樹(AST)。

全部代碼

// internal/lexer/lexer.go
package lexer

import (
    "bufio"
    "bytes"
    "io"
    "strings"
    "unicode"
)

// Lexer 詞法分析器
// reader:使用 bufio.Reader 進行高效的字符讀取
// ch:存儲當(dāng)前正在處理的字符
type Lexer struct {
    reader *bufio.Reader // 用于讀取輸入
    ch     rune          // 當(dāng)前字符
}

// NewLexer 創(chuàng)建一個新的 詞法分析器
// 初始化 reader 并讀取第一個字符
func NewLexer(r io.Reader) *Lexer {
    l := &Lexer{
       reader: bufio.NewReader(r),
    }
    l.readChar()
    return l
}

// 讀取字符
func (l *Lexer) readChar() {
    ch, _, err := l.reader.ReadRune()
    if err != nil {
       l.ch = 0 // 遇到錯誤或EOF時設(shè)置為0
    } else {
       l.ch = ch
    }
}

// NextToken 獲取下一個詞法單元
// 識別并返回下一個標(biāo)記
// 處理各種類型的標(biāo)記:運算符、分隔符、標(biāo)識符、數(shù)字、字符串等
func (l *Lexer) NextToken() Token {
    var tok Token
    // 跳過空白字符
    l.skipWhitespace()

    switch l.ch {
    case '=':
       tok = Token{Type: EQ, Literal: "="}
    case '>':
       tok = Token{Type: GT, Literal: ">"}
    case '<':
       tok = Token{Type: LT, Literal: "<"}
    case ',':
       tok = Token{Type: COMMA, Literal: ","}
    case ';':
       tok = Token{Type: SEMI, Literal: ";"}
    case '(':
       tok = Token{Type: LPAREN, Literal: "("}
    case ')':
       tok = Token{Type: RPAREN, Literal: ")"}
    case '*':
       tok = Token{Type: ASTERISK, Literal: "*"}
    case '\'':
       tok.Type = STRING
       // 讀取字符串字面量
       tok.Literal = l.readString()
       return tok
    case 0:
       tok = Token{Type: EOF, Literal: ""}
    default:
       if isLetter(l.ch) {
          // 讀取標(biāo)識符(表名、列名等)
          tok.Literal = l.readIdentifier()
          // 將讀取到的標(biāo)識符轉(zhuǎn)換為對應(yīng)的標(biāo)記類型(轉(zhuǎn)換為對應(yīng)tokenType)
          tok.Type = l.lookupIdentifier(tok.Literal)
          return tok
       } else if isDigit(l.ch) {
          tok.Type = INT_LIT
          // 讀取數(shù)字
          tok.Literal = l.readNumber()
          return tok
       } else {
          tok = Token{Type: ERROR, Literal: string(l.ch)}
       }
    }

    l.readChar()
    return tok
}

func (l *Lexer) skipWhitespace() {
    for unicode.IsSpace(l.ch) {
       l.readChar()
    }
}

// 讀取標(biāo)識符,如:列名、表名
func (l *Lexer) readIdentifier() string {
    var ident bytes.Buffer
    for isLetter(l.ch) || isDigit(l.ch) {
       ident.WriteRune(l.ch)
       l.readChar()
    }
    return ident.String()
}

func (l *Lexer) readNumber() string {
    var num bytes.Buffer
    for isDigit(l.ch) {
       num.WriteRune(l.ch)
       l.readChar()
    }
    return num.String()
}

// 讀取字符串字面量
func (l *Lexer) readString() string {
    var str bytes.Buffer
    l.readChar() // 跳過開始的引號
    for l.ch != '\'' && l.ch != 0 {
       str.WriteRune(l.ch)
       l.readChar()
    }
    l.readChar() // 跳過結(jié)束的引號
    return str.String()
}

func (l *Lexer) peekChar() rune {
    ch, _, err := l.reader.ReadRune()
    if err != nil {
       return 0
    }
    l.reader.UnreadRune()
    return ch
}

// lookupIdentifier 查找標(biāo)識符類型
// 將標(biāo)識符轉(zhuǎn)換為對應(yīng)的標(biāo)記類型
// 識別 SQL 關(guān)鍵字
func (l *Lexer) lookupIdentifier(ident string) TokenType {
    switch strings.ToUpper(ident) {
    case "SELECT":
       return SELECT
    case "FROM":
       return FROM
    case "WHERE":
       return WHERE
    case "CREATE":
       return CREATE
    case "TABLE":
       return TABLE
    case "INSERT":
       return INSERT
    case "INTO":
       return INTO
    case "VALUES":
       return VALUES
    case "UPDATE":
       return UPDATE
    case "SET":
       return SET
    case "DELETE":
       return DELETE
    case "DROP":
       return DROP
    case "PRIMARY":
       return PRIMARY
    case "KEY":
       return KEY
    case "INT":
       return INT
    case "TEXT":
       return TEXT
    case "LIKE":
       return LIKE
    default:
       return IDENT
    }
}

// 判斷字符是否為字母或下劃線
func isLetter(ch rune) bool {
    return unicode.IsLetter(ch) || ch == '_'
}

// 判斷字符是否為數(shù)字
func isDigit(ch rune) bool {
    return unicode.IsDigit(ch)
}

模塊二:抽象語法樹 (AST)

思路

抽象語法樹用于表示 SQL 語句的語法結(jié)構(gòu)。我們需要為每種 SQL 語句定義相應(yīng)的節(jié)點類型。

我們新建internal/ast/ast.go。

ast.go構(gòu)建不同SQL語句的結(jié)構(gòu),以及查詢結(jié)果等。
這個 AST 定義文件是 SQL 解析器的核心部分,它:

  • 定義了所有 SQL 語句的語法結(jié)構(gòu)
  • 提供了類型安全的方式來表示 SQL 語句
  • 支持復(fù)雜的表達(dá)式和條件
  • 便于后續(xù)的語義分析和執(zhí)行
    通過這個 AST,我們可以:
  • 驗證 SQL 語句的語法正確性
  • 進行語義分析
  • 生成執(zhí)行計劃
  • 執(zhí)行 SQL 語句

示例:

SELECT id, name FROM users WHERE age > 18;

交給語法分析器parser解析后的AST結(jié)構(gòu)為:

SelectStatement
├── Fields
│   ├── Identifier{Value: "id"}
│   └── Identifier{Value: "name"}
├── TableName: "users"
└── Where
    └── BinaryExpression
        ├── Left: Identifier{Value: "age"}
        ├── Operator: ">"
        └── Right: IntegerLiteral{Value: "18"}

全部代碼

package ast

import (
    "cursor-db/internal/lexer"
    "fmt"
)

// Node 表示AST中的節(jié)點
type Node interface {
    TokenLiteral() string
}

// Statement 表示SQL語句
type Statement interface {
    Node
    statementNode()
}

// Expression 表示表達(dá)式
type Expression interface {
    Node
    expressionNode()
}

// Program 表示整個SQL程序
type Program struct {
    Statements []Statement
}

// SelectStatement 表示SELECT語句
type SelectStatement struct {
    Token     lexer.Token
    Fields    []Expression
    TableName string
    Where     Expression
}

func (ss *SelectStatement) statementNode()       {}
func (ss *SelectStatement) TokenLiteral() string { return ss.Token.Literal }

// CreateTableStatement 表示CREATE TABLE語句
type CreateTableStatement struct {
    Token     lexer.Token
    TableName string
    Columns   []ColumnDefinition
}

func (cts *CreateTableStatement) statementNode()       {}
func (cts *CreateTableStatement) TokenLiteral() string { return cts.Token.Literal }

// InsertStatement 表示INSERT語句
type InsertStatement struct {
    Token     lexer.Token
    TableName string
    Values    []Expression
}

func (is *InsertStatement) statementNode()       {}
func (is *InsertStatement) TokenLiteral() string { return is.Token.Literal }

// ColumnDefinition 表示列定義
type ColumnDefinition struct {
    Name     string
    Type     string
    Primary  bool
    Nullable bool
}

// Cell 表示數(shù)據(jù)單元格
type Cell struct {
    Type      CellType
    IntValue  int32
    TextValue string
}

// CellType 表示單元格類型
type CellType int

const (
    CellTypeInt CellType = iota
    CellTypeText
)

// AsText 返回單元格的文本值
func (c *Cell) AsText() string {
    switch c.Type {
    case CellTypeInt:
       s := fmt.Sprintf("%d", c.IntValue)
       return s
    case CellTypeText:
       return c.TextValue
    default:
       return "NULL"
    }
}

// AsInt 返回單元格的整數(shù)值
func (c *Cell) AsInt() int32 {
    if c.Type == CellTypeInt {
       return c.IntValue
    }
    return 0
}

// String 返回單元格的字符串表示
func (c Cell) String() string {
    switch c.Type {
    case CellTypeInt:
       return fmt.Sprintf("%d", c.IntValue)
    case CellTypeText:
       return c.TextValue
    default:
       return "NULL"
    }
}

// Results 表示查詢結(jié)果
type Results struct {
    Columns []ResultColumn
    Rows    [][]Cell
}

// ResultColumn 表示結(jié)果列
type ResultColumn struct {
    Name string
    Type string
}

// StarExpression 表示星號表達(dá)式,如:select * from users;
type StarExpression struct{}

func (se *StarExpression) expressionNode()      {}
func (se *StarExpression) TokenLiteral() string { return "*" }

// LikeExpression 表示LIKE表達(dá)式, 如 LIKE '%b'
type LikeExpression struct {
    Token   lexer.Token
    Left    Expression
    Pattern string
}

func (le *LikeExpression) expressionNode()      {}
func (le *LikeExpression) TokenLiteral() string { return le.Token.Literal }

// BinaryExpression 表示二元表達(dá)式,如比較運算,大于小于比較等
type BinaryExpression struct {
    Token    lexer.Token
    Left     Expression
    Operator string
    Right    Expression
}

func (be *BinaryExpression) expressionNode()      {}
func (be *BinaryExpression) TokenLiteral() string { return be.Token.Literal }

// IntegerLiteral 表示整數(shù)字面量
type IntegerLiteral struct {
    Token lexer.Token
    Value string
}

func (il *IntegerLiteral) expressionNode()      {}
func (il *IntegerLiteral) TokenLiteral() string { return il.Token.Literal }

// StringLiteral 表示字符串字面量
type StringLiteral struct {
    Token lexer.Token
    Value string
}

func (sl *StringLiteral) expressionNode()      {}
func (sl *StringLiteral) TokenLiteral() string { return sl.Token.Literal }

// Identifier 表示標(biāo)識符(如列名)
type Identifier struct {
    Token lexer.Token
    Value string
}

func (i *Identifier) expressionNode()      {}
func (i *Identifier) TokenLiteral() string { return i.Token.Literal }

// UpdateStatement 表示UPDATE語句
type UpdateStatement struct {
    Token     lexer.Token
    TableName string
    Set       []SetClause
    Where     Expression
}

func (us *UpdateStatement) statementNode()       {}
func (us *UpdateStatement) TokenLiteral() string { return us.Token.Literal }

// SetClause 表示SET子句
type SetClause struct {
    Column string
    Value  Expression
}

// DeleteStatement 表示DELETE語句
type DeleteStatement struct {
    Token     lexer.Token
    TableName string
    Where     Expression
}

func (ds *DeleteStatement) statementNode()       {}
func (ds *DeleteStatement) TokenLiteral() string { return ds.Token.Literal }

// DropTableStatement 表示DROP TABLE語句
type DropTableStatement struct {
    Token     lexer.Token
    TableName string
}

func (ds *DropTableStatement) statementNode()       {}
func (ds *DropTableStatement) TokenLiteral() string { return ds.Token.Literal }

模塊三:語法分析器 (Parser)

思路

語法分析器負(fù)責(zé)將詞法分析器生成的標(biāo)記序列轉(zhuǎn)換為抽象語法樹。將token序列構(gòu)建成ast。

SQL 解析器(Parser)的實現(xiàn),負(fù)責(zé)將詞法分析器(Lexer)產(chǎn)生的標(biāo)記(Token)序列轉(zhuǎn)換為抽象語法樹(AST)。
語法分析器SQL 數(shù)據(jù)庫系統(tǒng)的關(guān)鍵組件,負(fù)責(zé):

  • 驗證 SQL 語句的語法正確性
  • 構(gòu)建抽象語法樹
  • 為后續(xù)的語義分析和執(zhí)行提供基礎(chǔ)

我們新建internal/parser/parser.go。

示例:

CREATE TABLE users (
    id INT PRIMARY KEY,
    name TEXT
);


解析過程:
1. 識別 CREATE 關(guān)鍵字
2. 解析 TABLE 關(guān)鍵字
3. 解析表名 "users"
4. 解析列定義:
    列名 "id",類型 INT,主鍵
    列名 "name",類型 TEXT
5. 生成 CREATE TABLE 語句的 AST

全部代碼

package parser

import (
    "fmt"
    "ziyi.db.com/internal/ast"
    "ziyi.db.com/internal/lexer"
)

// Parser 表示語法分析器
// 維護當(dāng)前和下一個標(biāo)記,實現(xiàn)向前查看(lookahead)
// 記錄解析過程中的錯誤
type Parser struct {
    l         *lexer.Lexer // 詞法分析器
    curToken  lexer.Token  // 當(dāng)前標(biāo)記
    peekToken lexer.Token  // 下一個標(biāo)記
    errors    []string     // 錯誤信息
}

// NewParser 創(chuàng)建新的語法分析器
// 初始化解析器
// 預(yù)讀兩個標(biāo)記
func NewParser(l *lexer.Lexer) *Parser {
    p := &Parser{
       l:      l,
       errors: []string{},
    }

    // 讀取兩個token,設(shè)置curToken和peekToken
    p.nextToken()
    p.nextToken()

    return p
}

// nextToken 移動到下一個詞法單元
func (p *Parser) nextToken() {
    p.curToken = p.peekToken
    p.peekToken = p.l.NextToken()
}

// ParseProgram 解析整個程序
// 解析整個 SQL 程序
// 循環(huán)解析每個語句直到結(jié)束
func (p *Parser) ParseProgram() (*ast.Program, error) {
    program := &ast.Program{
       Statements: []ast.Statement{},
    }

    for p.curToken.Type != lexer.EOF {
       stmt, err := p.parseStatement()
       if err != nil {
          return nil, err
       }
       if stmt != nil {
          program.Statements = append(program.Statements, stmt)
       }
       p.nextToken()
    }

    return program, nil
}

// parseStatement 解析語句
// 根據(jù)當(dāng)前標(biāo)記類型選擇相應(yīng)的解析方法
func (p *Parser) parseStatement() (ast.Statement, error) {
    switch p.curToken.Type {
    case lexer.CREATE:
       return p.parseCreateTableStatement()
    case lexer.INSERT:
       return p.parseInsertStatement()
    case lexer.SELECT:
       return p.parseSelectStatement()
    case lexer.UPDATE:
       return p.parseUpdateStatement()
    case lexer.DELETE:
       return p.parseDeleteStatement()
    case lexer.DROP:
       return p.parseDropTableStatement()
    case lexer.SEMI:
       return nil, nil
    default:
       return nil, fmt.Errorf("You have an error in your SQL syntax; check the manual that corresponds to your db server version for the right syntax to use near '%s'", p.curToken.Type)
    }
}

// parseCreateTableStatement 解析CREATE TABLE語句
// 解析表名
// 解析列定義
// 處理主鍵約束
func (p *Parser) parseCreateTableStatement() (*ast.CreateTableStatement, error) {
    stmt := &ast.CreateTableStatement{Token: p.curToken}

    if !p.expectPeek(lexer.TABLE) {
       return nil, fmt.Errorf("You have an error in your SQL syntax; check the manual that corresponds to your db server version for the right syntax to use near '%s'", p.peekToken.Literal)
    }

    if !p.expectPeek(lexer.IDENT) {
       return nil, fmt.Errorf("You have an error in your SQL syntax; check the manual that corresponds to your db server version for the right syntax to use near '%s'", p.peekToken.Literal)
    }
    stmt.TableName = p.curToken.Literal

    if !p.expectPeek(lexer.LPAREN) {
       return nil, fmt.Errorf("You have an error in your SQL syntax; check the manual that corresponds to your db server version for the right syntax to use near '%s'", p.peekToken.Literal)
    }

    // 解析列定義
    for !p.peekTokenIs(lexer.RPAREN) {
       p.nextToken()

       if !p.curTokenIs(lexer.IDENT) {
          return nil, fmt.Errorf("You have an error in your SQL syntax; check the manual that corresponds to your db server version for the right syntax to use near '%s'", p.curToken.Literal)
       }

       col := ast.ColumnDefinition{
          Name: p.curToken.Literal,
       }

       if !p.expectPeek(lexer.INT) && !p.expectPeek(lexer.TEXT) {
          return nil, fmt.Errorf("You have an error in your SQL syntax; check the manual that corresponds to your db server version for the right syntax to use near '%s'", p.peekToken.Literal)
       }

       col.Type = string(p.curToken.Type)

       if p.peekTokenIs(lexer.PRIMARY) {
          p.nextToken()
          if !p.expectPeek(lexer.KEY) {
             return nil, fmt.Errorf("You have an error in your SQL syntax; check the manual that corresponds to your db server version for the right syntax to use near '%s'", p.peekToken.Literal)
          }
          col.Primary = true
       }

       stmt.Columns = append(stmt.Columns, col)

       if p.peekTokenIs(lexer.COMMA) {
          p.nextToken()
       }
    }

    if !p.expectPeek(lexer.RPAREN) {
       return nil, fmt.Errorf("You have an error in your SQL syntax; check the manual that corresponds to your db server version for the right syntax to use near '%s'", p.peekToken.Literal)
    }

    return stmt, nil
}

// parseInsertStatement 解析INSERT語句
// 解析表名
// 解析 VALUES 子句
// 解析插入的值
func (p *Parser) parseInsertStatement() (*ast.InsertStatement, error) {
    stmt := &ast.InsertStatement{Token: p.curToken}

    if !p.expectPeek(lexer.INTO) {
       return nil, fmt.Errorf("You have an error in your SQL syntax; check the manual that corresponds to your db server version for the right syntax to use near '%s'", p.peekToken.Literal)
    }

    if !p.expectPeek(lexer.IDENT) {
       return nil, fmt.Errorf("You have an error in your SQL syntax; check the manual that corresponds to your db server version for the right syntax to use near '%s'", p.peekToken.Literal)
    }
    stmt.TableName = p.curToken.Literal

    if !p.expectPeek(lexer.VALUES) {
       return nil, fmt.Errorf("You have an error in your SQL syntax; check the manual that corresponds to your db server version for the right syntax to use near '%s'", p.peekToken.Literal)
    }

    if !p.expectPeek(lexer.LPAREN) {
       return nil, fmt.Errorf("You have an error in your SQL syntax; check the manual that corresponds to your db server version for the right syntax to use near '%s'", p.peekToken.Literal)
    }

    // 解析值列表
    for !p.peekTokenIs(lexer.RPAREN) {
       p.nextToken()

       expr, err := p.parseExpression()
       if err != nil {
          return nil, err
       }

       stmt.Values = append(stmt.Values, expr)

       if p.peekTokenIs(lexer.COMMA) {
          p.nextToken()
       }
    }

    if !p.expectPeek(lexer.RPAREN) {
       return nil, fmt.Errorf("You have an error in your SQL syntax; check the manual that corresponds to your db server version for the right syntax to use near '%s'", p.peekToken.Literal)
    }

    return stmt, nil
}

// parseSelectStatement 解析SELECT語句
// 解析選擇列表
// 解析 FROM 子句
// 解析 WHERE 子句
func (p *Parser) parseSelectStatement() (*ast.SelectStatement, error) {
    stmt := &ast.SelectStatement{Token: p.curToken}

    // 解析選擇列表
    for !p.peekTokenIs(lexer.FROM) {
       p.nextToken()

       if p.curToken.Type == lexer.ASTERISK {
          stmt.Fields = append(stmt.Fields, &ast.StarExpression{})
          break
       }

       expr, err := p.parseExpression()
       if err != nil {
          return nil, err
       }

       stmt.Fields = append(stmt.Fields, expr)

       if p.peekTokenIs(lexer.COMMA) {
          p.nextToken()
       }
    }

    if !p.expectPeek(lexer.FROM) {
       return nil, fmt.Errorf("You have an error in your SQL syntax; check the manual that corresponds to your db server version for the right syntax to use near '%s'", p.peekToken.Literal)
    }

    if !p.expectPeek(lexer.IDENT) {
       return nil, fmt.Errorf("You have an error in your SQL syntax; check the manual that corresponds to your db server version for the right syntax to use near '%s'", p.peekToken.Literal)
    }
    stmt.TableName = p.curToken.Literal

    // 解析WHERE子句
    if p.peekTokenIs(lexer.WHERE) {
       p.nextToken()
       p.nextToken()

       // 解析左操作數(shù)(列名)
       if !p.curTokenIs(lexer.IDENT) {
          return nil, fmt.Errorf("You have an error in your SQL syntax; check the manual that corresponds to your db server version for the right syntax to use near '%s'", p.curToken.Literal)
       }
       left := &ast.Identifier{
          Token: p.curToken,
          Value: p.curToken.Literal,
       }

       // 解析操作符
       p.nextToken()
       operator := p.curToken

       // 處理LIKE操作符
       if p.curTokenIs(lexer.LIKE) {
          p.nextToken()
          if !p.curTokenIs(lexer.STRING) {
             return nil, fmt.Errorf("You have an error in your SQL syntax; check the manual that corresponds to your db server version for the right syntax to use near '%s'", p.curToken.Literal)
          }
          // 移除字符串字面量的引號
          pattern := p.curToken.Literal
          if len(pattern) >= 2 && (pattern[0] == '\'' || pattern[0] == '"') {
             pattern = pattern[1 : len(pattern)-1]
          }
          stmt.Where = &ast.LikeExpression{
             Token:   operator,
             Left:    left,
             Pattern: pattern,
          }
          return stmt, nil
       }

       // 處理其他操作符
       if !p.curTokenIs(lexer.EQ) && !p.curTokenIs(lexer.GT) && !p.curTokenIs(lexer.LT) {
          return nil, fmt.Errorf("You have an error in your SQL syntax; check the manual that corresponds to your db server version for the right syntax to use near '%s'", operator.Type)
       }

       // 解析右操作數(shù)
       p.nextToken()
       right, err := p.parseExpression()
       if err != nil {
          return nil, err
       }

       stmt.Where = &ast.BinaryExpression{
          Token:    operator,
          Left:     left,
          Operator: operator.Literal,
          Right:    right,
       }
    }

    return stmt, nil
}

// parseUpdateStatement 解析UPDATE語句
// 解析表名
// 解析 SET 子句
// 解析 WHERE 子句
func (p *Parser) parseUpdateStatement() (*ast.UpdateStatement, error) {
    stmt := &ast.UpdateStatement{Token: p.curToken}

    if !p.expectPeek(lexer.IDENT) {
       return nil, fmt.Errorf("You have an error in your SQL syntax; check the manual that corresponds to your db server version for the right syntax to use near '%s'", p.peekToken.Literal)
    }
    stmt.TableName = p.curToken.Literal

    if !p.expectPeek(lexer.SET) {
       return nil, fmt.Errorf("You have an error in your SQL syntax; check the manual that corresponds to your db server version for the right syntax to use near '%s'", p.peekToken.Literal)
    }

    // 解析SET子句
    for {
       p.nextToken()
       if !p.curTokenIs(lexer.IDENT) {
          return nil, fmt.Errorf("You have an error in your SQL syntax; check the manual that corresponds to your db server version for the right syntax to use near '%s'", p.curToken.Literal)
       }
       column := p.curToken.Literal

       if !p.expectPeek(lexer.EQ) {
          return nil, fmt.Errorf("You have an error in your SQL syntax; check the manual that corresponds to your db server version for the right syntax to use near '%s'", p.peekToken.Literal)
       }

       p.nextToken()
       value, err := p.parseExpression()
       if err != nil {
          return nil, err
       }

       stmt.Set = append(stmt.Set, ast.SetClause{
          Column: column,
          Value:  value,
       })

       if !p.peekTokenIs(lexer.COMMA) {
          break
       }
       p.nextToken()
    }

    // 解析WHERE子句
    if p.peekTokenIs(lexer.WHERE) {
       p.nextToken()
       p.nextToken()

       // 解析左操作數(shù)(列名)
       if !p.curTokenIs(lexer.IDENT) {
          return nil, fmt.Errorf("You have an error in your SQL syntax; check the manual that corresponds to your db server version for the right syntax to use near '%s'", p.curToken.Literal)
       }
       left := &ast.Identifier{
          Token: p.curToken,
          Value: p.curToken.Literal,
       }

       // 解析操作符
       p.nextToken()
       operator := p.curToken
       if !p.curTokenIs(lexer.EQ) && !p.curTokenIs(lexer.GT) && !p.curTokenIs(lexer.LT) {
          return nil, fmt.Errorf("You have an error in your SQL syntax; check the manual that corresponds to your db server version for the right syntax to use near '%s'", operator.Type)
       }

       // 解析右操作數(shù)
       p.nextToken()
       right, err := p.parseExpression()
       if err != nil {
          return nil, err
       }

       stmt.Where = &ast.BinaryExpression{
          Token:    operator,
          Left:     left,
          Operator: operator.Literal,
          Right:    right,
       }
    }

    return stmt, nil
}

// parseDeleteStatement 解析DELETE語句
// 解析表名
// 解析 WHERE 子句
func (p *Parser) parseDeleteStatement() (*ast.DeleteStatement, error) {
    stmt := &ast.DeleteStatement{Token: p.curToken}

    if !p.expectPeek(lexer.FROM) {
       return nil, fmt.Errorf("You have an error in your SQL syntax; check the manual that corresponds to your db server version for the right syntax to use near '%s'", p.peekToken.Literal)
    }

    if !p.expectPeek(lexer.IDENT) {
       return nil, fmt.Errorf("You have an error in your SQL syntax; check the manual that corresponds to your db server version for the right syntax to use near '%s'", p.peekToken.Literal)
    }
    stmt.TableName = p.curToken.Literal

    // 解析WHERE子句
    if p.peekTokenIs(lexer.WHERE) {
       p.nextToken()
       p.nextToken()

       // 解析左操作數(shù)(列名)
       if !p.curTokenIs(lexer.IDENT) {
          return nil, fmt.Errorf("You have an error in your SQL syntax; check the manual that corresponds to your db server version for the right syntax to use near '%s'", p.curToken.Literal)
       }
       left := &ast.Identifier{
          Token: p.curToken,
          Value: p.curToken.Literal,
       }

       // 解析操作符
       p.nextToken()
       operator := p.curToken
       if !p.curTokenIs(lexer.EQ) && !p.curTokenIs(lexer.GT) && !p.curTokenIs(lexer.LT) {
          return nil, fmt.Errorf("You have an error in your SQL syntax; check the manual that corresponds to your db server version for the right syntax to use near '%s'", operator.Type)
       }

       // 解析右操作數(shù)
       p.nextToken()
       right, err := p.parseExpression()
       if err != nil {
          return nil, err
       }

       stmt.Where = &ast.BinaryExpression{
          Token:    operator,
          Left:     left,
          Operator: operator.Literal,
          Right:    right,
       }
    }

    return stmt, nil
}

// parseDropTableStatement 解析DROP TABLE語句
func (p *Parser) parseDropTableStatement() (*ast.DropTableStatement, error) {
    stmt := &ast.DropTableStatement{Token: p.curToken}

    if !p.expectPeek(lexer.TABLE) {
       return nil, fmt.Errorf("You have an error in your SQL syntax; check the manual that corresponds to your db server version for the right syntax to use near '%s'", p.peekToken.Literal)
    }

    if !p.expectPeek(lexer.IDENT) {
       return nil, fmt.Errorf("You have an error in your SQL syntax; check the manual that corresponds to your db server version for the right syntax to use near '%s'", p.peekToken.Literal)
    }
    stmt.TableName = p.curToken.Literal

    return stmt, nil
}

// parseExpression 解析表達(dá)式(字面量int、string類型,標(biāo)識符列名、表名等)
// 解析各種類型的表達(dá)式
// 支持字面量、標(biāo)識符等
func (p *Parser) parseExpression() (ast.Expression, error) {
    switch p.curToken.Type {
    case lexer.INT_LIT:
       return &ast.IntegerLiteral{
          Token: p.curToken,
          Value: p.curToken.Literal,
       }, nil
    case lexer.STRING:
       return &ast.StringLiteral{
          Token: p.curToken,
          Value: p.curToken.Literal,
       }, nil
    case lexer.IDENT:
       return &ast.Identifier{
          Token: p.curToken,
          Value: p.curToken.Literal,
       }, nil
    default:
       return nil, fmt.Errorf("You have an error in your SQL syntax; check the manual that corresponds to your db server version for the right syntax to use near '%s'", p.curToken.Type)
    }
}

// curTokenIs 檢查當(dāng)前token是否為指定類型
func (p *Parser) curTokenIs(t lexer.TokenType) bool {
    return p.curToken.Type == t
}

// peekTokenIs 檢查下一個token是否為指定類型
func (p *Parser) peekTokenIs(t lexer.TokenType) bool {
    return p.peekToken.Type == t
}

// expectPeek 檢查下一個詞法單元是否為預(yù)期類型
func (p *Parser) expectPeek(t lexer.TokenType) bool {
    if p.peekTokenIs(t) {
       p.nextToken()
       return true
    }
    return false
}

// parseWhereClause 解析WHERE子句
func (p *Parser) parseWhereClause() (ast.Expression, error) {
    p.nextToken()

    // 解析左操作數(shù)(列名)
    if !p.curTokenIs(lexer.IDENT) {
       return nil, fmt.Errorf("You have an error in your SQL syntax; check the manual that corresponds to your db server version for the right syntax to use near '%s'", p.curToken.Literal)
    }
    left := &ast.Identifier{
       Token: p.curToken,
       Value: p.curToken.Literal,
    }

    // 解析操作符
    p.nextToken()
    operator := p.curToken

    // 處理LIKE操作符
    if p.curTokenIs(lexer.LIKE) {
       p.nextToken()
       if !p.curTokenIs(lexer.STRING) {
          return nil, fmt.Errorf("You have an error in your SQL syntax; check the manual that corresponds to your db server version for the right syntax to use near '%s'", p.curToken.Literal)
       }
       // 移除字符串字面量的引號
       pattern := p.curToken.Literal
       if len(pattern) >= 2 && (pattern[0] == '\'' || pattern[0] == '"') {
          pattern = pattern[1 : len(pattern)-1]
       }
       return &ast.LikeExpression{
          Token:   operator,
          Left:    left,
          Pattern: pattern,
       }, nil
    }

    // 處理其他操作符
    if !p.curTokenIs(lexer.EQ) && !p.curTokenIs(lexer.GT) && !p.curTokenIs(lexer.LT) {
       return nil, fmt.Errorf("You have an error in your SQL syntax; check the manual that corresponds to your db server version for the right syntax to use near '%s'", operator.Type)
    }

    // 解析右操作數(shù)
    p.nextToken()
    right, err := p.parseExpression()
    if err != nil {
       return nil, err
    }

    return &ast.BinaryExpression{
       Token:    operator,
       Left:     left,
       Operator: operator.Literal,
       Right:    right,
    }, nil
}

模塊四:存儲引擎 (Storage)

思路

存儲引擎負(fù)責(zé)實際的數(shù)據(jù)存儲和檢索操作,執(zhí)行引擎中的數(shù)據(jù)操作CURD。

我們需要新建internal/storage/memory.go文件。

這是內(nèi)存存儲引擎的實現(xiàn),負(fù)責(zé)處理 SQL 語句的實際執(zhí)行和數(shù)據(jù)存儲。

本期存儲引擎實現(xiàn)了:

  • 完整的數(shù)據(jù)操作(CRUD)
  • 主鍵約束
  • 索引支持
  • 類型檢查
  • 條件評估
  • 模式匹配

它是 SQL 數(shù)據(jù)庫系統(tǒng)的核心組件,負(fù)責(zé):

  • 數(shù)據(jù)存儲和管理
  • 查詢執(zhí)行
  • 數(shù)據(jù)完整性維護
  • 性能優(yōu)化(通過索引)

原理解析:

-- 創(chuàng)建表
CREATE TABLE users (
    id INT PRIMARY KEY,
    name TEXT
);

-- 插入數(shù)據(jù)
INSERT INTO users VALUES (1, 'Alice');

-- 查詢數(shù)據(jù)
SELECT * FROM users WHERE name LIKE 'A%';

-- 更新數(shù)據(jù)
UPDATE users SET name = 'Bob' WHERE id = 1;

-- 刪除數(shù)據(jù)
DELETE FROM users WHERE id = 1;


存儲引擎會根據(jù)解析后的語法分析器,創(chuàng)建出對應(yīng)的數(shù)據(jù)結(jié)構(gòu)(如:在內(nèi)存中),以及對外暴露對該數(shù)據(jù)的操作(CRUD)

全部代碼

// internal/storage/memory.go
package storage

import (
    "fmt"
    "regexp"
    "strconv"
    "strings"
    "ziyi.db.com/internal/ast"
)

// MemoryBackend 內(nèi)存存儲引擎,管理所有表
type MemoryBackend struct {
    tables map[string]*Table
}

// Table 數(shù)據(jù)表,包含列定義、數(shù)據(jù)行和索引
type Table struct {
    Name    string
    Columns []ast.ColumnDefinition
    Rows    [][]ast.Cell
    Indexes map[string]*Index // 值到行索引的映射
}

// Index 索引,用于加速查詢
type Index struct {
    Column string
    Values map[string][]int // 值到行索引的映射
}

// NewMemoryBackend 創(chuàng)建新的內(nèi)存存儲引擎
func NewMemoryBackend() *MemoryBackend {
    return &MemoryBackend{
       tables: make(map[string]*Table),
    }
}

// CreateTable 創(chuàng)建表
// 驗證表名唯一性
// 創(chuàng)建表結(jié)構(gòu)
// 為主鍵列創(chuàng)建索引
func (b *MemoryBackend) CreateTable(stmt *ast.CreateTableStatement) error {
    if _, exists := b.tables[stmt.TableName]; exists {
       return fmt.Errorf("Table '%s' already exists", stmt.TableName)
    }

    table := &Table{
       Name:    stmt.TableName,
       Columns: stmt.Columns,
       Rows:    make([][]ast.Cell, 0),
       Indexes: make(map[string]*Index),
    }

    // 為主鍵創(chuàng)建索引
    for _, col := range stmt.Columns {
       if col.Primary {
          table.Indexes[col.Name] = &Index{
             Column: col.Name,
             Values: make(map[string][]int),
          }
       }
    }

    b.tables[stmt.TableName] = table
    return nil
}

// Insert 插入數(shù)據(jù)
// 驗證表存在性
// 檢查數(shù)據(jù)完整性
// 處理主鍵約束
// 維護索引
func (b *MemoryBackend) Insert(stmt *ast.InsertStatement) error {
    table, exists := b.tables[stmt.TableName]
    if !exists {
       return fmt.Errorf("Table '%s' doesn't exist", stmt.TableName)
    }

    if len(stmt.Values) != len(table.Columns) {
       return fmt.Errorf("Column count doesn't match value count at row 1")
    }

    // 轉(zhuǎn)換值
    row := make([]ast.Cell, len(stmt.Values))
    for i, expr := range stmt.Values {
       value, err := evaluateExpression(expr)
       if err != nil {
          return err
       }

       switch v := value.(type) {
       case string:
          if table.Columns[i].Type == "INT" {
             // 嘗試將字符串轉(zhuǎn)換為整數(shù)
             intVal, err := strconv.ParseInt(v, 10, 32)
             if err != nil {
                return fmt.Errorf("Incorrect integer value: '%s' for column '%s'", v, table.Columns[i].Name)
             }
             row[i] = ast.Cell{Type: ast.CellTypeInt, IntValue: int32(intVal)}
          } else {
             row[i] = ast.Cell{Type: ast.CellTypeText, TextValue: v}
          }
       case int32:
          row[i] = ast.Cell{Type: ast.CellTypeInt, IntValue: v}
       default:
          return fmt.Errorf("Unsupported value type: %T for column '%s'", value, table.Columns[i].Name)
       }
    }

    // 檢查主鍵約束
    for i, col := range table.Columns {
       if col.Primary {
          key := row[i].String()
          if _, exists := table.Indexes[col.Name].Values[key]; exists {
             return fmt.Errorf("Duplicate entry '%s' for key '%s'", key, col.Name)
          }
       }
    }

    // 插入數(shù)據(jù)
    rowIndex := len(table.Rows)
    table.Rows = append(table.Rows, row)

    // 更新索引
    for i, col := range table.Columns {
       if col.Primary {
          key := row[i].String()
          table.Indexes[col.Name].Values[key] = append(table.Indexes[col.Name].Values[key], rowIndex)
       }
    }

    return nil
}

// Select 查詢數(shù)據(jù)
// 支持 SELECT * 和指定列
// 處理 WHERE 條件
// 返回查詢結(jié)果
func (b *MemoryBackend) Select(stmt *ast.SelectStatement) (*ast.Results, error) {
    table, exists := b.tables[stmt.TableName]
    if !exists {
       return nil, fmt.Errorf("Table '%s' doesn't exist", stmt.TableName)
    }

    results := &ast.Results{
       Columns: make([]ast.ResultColumn, 0),
       Rows:    make([][]ast.Cell, 0),
    }

    // 處理選擇列表
    if len(stmt.Fields) == 1 && stmt.Fields[0].(*ast.StarExpression) != nil {
       // SELECT *
       for _, col := range table.Columns {
          results.Columns = append(results.Columns, ast.ResultColumn{
             Name: col.Name,
             Type: col.Type,
          })
       }
    } else {
       // 處理指定的列
       for _, expr := range stmt.Fields {
          switch e := expr.(type) {
          case *ast.Identifier:
             // 查找列
             found := false
             for _, col := range table.Columns {
                if col.Name == e.Value {
                   results.Columns = append(results.Columns, ast.ResultColumn{
                      Name: col.Name,
                      Type: col.Type,
                   })
                   found = true
                   break
                }
             }
             if !found {
                return nil, fmt.Errorf("Unknown column '%s' in 'field list'", e.Value)
             }
          default:
             return nil, fmt.Errorf("Unsupported select expression type")
          }
       }
    }

    // 處理WHERE子句
    for _, row := range table.Rows {
       if stmt.Where != nil {
          match, err := evaluateWhereCondition(stmt.Where, row, table.Columns)
          if err != nil {
             return nil, err
          }
          if !match {
             continue
          }
       }

       // 構(gòu)建結(jié)果行
       resultRow := make([]ast.Cell, len(results.Columns))
       for j, col := range results.Columns {
          // 查找列在原始行中的位置
          for k, tableCol := range table.Columns {
             if tableCol.Name == col.Name {
                resultRow[j] = row[k]
                break
             }
          }
       }
       results.Rows = append(results.Rows, resultRow)
    }

    return results, nil
}

// Update 執(zhí)行UPDATE操作
// 驗證表和列存在性
// 處理 WHERE 條件
// 更新符合條件的行
func (mb *MemoryBackend) Update(stmt *ast.UpdateStatement) error {
    table, ok := mb.tables[stmt.TableName]
    if !ok {
       return fmt.Errorf("Table '%s' doesn't exist", stmt.TableName)
    }

    // 獲取列索引
    columnIndices := make(map[string]int)
    for i, col := range table.Columns {
       columnIndices[col.Name] = i
    }

    // 驗證所有要更新的列是否存在
    for _, set := range stmt.Set {
       if _, ok := columnIndices[set.Column]; !ok {
          return fmt.Errorf("Unknown column '%s' in 'field list'", set.Column)
       }
    }

    // 更新符合條件的行
    for i := range table.Rows {
       if stmt.Where != nil {
          // 評估WHERE條件
          result, err := evaluateWhereCondition(stmt.Where, table.Rows[i], table.Columns)
          if err != nil {
             return err
          }
          if !result {
             continue
          }
       }

       // 更新行
       for _, set := range stmt.Set {
          colIndex := columnIndices[set.Column]
          value, err := evaluateExpression(set.Value)
          if err != nil {
             return err
          }

          switch v := value.(type) {
          case int32:
             table.Rows[i][colIndex] = ast.Cell{Type: ast.CellTypeInt, IntValue: v}
          case string:
             table.Rows[i][colIndex] = ast.Cell{Type: ast.CellTypeText, TextValue: v}
          default:
             return fmt.Errorf("Unsupported value type: %T for column '%s'", value, set.Column)
          }
       }
    }

    return nil
}

// Delete 執(zhí)行DELETE操作
// 驗證表存在性
// 處理 WHERE 條件
// 刪除符合條件的行
func (mb *MemoryBackend) Delete(stmt *ast.DeleteStatement) error {
    table, ok := mb.tables[stmt.TableName]
    if !ok {
       return fmt.Errorf("Table '%s' doesn't exist", stmt.TableName)
    }

    // 找出要刪除的行
    rowsToDelete := make([]int, 0)
    for i := range table.Rows {
       if stmt.Where != nil {
          // 評估WHERE條件
          result, err := evaluateWhereCondition(stmt.Where, table.Rows[i], table.Columns)
          if err != nil {
             return err
          }
          if !result {
             continue
          }
       }
       rowsToDelete = append(rowsToDelete, i)
    }

    // 從后向前刪除行,以避免索引變化
    for i := len(rowsToDelete) - 1; i >= 0; i-- {
       rowIndex := rowsToDelete[i]
       table.Rows = append(table.Rows[:rowIndex], table.Rows[rowIndex+1:]...)
    }

    return nil
}

// DropTable 刪除表
// 驗證表是否存在
// 從存儲引擎中刪除表
func (mb *MemoryBackend) DropTable(stmt *ast.DropTableStatement) error {
    if _, exists := mb.tables[stmt.TableName]; !exists {
       return fmt.Errorf("Unknown table '%s'", stmt.TableName)
    }

    delete(mb.tables, stmt.TableName)
    return nil
}

// evaluateExpression 評估表達(dá)式的值
// 計算表達(dá)式的值
// 處理不同類型的數(shù)據(jù)
func evaluateExpression(expr ast.Expression) (interface{}, error) {
    switch e := expr.(type) {
    case *ast.IntegerLiteral:
       val, err := strconv.ParseInt(e.Value, 10, 32)
       if err != nil {
          return nil, fmt.Errorf("Incorrect integer value: '%s'", e.Value)
       }
       return int32(val), nil
    case *ast.StringLiteral:
       return e.Value, nil
    case *ast.Identifier:
       return nil, fmt.Errorf("Cannot evaluate identifier: '%s'", e.Value)
    default:
       return nil, fmt.Errorf("Unknown expression type: %T", expr)
    }
}

// matchLikePattern 檢查字符串是否匹配LIKE模式
func matchLikePattern(str, pattern string) bool {
    // 將SQL LIKE模式轉(zhuǎn)換為正則表達(dá)式
    regexPattern := "^"
    for i := 0; i < len(pattern); i++ {
       switch pattern[i] {
       case '%':
          regexPattern += ".*"
       case '_':
          regexPattern += "."
       case '\\':
          if i+1 < len(pattern) {
             regexPattern += "\\" + string(pattern[i+1])
             i++
          }
       default:
          // 轉(zhuǎn)義正則表達(dá)式特殊字符
          if strings.ContainsAny(string(pattern[i]), ".+*?^$()[]{}|") {
             regexPattern += "\\" + string(pattern[i])
          } else {
             regexPattern += string(pattern[i])
          }
       }
    }
    regexPattern += "$"

    // 編譯正則表達(dá)式
    re, err := regexp.Compile(regexPattern)
    if err != nil {
       return false
    }

    // 執(zhí)行匹配
    return re.MatchString(str)
}

// evaluateWhereCondition 評估WHERE條件
// 評估 WHERE 條件
// 支持比較運算符和 LIKE 操作符
func evaluateWhereCondition(expr ast.Expression, row []ast.Cell, columns []ast.ColumnDefinition) (bool, error) {
    switch e := expr.(type) {
    case *ast.BinaryExpression:
       // 獲取左操作數(shù)的值
       leftValue, err := getColumnValue(e.Left, row, columns)
       if err != nil {
          return false, err
       }

       // 獲取右操作數(shù)的值
       rightValue, err := getColumnValue(e.Right, row, columns)
       if err != nil {
          return false, err
       }

       // 根據(jù)操作符比較值
       switch e.Operator {
       case "=":
          return compareValues(leftValue, rightValue, "=")
       case ">":
          return compareValues(leftValue, rightValue, ">")
       case "<":
          return compareValues(leftValue, rightValue, "<")
       default:
          return false, fmt.Errorf("Unknown operator: '%s'", e.Operator)
       }
    case *ast.LikeExpression:
       // 獲取左操作數(shù)的值
       leftValue, err := getColumnValue(e.Left, row, columns)
       if err != nil {
          return false, err
       }

       // 確保左操作數(shù)是字符串類型
       strValue, ok := leftValue.(string)
       if !ok {
          return false, fmt.Errorf("LIKE operator requires string operand")
       }

       // 執(zhí)行LIKE匹配
       return matchLikePattern(strValue, e.Pattern), nil
    default:
       return false, fmt.Errorf("Unknown expression type: %T", expr)
    }
}

// compareValues 比較兩個值
func compareValues(left, right interface{}, operator string) (bool, error) {
    switch l := left.(type) {
    case int32:
       if r, ok := right.(int32); ok {
          switch operator {
          case "=":
             return l == r, nil
          case ">":
             return l > r, nil
          case "<":
             return l < r, nil
          }
       }
    case string:
       if r, ok := right.(string); ok {
          switch operator {
          case "=":
             return l == r, nil
          case ">":
             return l > r, nil
          case "<":
             return l < r, nil
          }
       }
    }
    return false, fmt.Errorf("Cannot compare values of different types: %T and %T", left, right)
}

// getColumnValue 獲取列的值
func getColumnValue(expr ast.Expression, row []ast.Cell, columns []ast.ColumnDefinition) (interface{}, error) {
    switch e := expr.(type) {
    case *ast.Identifier:
       // 查找列索引
       for i, col := range columns {
          if col.Name == e.Value {
             switch row[i].Type {
             case ast.CellTypeInt:
                return row[i].IntValue, nil
             case ast.CellTypeText:
                return row[i].TextValue, nil
             default:
                return nil, fmt.Errorf("Unknown cell type: %v", row[i].Type)
             }
          }
       }
       return nil, fmt.Errorf("Unknown column '%s' in 'where clause'", e.Value)
    case *ast.IntegerLiteral:
       val, err := strconv.ParseInt(e.Value, 10, 32)
       if err != nil {
          return nil, fmt.Errorf("Incorrect integer value: '%s'", e.Value)
       }
       return int32(val), nil
    case *ast.StringLiteral:
       return e.Value, nil
    default:
       return nil, fmt.Errorf("Unknown expression type: %T", expr)
    }
}

//后續(xù)拓展新的存儲引擎,如落地到文件...

模塊五:REPL 交互界面

思路

最后,我們需要實現(xiàn)一個交互式的命令行界面,讓用戶可以輸入 SQL 命令并查看結(jié)果。
這是 ZiyiDB 的主程序,實現(xiàn)了一個交互式的 SQL 命令行界面。

為了實現(xiàn)客戶端可以上下翻找之前執(zhí)行的命令以及cli客戶端的美觀,我們這里使用"github.com/c-bata/go-prompt"庫

// 安裝依賴
go get "github.com/c-bata/go-prompt"

我們需要新建cmd/main.go文件。

主要實現(xiàn):

  • 交互式命令行界面
  • SQL 命令解析和執(zhí)行
  • 命令歷史記錄
  • 查詢結(jié)果格式化
  • 錯誤處理和提示

全部代碼

package main

import (
    "fmt"
    "github.com/c-bata/go-prompt"
    "os"
    "strings"
    "ziyi.db.com/internal/ast"
    "ziyi.db.com/internal/lexer"
    "ziyi.db.com/internal/parser"
    "ziyi.db.com/internal/storage"
)

var history []string               // 存儲命令歷史
var backend *storage.MemoryBackend // 存儲引擎實例
var historyIndex int               // 當(dāng)前歷史記錄索引

// 處理用戶輸入的命令
func executor(t string) {
    t = strings.TrimSpace(t)
    if t == "" {
       return
    }

    // 添加到歷史記錄
    history = append(history, t)
    historyIndex = len(history) // 重置歷史記錄索引

    // 處理退出命令
    if strings.ToLower(t) == "exit" {
       fmt.Println("Bye!")
       os.Exit(0)
    }

    // 創(chuàng)建詞法分析器
    l := lexer.NewLexer(strings.NewReader(t))

    // 創(chuàng)建語法分析器
    p := parser.NewParser(l)

    // 解析SQL語句
    stmt, err := p.ParseProgram()
    if err != nil {
       fmt.Printf("Parse error: %v\n", err)
       return
    }

    // 執(zhí)行SQL語句
    for _, statement := range stmt.Statements {
       switch s := statement.(type) {
       case *ast.CreateTableStatement:
          if err := backend.CreateTable(s); err != nil {
             fmt.Printf("Error: %v\n", err)
          } else {
             fmt.Println("Table created successfully")
          }
       case *ast.InsertStatement:
          if err := backend.Insert(s); err != nil {
             fmt.Printf("Error: %v\n", err)
          } else {
             fmt.Println("1 row inserted")
          }
       case *ast.SelectStatement:
          results, err := backend.Select(s)
          if err != nil {
             fmt.Printf("Error: %v\n", err)
          } else {
             // 計算每列的最大寬度
             colWidths := make([]int, len(results.Columns))
             for i, col := range results.Columns {
                colWidths[i] = len(col.Name)
             }
             for _, row := range results.Rows {
                for i, cell := range row {
                   cellLen := len(cell.String())
                   if cellLen > colWidths[i] {
                      colWidths[i] = cellLen
                   }
                }
             }

             // 打印表頭
             fmt.Print("+")
             for _, width := range colWidths {
                fmt.Print(strings.Repeat("-", width+2))
                fmt.Print("+")
             }
             fmt.Println()

             // 打印列名
             fmt.Print("|")
             for i, col := range results.Columns {
                fmt.Printf(" %-*s |", colWidths[i], col.Name)
             }
             fmt.Println()

             // 打印分隔線
             fmt.Print("+")
             for _, width := range colWidths {
                fmt.Print(strings.Repeat("-", width+2))
                fmt.Print("+")
             }
             fmt.Println()

             // 打印數(shù)據(jù)行
             for _, row := range results.Rows {
                fmt.Print("|")
                for i, cell := range row {
                   fmt.Printf(" %-*s |", colWidths[i], cell.String())
                }
                fmt.Println()
             }

             // 打印底部邊框
             fmt.Print("+")
             for _, width := range colWidths {
                fmt.Print(strings.Repeat("-", width+2))
                fmt.Print("+")
             }
             fmt.Println()

             // 打印行數(shù)統(tǒng)計
             fmt.Printf("%d rows in set\n", len(results.Rows))
          }
       case *ast.UpdateStatement:
          if err := backend.Update(s); err != nil {
             fmt.Printf("Error: %v\n", err)
          } else {
             fmt.Println("Query OK, 1 row affected")
          }
       case *ast.DeleteStatement:
          if err := backend.Delete(s); err != nil {
             fmt.Printf("Error: %v\n", err)
          } else {
             fmt.Println("Query OK, 1 row affected")
          }
       case *ast.DropTableStatement:
          if err := backend.DropTable(s); err != nil {
             fmt.Printf("Error: %v\n", err)
          } else {
             fmt.Println("Table dropped successfully")
          }
       default:
          fmt.Printf("Unsupported statement type: %T\n", s)
       }
    }
}

// 提供命令補全功能
func completer(d prompt.Document) []prompt.Suggest {
    s := []prompt.Suggest{}
    return prompt.FilterHasPrefix(s, d.GetWordBeforeCursor(), true)
}

func main() {
    // 初始化存儲引擎
    backend = storage.NewMemoryBackend()
    historyIndex = 0

    fmt.Println("Welcome to ZiyiDB!")
    fmt.Println("Type your SQL commands (type 'exit' to quit)")

    p := prompt.New(
       executor,
       completer,
       prompt.OptionTitle("ZiyiDB: A Simple SQL Database"),
       prompt.OptionPrefix("ziyidb> "),
       prompt.OptionHistory(history),
       prompt.OptionLivePrefix(func() (string, bool) {
          return "ziyidb> ", true
       }),
       //實現(xiàn)方向鍵上下翻閱歷史命令
       // 上鍵綁定
       prompt.OptionAddKeyBind(prompt.KeyBind{
          Key: prompt.Up,
          Fn: func(buf *prompt.Buffer) {
             if historyIndex > 0 {
                historyIndex--
                buf.DeleteBeforeCursor(len(buf.Text()))
                buf.InsertText(history[historyIndex], false, true)
             }
          },
       }),
       // 下鍵綁定
       prompt.OptionAddKeyBind(prompt.KeyBind{
          Key: prompt.Down,
          Fn: func(buf *prompt.Buffer) {
             if historyIndex < len(history)-1 {
                historyIndex++
                buf.DeleteBeforeCursor(len(buf.Text()))
                buf.InsertText(history[historyIndex], false, true)
             } else if historyIndex == len(history)-1 {
                historyIndex++
                buf.DeleteBeforeCursor(len(buf.Text()))
             }
          },
       }),
    )
    p.Run()
}

整體測試

編寫完第一版后,現(xiàn)在我們來整體測試一下。

測試腳本

test_cast.sql:

-- 1. 創(chuàng)建表
CREATE TABLE users (id INT PRIMARY KEY,name TEXT ,age INT);


-- 2. 插入用戶數(shù)據(jù)
INSERT INTO users VALUES (1, 'Alice', 20);
INSERT INTO users VALUES (2, 'Bob', 25);
INSERT INTO users VALUES (3, 'Charlie', 30);
INSERT INTO users VALUES (4, 'David', 35);
INSERT INTO users VALUES (5, 'Eve', 40);

-- 3. 測試主鍵沖突
INSERT INTO users VALUES (1, 'Tomas', 21);


-- 4. 基本查詢測試
-- 4.1 查詢所有數(shù)據(jù)
SELECT * FROM users;


-- 4.2 查詢特定列
SELECT id, name FROM users;

-- 5. WHERE 子句測試
SELECT * FROM users WHERE age > 25;
SELECT * FROM users WHERE age < 30;

-- 6. LIKE 操作符測試
-- 6.1 基本模式匹配
SELECT * FROM users WHERE name LIKE 'A%';  -- 以 A 開頭
SELECT * FROM users WHERE name LIKE '%e';  -- 以 e 結(jié)尾

-- 6.2 轉(zhuǎn)義字符測試
INSERT INTO users VALUES (6, 'Bob%Smith', 45);
SELECT * FROM users WHERE name LIKE 'Bob\%Smith';

-- 7. 更新操作測試
-- 7.1 更新單個字段
UPDATE users SET age = 21 WHERE name = 'Alice';

-- 7.2 更新多個字段
UPDATE users SET name = 'Robert', age = 8 WHERE id = 2;


-- 8. 刪除操作測試
DELETE FROM users WHERE age > 30;

-- 9. 清理測試數(shù)據(jù)
DROP TABLE users;

-- 10. 驗證表已刪除
SELECT * FROM users;    -- 應(yīng)該失敗


todo::
    1. 實現(xiàn)!= >= <=等運算符
    2. 支持更多數(shù)據(jù)類型
    3. 支持更多函數(shù)
    4. 優(yōu)化查詢結(jié)果展示
    5. 支持更多索引類型
    6. 支持null值等
    7. 支持?jǐn)?shù)據(jù)落地本地文件
    8. 支持事務(wù)操作等

運行效果

cd ZiyiDB
go run cmd/main.go

在這里插入圖片描述

參考文章:https://notes.eatonphil.com/database-basics.html

到此這篇關(guān)于Go手寫數(shù)據(jù)庫ZiyiDB的實現(xiàn)的文章就介紹到這了,更多相關(guān)Go手寫ZiyiDB內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • Go語言中sync.Cond使用詳解

    Go語言中sync.Cond使用詳解

    本文主要介紹了Go語言中sync.Cond使用詳解,文中通過示例代碼介紹的非常詳細(xì),具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2021-09-09
  • golang批量執(zhí)行任務(wù)的通用模板分享

    golang批量執(zhí)行任務(wù)的通用模板分享

    這篇文章主要為大家詳細(xì)介紹了golang實現(xiàn)批量執(zhí)行任務(wù)的通用模板,文中的示例代碼講解詳細(xì),具有一定的學(xué)習(xí)價值,感興趣的小伙伴可以了解一下
    2023-11-11
  • Go 實現(xiàn)一次性打包各個平臺的可執(zhí)行程序

    Go 實現(xiàn)一次性打包各個平臺的可執(zhí)行程序

    這篇文章主要介紹了Go 實現(xiàn)一次性打包各個平臺的可執(zhí)行程序,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧
    2020-12-12
  • golang通過反射設(shè)置結(jié)構(gòu)體變量的值

    golang通過反射設(shè)置結(jié)構(gòu)體變量的值

    這篇文章主要介紹了golang通過反射設(shè)置結(jié)構(gòu)體變量的值操作,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧
    2021-04-04
  • golang獲取當(dāng)前時間、時間戳和時間字符串及它們之間的相互轉(zhuǎn)換方法

    golang獲取當(dāng)前時間、時間戳和時間字符串及它們之間的相互轉(zhuǎn)換方法

    這篇文章主要介紹了golang獲取當(dāng)前時間、時間戳和時間字符串及它們之間的相互轉(zhuǎn)換,本文通過實例代碼給大家介紹的非常詳細(xì),感興趣的朋友一起看看吧
    2025-04-04
  • Go設(shè)計模式之享元模式講解和代碼示例

    Go設(shè)計模式之享元模式講解和代碼示例

    享元是一種結(jié)構(gòu)型設(shè)計模式,它允許你在消耗少量內(nèi)存的情況下支持大量對象,模式通過共享多個對象的部分狀態(tài)來實現(xiàn)上述功能,換句話來說,享元會將不同對象的相同數(shù)據(jù)進行緩存以節(jié)省內(nèi)存,本文就將通過代碼示例給大家詳細(xì)介紹一下享元模式
    2023-06-06
  • 聊聊golang的defer的使用

    聊聊golang的defer的使用

    這篇文章主要介紹了聊聊golang的defer的使用,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2020-12-12
  • 淺談golang通道類型

    淺談golang通道類型

    本文主要介紹了淺談golang通道類型,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2023-02-02
  • go獲取協(xié)程(goroutine)號的實例

    go獲取協(xié)程(goroutine)號的實例

    這篇文章主要介紹了go獲取協(xié)程(goroutine)號的實例,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧
    2020-12-12
  • Golang學(xué)習(xí)之平滑重啟

    Golang學(xué)習(xí)之平滑重啟

    這篇文章主要介紹了Golang學(xué)習(xí)之平滑重啟,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧
    2018-08-08

最新評論

欧美精品免费aaaaaa| 欧洲国产成人精品91铁牛tv| 欧美熟妇一区二区三区仙踪林| 青青青青青手机视频| 午夜精品亚洲精品五月色| 中国视频一区二区三区| 手机看片福利盒子日韩在线播放| 99精品亚洲av无码国产另类 | 亚洲精品ww久久久久久| 国产刺激激情美女网站| 亚洲天堂av最新网址| 男人的天堂av日韩亚洲| 男人操女人逼逼视频网站| 国产综合高清在线观看| 欧美专区日韩专区国产专区| 91人妻人人做人人爽在线| 99久久99一区二区三区| 粉嫩av懂色av蜜臀av| 午夜场射精嗯嗯啊啊视频| 欧美精产国品一二三产品价格| 老司机免费福利视频网| 亚洲av黄色在线网站| 成熟丰满熟妇高潮xx×xx| 成人区人妻精品一区二视频| 日韩欧美一级精品在线观看| 亚洲欧美清纯唯美另类| 99久久中文字幕一本人| 精品一区二区三区三区88| 免费福利av在线一区二区三区| 久久热久久视频在线观看| 特大黑人巨大xxxx| 一区二区麻豆传媒黄片| 美女在线观看日本亚洲一区| 国产丰满熟女成人视频| 青草青永久在线视频18| 在线国产中文字幕视频| 和邻居少妇愉情中文字幕| 午夜av一区二区三区| 国产精品午夜国产小视频| 国产伦精品一区二区三区竹菊| 3344免费偷拍视频| 91极品新人『兔兔』精品新作 | 日韩亚洲高清在线观看| 国产露脸对白在线观看| 国产妇女自拍区在线观看| av中文字幕在线观看第三页| 亚洲成a人片777777| 日韩三级电影华丽的外出| 亚洲一区二区三区av网站| av日韩在线观看大全| 国产白袜脚足J棉袜在线观看| 爱有来生高清在线中文字幕| 国产黄色高清资源在线免费观看| 男生用鸡操女生视频动漫| 好太好爽好想要免费| 中文字幕日本人妻中出| 国产激情av网站在线观看| av欧美网站在线观看| 天天日天天干天天要| 中文字幕人妻熟女在线电影| 黄网十四区丁香社区激情五月天| 专门看国产熟妇的网站| 爆乳骚货内射骚货内射在线| 绝顶痉挛大潮喷高潮无码| 中文字幕免费福利视频6| 亚洲欧美清纯唯美另类| 91色网站免费在线观看| 好太好爽好想要免费| 啊啊好大好爽啊啊操我啊啊视频 | 精品成人啪啪18免费蜜臀| 亚洲粉嫩av一区二区三区| 毛片一级完整版免费| 精品一区二区三区欧美| av在线免费中文字幕| 天天干天天操天天插天天日| 久草视频首页在线观看| 国产激情av网站在线观看| 蝴蝶伊人久久中文娱乐网| 日本黄在免费看视频| 午夜精品福利一区二区三区p| 日本熟妇色熟妇在线观看| 91中文字幕免费在线观看| 亚洲成人激情av在线| 中文字幕第一页国产在线| 中国黄片视频一区91| 成熟熟女国产精品一区| 特一级特级黄色网片| 亚洲av香蕉一区区二区三区犇| 午夜毛片不卡免费观看视频| 97年大学生大白天操逼| 久久99久久99精品影院| 91人妻精品久久久久久久网站| a v欧美一区=区三区| 亚洲欧洲一区二区在线观看| 亚洲av成人网在线观看| 国产精品免费不卡av| 视频久久久久久久人妻| 天天干夜夜操啊啊啊| 桃色视频在线观看一区二区| 久久这里有免费精品| 在线视频精品你懂的| 亚洲人妻av毛片在线| 直接观看免费黄网站| 91桃色成人网络在线观看| 亚洲蜜臀av一区二区三区九色| 制丝袜业一区二区三区| 97超碰最新免费在线观看| 日本午夜爽爽爽爽爽视频在线观看| 国产极品精品免费视频| 国产麻豆91在线视频| 日韩三级电影华丽的外出| av中文字幕福利网| 亚洲欧美激情中文字幕| nagger可以指黑人吗| 国产日本欧美亚洲精品视| 久久农村老妇乱69系列| 91精品国产高清自在线看香蕉网| 欧美亚洲一二三区蜜臀| 免费在线观看视频啪啪| 少妇露脸深喉口爆吞精| 亚洲视频在线观看高清| 硬鸡巴动态操女人逼视频| 国产欧美日韩第三页| 午夜精品久久久久麻豆影视| 欧美专区第八页一区在线播放 | 在线国产日韩欧美视频| 成人国产激情自拍三区| 精品一区二区三区欧美| 久久久久久97三级| 中文字幕人妻被公上司喝醉在线| 激情人妻校园春色亚洲欧美| 98视频精品在线观看| 天天日天天干天天要| 大香蕉玖玖一区2区| 成人蜜桃美臀九一一区二区三区| 成人动漫大肉棒插进去视频| 女生被男生插的视频网站| 在线网站你懂得老司机| 精品日产卡一卡二卡国色天香| 亚洲国产欧美一区二区三区…| av天堂加勒比在线| 中文字幕亚洲久久久| 亚洲男人让女人爽的视频| 中文字幕在线第一页成人 | 亚洲 欧美 精品 激情 偷拍| 欧美日韩人妻久久精品高清国产| 天天色天天操天天透| 亚洲护士一区二区三区| 日韩黄色片在线观看网站| 精品国产在线手机在线| 97精品综合久久在线| 1024久久国产精品| 亚洲欧美一区二区三区爱爱动图| 岛国毛片视频免费在线观看| 视频一区 二区 三区 综合| 日韩美女搞黄视频免费| 特级欧美插插插插插bbbbb| 中文字幕,亚洲人妻| 亚洲国产欧美一区二区丝袜黑人| 超pen在线观看视频公开97| 色偷偷伊人大杳蕉综合网| chinese国产盗摄一区二区| 深田咏美亚洲一区二区| 人妻熟女在线一区二区| 亚洲青青操骚货在线视频| 93视频一区二区三区| 2022中文字幕在线| 精品人人人妻人人玩日产欧| 亚洲成人精品女人久久久| 大骚逼91抽插出水视频| 青青青青视频在线播放| 青青草国内在线视频精选| 19一区二区三区在线播放| 3344免费偷拍视频| 欧美一区二区三区高清不卡tv | 制服丝袜在线人妻中文字幕| 亚洲成人情色电影在线观看| 中文字幕AV在线免费看 | 国产午夜福利av导航| 午夜毛片不卡免费观看视频| 欧美成一区二区三区四区| 成年人黄色片免费网站| 中文字幕第1页av一天堂网 | 精品成人啪啪18免费蜜臀| 欲乱人妻少妇在线视频裸| 热99re69精品8在线播放| av线天堂在线观看| 天天操天天干天天插| 国产卡一卡二卡三乱码手机| 一级黄色片夫妻性生活| 婷婷综合蜜桃av在线| 黑人变态深video特大巨大| 大肉大捧一进一出好爽在线视频| 99精品国产aⅴ在线观看| 99精品国产自在现线观看| 久久这里只有精彩视频免费| 日韩欧美亚洲熟女人妻| 色婷婷综合激情五月免费观看| 青青青青操在线观看免费| 欧美精品久久久久久影院| 亚洲免费视频欧洲免费视频| 大香蕉大香蕉大香蕉大香蕉大香蕉| 老司机福利精品视频在线| 久久艹在线观看视频| 午夜在线观看一区视频| 一级黄片大鸡巴插入美女| 狠狠躁夜夜躁人人爽天天天天97| 在线观看视频一区麻豆| 888欧美视频在线| 香港三日本三韩国三欧美三级| 啪啪啪啪啪啪啪啪av| 噜噜色噜噜噜久色超碰| 国产成人精品av网站| 亚洲激情唯美亚洲激情图片| 人妻熟女中文字幕aⅴ在线 | 午夜场射精嗯嗯啊啊视频| 亚洲少妇人妻无码精品| 国产一级麻豆精品免费| 人人妻人人澡欧美91精品| 成年女人免费播放视频| 经典av尤物一区二区| 中文字幕av熟女人妻| 亚洲av无女神免非久久| 亚洲男人的天堂a在线| 日本免费午夜视频网站| 亚洲高清自偷揄拍自拍| 99国产精品窥熟女精品| 中文字幕中文字幕人妻| 在线播放 日韩 av| 成熟熟女国产精品一区| 97超碰人人搞人人| 日韩欧美国产精品91| 亚洲色偷偷综合亚洲AV伊人| 亚洲成人国产综合一区| 午夜成午夜成年片在线观看| 青春草视频在线免费播放| 国产白嫩美女一区二区| 天天摸天天干天天操科普| 一区二区麻豆传媒黄片| 日本黄色特一级视频| 蝴蝶伊人久久中文娱乐网| 一区二区三区另类在线| 男人的天堂av日韩亚洲| 中文字幕在线视频一区二区三区| 一区二区在线视频中文字幕| 日韩成人性色生活片| 早川濑里奈av黑人番号| 2020国产在线不卡视频 | 国产欧美精品免费观看视频| 亚洲精品 日韩电影| 精品成人午夜免费看| 99re国产在线精品| 77久久久久国产精产品| 五十路息与子猛烈交尾视频| 亚洲 国产 成人 在线| 日韩av有码中文字幕| 日本丰满熟妇BBXBBXHD| 亚洲另类伦春色综合小| 天天操天天爽天天干| 色综合天天综合网国产成人| 亚洲中文字幕乱码区| 福利视频一区二区三区筱慧| 久久尻中国美女视频| 亚洲午夜电影之麻豆| 青青青青青操视频在线观看| 国产亚洲视频在线二区| 日本av高清免费网站| 老司机福利精品视频在线| 美女操逼免费短视频下载链接| 国产高清在线在线视频| 久久久制服丝袜中文字幕| 国产熟妇人妻ⅹxxxx麻豆| 欧美日韩熟女一区二区三区| 黄网十四区丁香社区激情五月天| 中文字幕亚洲中文字幕| 亚洲av无码成人精品区辽| 亚洲中文字字幕乱码| 日韩欧美高清免费在线 | 国产妇女自拍区在线观看| 91精品国产高清自在线看香蕉网| 2022国产精品视频| 五十路老熟女码av| 亚国产成人精品久久久| 亚洲一区久久免费视频| 国产性生活中老年人视频网站| 亚洲成人av一区在线| 成年午夜影片国产片| 欧美aa一级一区三区四区 | 免费观看理论片完整版| 19一区二区三区在线播放| 国产精品一二三不卡带免费视频| 一区二区三区精品日本| 日韩欧美在线观看不卡一区二区| av森泽佳奈在线观看| 快插进小逼里大鸡吧视频| 欧美激情精品在线观看| 国产使劲操在线播放| 欧亚乱色一区二区三区| 日韩欧美国产一区不卡| 在线免费观看靠比视频的网站 | 自拍偷拍vs一区二区三区| 黑人变态深video特大巨大| 亚洲中文字幕综合小综合| 日本熟女50视频免费| 91www一区二区三区| 亚洲精品国品乱码久久久久| 999热精品视频在线| 亚洲国产第一页在线观看| 日韩黄色片在线观看网站| 亚洲高清自偷揄拍自拍| 亚洲一区二区三区五区 | 亚洲精品 日韩电影| xxx日本hd高清| 亚洲另类在线免费观看| 亚洲成人av一区在线| 国产综合高清在线观看| 欧美精品国产综合久久| 国产va在线观看精品| 小穴多水久久精品免费看| 国产麻豆国语对白露脸剧情 | 亚洲天堂成人在线观看视频网站| 精品美女福利在线观看| av手机在线观播放网站| 欧美综合婷婷欧美综合| 亚洲成av人无码不卡影片一| 99精品国自产在线人| 亚洲欧美日韩视频免费观看| 天天射夜夜操狠狠干| 亚洲va国产va欧美va在线| 18禁精品网站久久| 人人超碰国字幕观看97| 亚洲国际青青操综合网站| 国产普通话插插视频| 成人av天堂丝袜在线观看| 亚洲午夜电影之麻豆| 最后99天全集在线观看| 精品亚洲在线免费观看| 加勒比视频在线免费观看| 欧美日韩高清午夜蜜桃大香蕉| 黄色视频在线观看高清无码 | 亚洲国产精品久久久久久6| 亚洲一区二区三区av网站| 黑人解禁人妻叶爱071| 中文字幕高清在线免费播放| 污污小视频91在线观看| 久久久久久久久久一区二区三区 | 日本成人一区二区不卡免费在线| 55夜色66夜色国产精品站| 五月天色婷婷在线观看视频免费| 亚洲av可乐操首页| 亚洲熟女女同志女同| 91人妻人人做人人爽在线| av老司机精品在线观看| 亚洲嫩模一区二区三区| 综合一区二区三区蜜臀| 亚洲最大黄了色网站| 国产麻豆精品人妻av| 丝袜美腿欧美另类 中文字幕| 天天干天天操天天扣| 欧美日韩一区二区电影在线观看| 黄色录像鸡巴插进去| 久久久久久97三级| 日本av熟女在线视频| 揄拍成人国产精品免费看视频| 超黄超污网站在线观看| 中文字幕乱码av资源| 亚洲一区二区三区久久午夜| 国产成人精品av网站| 在线观看的a站 最新| 日韩美女搞黄视频免费| 亚洲欧美激情人妻偷拍| 涩涩的视频在线观看视频| 日韩美女搞黄视频免费| 岛国一区二区三区视频在线| 久久这里只有精品热视频| 国产自拍在线观看成人| 国产在线拍揄自揄视频网站| 77久久久久国产精产品| av在线免费观看亚洲天堂| 国产女人被做到高潮免费视频| 亚洲av黄色在线网站| 国产欧美精品一区二区高清 | 天天操天天干天天日狠狠插| 国产免费高清视频视频| 婷婷久久久久深爱网| 日本午夜爽爽爽爽爽视频在线观看| 青青擦在线视频国产在线| 最新国产精品网址在线观看| av天堂加勒比在线| av网站色偷偷婷婷网男人的天堂| 国产精品视频男人的天堂| 老司机免费视频网站在线看| 午夜美女少妇福利视频| 99婷婷在线观看视频| 51国产偷自视频在线播放| 在线国产日韩欧美视频| 特一级特级黄色网片| 亚洲午夜精品小视频| 97年大学生大白天操逼 | 9国产精品久久久久老师| 亚洲午夜福利中文乱码字幕| 动漫精品视频在线观看| 在线国产中文字幕视频| 欧美黑人巨大性xxxxx猛交| 干逼又爽又黄又免费的视频| 国产一区自拍黄视频免费观看| 我想看操逼黄色大片| 国产一区自拍黄视频免费观看| 中文字幕中文字幕人妻| 玩弄人妻熟妇性色av少妇| 人妻少妇性色欲欧美日韩| 欧美中国日韩久久精品| 国产三级精品三级在线不卡| 国产视频一区二区午夜| 少妇露脸深喉口爆吞精| 啊啊好慢点插舔我逼啊啊啊视频 | 国产精品伦理片一区二区| 福利视频广场一区二区| 天天做天天爽夜夜做少妇| 亚洲欧美久久久久久久久| 亚洲天堂精品福利成人av| 91国产在线视频免费观看| 亚洲人一区二区中文字幕| 亚洲一区久久免费视频| 欧美韩国日本国产亚洲| 亚洲视频在线视频看视频在线| 99热久久这里只有精品| 自拍偷拍亚洲精品第2页| 91精品国产高清自在线看香蕉网| 日韩熟女av天堂系列| 中文字幕无码日韩专区免费| 一区二区在线视频中文字幕| 啊啊好大好爽啊啊操我啊啊视频| 啪啪啪啪啪啪啪啪啪啪黄色| 亚洲精品成人网久久久久久小说| 91she九色精品国产| 欧洲国产成人精品91铁牛tv| 91久久综合男人天堂| 东游记中文字幕版哪里可以看到| 日本韩国免费福利精品| 91 亚洲视频在线观看| 色综合久久五月色婷婷综合| 亚洲一区二区三区精品乱码| 国产午夜无码福利在线看| 91色老99久久九九爱精品| 亚洲偷自拍高清视频| 人人爱人人妻人人澡39| 欧美黑人与人妻精品| 老司机免费视频网站在线看| 99视频精品全部15| 99热99这里精品6国产| 国产自拍在线观看成人| 成人国产小视频在线观看| 国产女人被做到高潮免费视频| 麻豆精品成人免费视频| 超污视频在线观看污污污| 五月天色婷婷在线观看视频免费| 免费看高清av的网站| av网站色偷偷婷婷网男人的天堂| 天天日天天干天天干天天日| 中文字幕高清在线免费播放| 年轻的人妻被夫上司侵犯| 国产精品sm调教视频| 摧残蹂躏av一二三区| 91精品国产91久久自产久强 | 天天操夜夜骑日日摸| 国产黄色大片在线免费播放| 国产精品久久久久国产三级试频| 少妇被强干到高潮视频在线观看| 中英文字幕av一区| 国产露脸对白在线观看| 亚洲一区二区久久久人妻| 国产精品黄大片在线播放| 国产一线二线三线的区别在哪| 天天日天天透天天操| ka0ri在线视频| 懂色av蜜桃a v| 国产精品自偷自拍啪啪啪| 午夜福利人人妻人人澡人人爽| 六月婷婷激情一区二区三区| 亚洲久久午夜av一区二区| 久久99久久99精品影院| 亚洲国产成人在线一区| 国产精品久久久久国产三级试频 | 欧美精品资源在线观看| 久久精品久久精品亚洲人| 香蕉aⅴ一区二区三区| 农村胖女人操逼视频| 亚洲欧美色一区二区| 嫩草aⅴ一区二区三区| www天堂在线久久| 精品国产高潮中文字幕| 日本少妇人妻xxxxxhd| 超级福利视频在线观看| 2022国产综合在线干| 日韩国产乱码中文字幕| 91天堂精品一区二区| 国产欧美精品一区二区高清 | 亚洲卡1卡2卡三卡四老狼| 国产福利小视频二区| 人妻激情图片视频小说| 漂亮 人妻被中出中文| 国产亚州色婷婷久久99精品| 亚洲精品中文字幕下载| 亚洲护士一区二区三区| 亚洲av日韩高清hd| 免费在线黄色观看网站| 欧美成人小视频在线免费看| 操人妻嗷嗷叫视频一区二区| 国产一区二区神马久久| 亚洲一区二区三区精品视频在线| 在线免费91激情四射 | 婷婷久久久综合中文字幕| 欧美精品亚洲精品日韩在线| 福利午夜视频在线观看| 国产美女精品福利在线| 98精产国品一二三产区区别| 91精品激情五月婷婷在线| 曰本无码人妻丰满熟妇啪啪| 国产精品一区二区三区蜜臀av| 91精品一区二区三区站长推荐| 欧美熟妇一区二区三区仙踪林| 国产真实灌醉下药美女av福利| 老司机免费福利视频网| 一区二区三区综合视频| 日本免费一级黄色录像| 首之国产AV医生和护士小芳| av中文字幕国产在线观看| 国产熟妇人妻ⅹxxxx麻豆| 把腿张开让我插进去视频| 国产真实乱子伦a视频 | 在线免费91激情四射| 欧美日韩高清午夜蜜桃大香蕉| 日本一本午夜在线播放| 蜜臀av久久久久蜜臀av麻豆| 中文字幕人妻被公上司喝醉在线| 91超碰青青中文字幕| 五月精品丁香久久久久福利社| 亚洲青青操骚货在线视频| 2021最新热播中文字幕| 日韩精品中文字幕福利| 精品区一区二区三区四区人妻| 久久久精品国产亚洲AV一| 青青尤物在线观看视频网站| 一区二区三区国产精选在线播放 | 天天操夜夜操天天操天天操| 国产九色91在线视频| 欧美亚洲少妇福利视频| av完全免费在线观看av| av新中文天堂在线网址| 岳太深了紧紧的中文字幕| 日韩美女福利视频网| av森泽佳奈在线观看| 色婷婷六月亚洲综合香蕉| 天天干夜夜操天天舔| 免费看高清av的网站| 极品粉嫩小泬白浆20p主播| 久草视频 久草视频2| 国产视频在线视频播放| 亚洲精品 欧美日韩| 成年人啪啪视频在线观看| 大胆亚洲av日韩av| 色婷婷久久久久swag精品| 精品久久久久久久久久中文蒉 | 午夜毛片不卡在线看| 精品亚洲在线免费观看| 国产高清在线在线视频| 亚洲欧美日韩视频免费观看| 亚洲国产成人av在线一区| 最新欧美一二三视频| 中字幕人妻熟女人妻a62v网| 视频在线免费观看你懂得| 欧美一区二区三区久久久aaa| 超级福利视频在线观看| 中文字幕高清免费在线人妻| 青娱乐蜜桃臀av色| 91极品大一女神正在播放| 啊啊啊视频试看人妻| 黄工厂精品视频在线观看| 国产精品视频男人的天堂| 男人天堂色男人av| 天天日天天日天天射天天干| 亚洲精品国产综合久久久久久久久 | 黄色男人的天堂视频| 亚洲精品欧美日韩在线播放| 亚洲午夜高清在线观看| 中出中文字幕在线观看| 天天操天天干天天艹| 97精品综合久久在线| 大肉大捧一进一出好爽在线视频| 国产伦精品一区二区三区竹菊| 熟女人妻三十路四十路人妻斩| 伊拉克及约旦宣布关闭领空| 黑人巨大的吊bdsm| 好太好爽好想要免费| 久久丁香婷婷六月天| 最新国产精品网址在线观看| 亚洲 清纯 国产com| 亚洲欧美一区二区三区电影| 欧美黑人性猛交xxxxⅹooo| 亚洲精品 日韩电影| 欧美另类重口味极品在线观看| 国产男女视频在线播放| 成人性黑人一级av| 天天射,天天操,天天说| 大屁股肉感人妻中文字幕在线| 绝顶痉挛大潮喷高潮无码| 极品丝袜一区二区三区| 91p0rny九色露脸熟女| 人人妻人人澡欧美91精品| 亚洲一级美女啪啪啪| 欧美日韩高清午夜蜜桃大香蕉| 国产九色91在线观看精品| 亚洲精品av在线观看| 国产一区av澳门在线观看| 91片黄在线观看喷潮| 中文字幕高清资源站| 亚洲熟女综合色一区二区三区四区| 中文字幕人妻三级在线观看| 99热国产精品666| wwwxxx一级黄色片| 亚洲一区二区三区久久午夜 | 久久丁香花五月天色婷婷| 亚洲午夜精品小视频| 国产精品3p和黑人大战| 91人妻人人做人人爽在线| av一区二区三区人妻| 亚洲精品亚洲人成在线导航| 另类av十亚洲av| 93人妻人人揉人人澡人人| 亚洲 图片 欧美 图片| 天天射夜夜操狠狠干| 欧美地区一二三专区| 黑人进入丰满少妇视频| 91桃色成人网络在线观看| 38av一区二区三区| 国产亚洲欧美另类在线观看| 午夜精品久久久久久99热| 亚洲成a人片777777| 91免费观看在线网站| 日本美女成人在线视频| 大鸡吧插逼逼视频免费看| 欧美精产国品一二三区| 国产亚洲精品欧洲在线观看| 欧美亚洲自偷自拍 在线| 国产一区二区欧美三区| 中文字幕 人妻精品| 亚洲欧美一区二区三区电影| 人妻少妇亚洲精品中文字幕| 国产精品黄色的av| 在线不卡成人黄色精品| 男人天堂最新地址av| lutube在线成人免费看| 亚洲免费成人a v| 国产精品自拍视频大全| 中文字幕 亚洲av| 国产成人精品福利短视频| 亚洲另类伦春色综合小| 日韩中文字幕福利av| 老司机你懂得福利视频| 久草视频在线看免费| 午夜精品福利一区二区三区p | 日本精品美女在线观看| 激情五月婷婷综合色啪| 免费观看理论片完整版| 成人高清在线观看视频| 啪啪啪操人视频在线播放| 东京热男人的av天堂| 久久永久免费精品人妻专区| 粉嫩小穴流水视频在线观看| 九色视频在线观看免费| 国产实拍勾搭女技师av在线| 国产一区二区欧美三区| 日韩欧美一级aa大片| 国产黄色片在线收看| 99精品视频在线观看婷婷| 亚洲va国产va欧美精品88| 中文字幕日韩无敌亚洲精品| 日本精品视频不卡一二三| 第一福利视频在线观看| 91综合久久亚洲综合| 亚洲视频在线视频看视频在线| 亚洲卡1卡2卡三卡四老狼| av网址在线播放大全| 久草福利电影在线观看| 日本高清成人一区二区三区| 青青青青青青青在线播放视频| 黄色中文字幕在线播放| 3344免费偷拍视频| 国产中文字幕四区在线观看| 91试看福利一分钟| 韩国黄色一级二级三级| 大香蕉伊人中文字幕| 青青伊人一精品视频| 19一区二区三区在线播放| 久久久久久cao我的性感人妻| 三级av中文字幕在线观看| 99热国产精品666| 天天日天天干天天干天天日| 人妻凌辱欧美丰满熟妇| 成年女人免费播放视频| 午夜福利人人妻人人澡人人爽| 亚洲成人精品女人久久久| 一区二区视频在线观看视频在线| 5528327男人天堂| 欧美久久一区二区伊人| 97人人模人人爽人人喊| 91传媒一区二区三区| 熟女人妻一区二区精品视频| 成年午夜免费无码区| 亚洲综合乱码一区二区| 在线观看的a站 最新| 日韩伦理短片在线观看| 日本黄色特一级视频| 精品高跟鞋丝袜一区二区| 老熟妇xxxhd老熟女| 天天色天天操天天舔| 狠狠躁狠狠爱网站视频| 亚洲一区二区三区在线高清| 人人妻人人爽人人添夜| 开心 色 六月 婷婷| 粉嫩av蜜乳av蜜臀| 午夜婷婷在线观看视频| 91久久国产成人免费网站| 91色秘乱一区二区三区| 人妻熟女在线一区二区| 精品一区二区三区午夜| 天天操夜夜骑日日摸| 国产精品大陆在线2019不卡| 精品美女久久久久久| 岛国黄色大片在线观看| 人人人妻人人澡人人| 男人和女人激情视频| 国产麻豆91在线视频| av高潮迭起在线观看| 91麻豆精品久久久久| 伊拉克及约旦宣布关闭领空| 日韩少妇人妻精品无码专区| 不戴胸罩引我诱的隔壁的人妻| 天天干天天操天天爽天天摸| 国产精品一区二区三区蜜臀av| tube69日本少妇| av网站色偷偷婷婷网男人的天堂| 91超碰青青中文字幕| 激情伦理欧美日韩中文字幕| 绝顶痉挛大潮喷高潮无码| 精品国产在线手机在线| 日韩一区二区电国产精品| 精品黑人一区二区三区久久国产| 国产精品一区二区久久久av| 亚洲精品麻豆免费在线观看| 亚洲 欧美 自拍 偷拍 在线| 亚洲av日韩av网站| 在线播放一区二区三区Av无码| 年轻的人妻被夫上司侵犯| 天天操天天干天天日狠狠插| 欧美日本aⅴ免费视频| 亚洲最大黄了色网站| 日韩中文字幕在线播放第二页| 亚洲一级美女啪啪啪| av天堂资源最新版在线看| 久草视频 久草视频2| 狠狠鲁狠狠操天天晚上干干| 热思思国产99re| 天天日天天爽天天干| 一区二区麻豆传媒黄片| 国产白嫩美女一区二区| 国产又色又刺激在线视频| 全国亚洲男人的天堂| 国产91久久精品一区二区字幕| 欧美国产亚洲中英文字幕| 亚洲美女自偷自拍11页| 9l人妻人人爽人人爽| 亚洲av日韩av网站| 青青热久免费精品视频在线观看 | 国产黑丝高跟鞋视频在线播放| 五十路丰满人妻熟妇| 超黄超污网站在线观看| 五十路丰满人妻熟妇| 五十路熟女人妻一区二区9933| 色哟哟在线网站入口| 热久久只有这里有精品| 91人妻精品一区二区在线看| 中文字幕在线观看极品视频| 欧美偷拍亚洲一区二区| 亚洲福利精品福利精品福利| 一个色综合男人天堂| 亚洲午夜在线视频福利| 自拍偷拍,中文字幕| 午夜福利资源综合激情午夜福利资| 五十路在线观看完整版| 亚洲中文字幕国产日韩| 777奇米久久精品一区| 亚洲在线免费h观看网站| 中文字日产幕乱六区蜜桃| 888欧美视频在线| 欧美日本在线观看一区二区| 专门看国产熟妇的网站| 五十路熟女人妻一区二| 伊人日日日草夜夜草| 激情内射在线免费观看| 日本一二三区不卡无| 经典国语激情内射视频| 超碰在线中文字幕一区二区| 91精品视频在线观看免费| 欧美日韩精品永久免费网址 | 91大神福利视频网| 久久精品国产23696| 国产实拍勾搭女技师av在线| 一色桃子人妻一区二区三区| 日本韩国亚洲综合日韩欧美国产| 色吉吉影音天天干天天操| 精彩视频99免费在线| 国产午夜亚洲精品不卡在线观看| 日韩精品激情在线观看| 9l人妻人人爽人人爽| 亚洲一区制服丝袜美腿| 天天干天天插天天谢| 岛国毛片视频免费在线观看| 国产黄网站在线观看播放| 不卡一不卡二不卡三| 91快播视频在线观看| 国产精品视频一区在线播放| 最新激情中文字幕视频| 91在线视频在线精品3| 六月婷婷激情一区二区三区| 蜜臀av久久久久蜜臀av麻豆| 91片黄在线观看喷潮| 男人的天堂在线黄色| 五十路在线观看完整版| 97年大学生大白天操逼| 老熟妇xxxhd老熟女| 99精品国产自在现线观看| 沈阳熟妇28厘米大战黑人| 精品一区二区三四区| 亚洲av香蕉一区区二区三区犇| 日本熟妇一区二区x x| 国产精品久久久久久久女人18| 视频一区二区综合精品| 男人天堂最新地址av| 中文字幕在线观看国产片| 国产av自拍偷拍盛宴| 93视频一区二区三区| 老师让我插进去69AV| 99精品国产免费久久| 久久久精品欧洲亚洲av| 国产白袜脚足J棉袜在线观看| 国产精品视频男人的天堂| 超级福利视频在线观看| 国产精品大陆在线2019不卡| 在线免费观看日本伦理| 天天摸天天干天天操科普| 国产大学生援交正在播放| 亚洲特黄aaaa片| 2020久久躁狠狠躁夜夜躁| 十八禁在线观看地址免费| 2017亚洲男人天堂| 99人妻视频免费在线| 女同互舔一区二区三区| 99精品亚洲av无码国产另类| 五十路人妻熟女av一区二区| 中文字幕一区二区三区蜜月| 亚洲精品福利网站图片| 在线免费观看av日韩| 沙月文乃人妻侵犯中文字幕在线 | 精产国品久久一二三产区区别| 爱爱免费在线观看视频| 强行扒开双腿猛烈进入免费版| 免费在线黄色观看网站| 爱有来生高清在线中文字幕| 免费在线播放a级片| 自拍 日韩 欧美激情| 日本一区美女福利视频| 中文乱理伦片在线观看| 婷婷综合蜜桃av在线| 亚洲1卡2卡三卡4卡在线观看| 亚洲综合在线视频可播放| 顶级尤物粉嫩小尤物网站| 精品一区二区三区午夜| 99久久激情婷婷综合五月天| 免费av岛国天堂网站| 直接能看的国产av| 中文字幕高清免费在线人妻| 不卡一区一区三区在线| 在线免费观看黄页视频| 日韩亚洲高清在线观看| 国产精彩福利精品视频| 日本韩国免费福利精品| 中文字幕1卡1区2区3区| 青青草原色片网站在线观看| 丝袜国产专区在线观看| 久草福利电影在线观看| 亚洲国产香蕉视频在线播放| 国产一区二区火爆视频| 久久精品久久精品亚洲人| 啊啊好慢点插舔我逼啊啊啊视频 | 人妻少妇亚洲一区二区| 国内自拍第一页在线观看| 国产精品黄大片在线播放| 久久精品国产999| 大胆亚洲av日韩av| av中文字幕在线导航| 果冻传媒av一区二区三区| 欧美爆乳肉感大码在线观看| 一区二区三区激情在线| 国产激情av网站在线观看| 国产成人一区二区三区电影网站| 大香蕉伊人国产在线| 亚洲无码一区在线影院| 欧美日韩高清午夜蜜桃大香蕉| gogo国模私拍视频| 男人天堂最新地址av| 视频一区 二区 三区 综合| 中文字幕 亚洲av| 91精品激情五月婷婷在线| 国产福利小视频大全| 精品首页在线观看视频| av中文在线天堂精品| 午夜久久久久久久99| 性色av一区二区三区久久久| 亚洲特黄aaaa片| 日本三极片视频网站观看| 国产精品久久久久久久女人18| 99热色原网这里只有精品| 午夜青青草原网在线观看| 国产午夜福利av导航| 啊用力插好舒服视频| 中国黄片视频一区91| 久久久久五月天丁香社区| 人人妻人人澡欧美91精品| 不卡精品视频在线观看| 亚洲av日韩av网站| 女警官打开双腿沦为性奴| 亚洲第一黄色在线观看| 在线观看国产网站资源| 日本少妇的秘密免费视频| 免费十精品十国产网站| 超级福利视频在线观看| 精品av国产一区二区三区四区| 很黄很污很色的午夜网站在线观看| 黄色片黄色片wyaa| av在线免费中文字幕| 东京热男人的av天堂| 亚洲美女美妇久久字幕组| 日本高清成人一区二区三区| 人妻素人精油按摩中出| 久久丁香婷婷六月天| 又黄又刺激的午夜小视频| 夜夜躁狠狠躁日日躁麻豆内射 | 国产黄色片在线收看| 欧美精品激情在线最新观看视频| 家庭女教师中文字幕在线播放| 人妻丝袜精品中文字幕| 黄色的网站在线免费看| 亚洲欧美成人综合在线观看| 夜色17s精品人妻熟女| 国产污污污污网站在线| 91精品国产黑色丝袜| 亚洲精品欧美日韩在线播放| 青青青艹视频在线观看| 精品国产在线手机在线| 啪啪啪啪啪啪啪啪av| 午夜福利资源综合激情午夜福利资 | 青青草亚洲国产精品视频| 啪啪啪啪啪啪啪啪啪啪黄色| 日本五十路熟新垣里子| 天天操夜夜操天天操天天操| 中文字幕亚洲久久久| 中文字幕欧美日韩射射一| 亚洲精品在线资源站| 亚国产成人精品久久久| 免费无毒热热热热热热久| 亚洲精品在线资源站| 亚洲精品 日韩电影| av中文在线天堂精品| 成人精品视频99第一页| 激情人妻校园春色亚洲欧美| 九色精品视频在线播放| 国产精品国色综合久久 | 天天做天天干天天操天天射| 亚洲天堂成人在线观看视频网站| 91大神福利视频网| 亚洲欧美一区二区三区爱爱动图| 亚洲另类伦春色综合小| 99精品国自产在线人| 精品久久婷婷免费视频| 男人靠女人的逼视频| 3337p日本欧洲大胆色噜噜| 国产视频精品资源网站| 激情国产小视频在线| 初美沙希中文字幕在线| 无套猛戳丰满少妇人妻| 欧美80老妇人性视频| 久久h视频在线观看| 精品suv一区二区69| 成人30分钟免费视频| 天天日天天玩天天摸| 欧美性受xx黑人性猛交| 亚欧在线视频你懂的| 在线观看免费岛国av| 深田咏美亚洲一区二区| 欧美亚洲一二三区蜜臀| 青青草原网站在线观看| avjpm亚洲伊人久久| 青青草人人妻人人妻| aiss午夜免费视频| 欧美偷拍亚洲一区二区| 男生用鸡操女生视频动漫| 日本免费午夜视频网站| 欧美性感尤物人妻在线免费看| 75国产综合在线视频| 亚洲精品午夜aaa久久| 日韩精品中文字幕在线| 加勒比视频在线免费观看| 日曰摸日日碰夜夜爽歪歪| 91久久人澡人人添人人爽乱| 99热久久这里只有精品| 日本在线不卡免费视频| 精内国产乱码久久久久久| 91亚洲国产成人精品性色| 91在线视频在线精品3| 一区二区三区 自拍偷拍| 啊啊好大好爽啊啊操我啊啊视频| 天天射夜夜操综合网| 内射久久久久综合网| 精品日产卡一卡二卡国色天香 | 美女大bxxxx内射| 国产夫妻视频在线观看免费| 福利午夜视频在线观看| 美女福利视频导航网站| 国产成人一区二区三区电影网站| 大鸡八强奸视频在线观看| 日本性感美女三级视频| 深夜男人福利在线观看| 91中文字幕最新合集| 黄色片黄色片wyaa| 亚洲男人让女人爽的视频| 阿v天堂2014 一区亚洲| 91人妻精品一区二区久久| 91精品视频在线观看免费| 人妻丝袜榨强中文字幕| 欧美怡红院视频在线观看| 黑人乱偷人妻中文字幕| 亚洲免费va在线播放| 国产成人精品午夜福利训2021| 欧美麻豆av在线播放| 免费黄高清无码国产| 99精品免费观看视频| 成人av亚洲一区二区| 午夜精品九一唐人麻豆嫩草成人| 丝袜长腿第一页在线| 一区二区三区四区五区性感视频| 国产片免费观看在线观看| 白嫩白嫩美女极品国产在线观看| av网址在线播放大全| 巨乳人妻日下部加奈被邻居中出| 夜色撩人久久7777| 中文字幕在线欧美精品| 中国视频一区二区三区| 黄色视频成年人免费观看| 欧美成人一二三在线网| 边摸边做超爽毛片18禁色戒| 国产日韩欧美美利坚蜜臀懂色| 天堂女人av一区二区| 久久丁香花五月天色婷婷| 黄色视频在线观看高清无码 | 亚洲国产在人线放午夜| 护士特殊服务久久久久久久| 欧美男同性恋69视频| 女同性ⅹxx女同hd| 久久精品国产999| 精品一区二区三四区| 日韩欧美高清免费在线| 亚洲蜜臀av一区二区三区九色| 91麻豆精品传媒国产黄色片| 亚洲人人妻一区二区三区| 红杏久久av人妻一区| 天天日天天爽天天爽| 中国熟女@视频91| 国产97视频在线精品| 337p日本大胆欧美人| 美女小视频网站在线| 天天日天天干天天要| 人妻少妇av在线观看| 国产精品中文av在线播放| 人人妻人人人操人人人爽| av成人在线观看一区| 国产av福利网址大全| 日本三极片中文字幕| 97香蕉碰碰人妻国产樱花| 99人妻视频免费在线| 少妇人妻二三区视频 | 天干天天天色天天日天天射| 又色又爽又黄又刺激av网站| 成人av久久精品一区二区| 激情图片日韩欧美人妻| 精品亚洲中文字幕av| lutube在线成人免费看| 亚洲色偷偷综合亚洲AV伊人| 97人妻色免费视频| 五十路老熟女码av| 天堂av狠狠操蜜桃| 中文字幕1卡1区2区3区| 欧美偷拍亚洲一区二区| 一区二区三区激情在线| 午夜激情精品福利视频| 久久这里只有精品热视频| 国产精彩对白一区二区三区| 最新黄色av网站在线观看| 黑人变态深video特大巨大| 家庭女教师中文字幕在线播放| 91精品国产黑色丝袜| 狍和女人的王色毛片| 日韩中文字幕精品淫| 天天日天天鲁天天操| 老鸭窝在线观看一区| 国产污污污污网站在线| 欧美日韩人妻久久精品高清国产| 五十路丰满人妻熟妇| av完全免费在线观看av| 国产精品成久久久久三级蜜臀av| 97人人妻人人澡人人爽人人精品| av中文字幕电影在线看| 久久精品在线观看一区二区| 日韩一区二区三区三州| 男女啪啪视频免费在线观看 | 国产品国产三级国产普通话三级| 男人操女人的逼免费视频| 青娱乐最新视频在线| 夜夜嗨av一区二区三区中文字幕| 天天艹天天干天天操| 骚货自慰被发现爆操| 岛国青草视频在线观看| okirakuhuhu在线观看| 久久久久久性虐视频| 夫妻在线观看视频91| 日韩美av高清在线| 国产麻豆精品人妻av| 亚洲公开视频在线观看| 狠狠躁夜夜躁人人爽天天天天97| 久久亚洲天堂中文对白| 在线观看黄色成年人网站| 亚洲福利精品福利精品福利| 综合页自拍视频在线播放| 夜夜骑夜夜操夜夜奸| 午夜精品福利一区二区三区p| 亚洲欧美清纯唯美另类 | aaa久久久久久久久| 亚洲一级 片内射视正片| 无码日韩人妻精品久久| 日本午夜福利免费视频| 久久久精品999精品日本| 中文字幕在线视频一区二区三区| 巨乳人妻日下部加奈被邻居中出 | 少妇露脸深喉口爆吞精| 免费看国产又粗又猛又爽又黄视频| 操的小逼流水的文章| 午夜精品福利一区二区三区p | 中文字幕一区二区自拍| av在线播放国产不卡| 在线观看视频网站麻豆| 亚洲成人av一区在线| 国产视频在线视频播放| 9色在线视频免费观看| 91在线免费观看成人| 99精品免费久久久久久久久a| 日韩欧美国产精品91| 美日韩在线视频免费看| 91极品新人『兔兔』精品新作| 操日韩美女视频在线免费看 | 99国内精品永久免费视频| 青青在线视频性感少妇和隔壁黑丝| 很黄很污很色的午夜网站在线观看| 亚洲一区二区三区在线高清| 亚洲免费在线视频网站| 亚洲国产精品黑丝美女| 国产乱子伦一二三区| 99热久久极品热亚洲| 2020国产在线不卡视频| 97国产在线av精品| 初美沙希中文字幕在线| 亚洲av成人免费网站| av中文字幕在线观看第三页| 日韩av中文在线免费观看| av资源中文字幕在线观看| 亚洲成人线上免费视频观看| 超碰在线观看免费在线观看| 91人妻精品一区二区在线看| 偷拍自拍视频图片免费| 人妻少妇亚洲一区二区| 国产成人精品亚洲男人的天堂| 青青草原网站在线观看| 丝袜亚洲另类欧美变态| 亚洲中文字幕综合小综合| 大香蕉大香蕉大香蕉大香蕉大香蕉| 中文字幕日韩精品就在这里| 动漫精品视频在线观看| 国产精品一区二区av国| 五月精品丁香久久久久福利社| 91色老99久久九九爱精品| 亚洲国产精品久久久久久6| 三级av中文字幕在线观看| 手机看片福利盒子日韩在线播放| 午夜毛片不卡免费观看视频| 国产精品黄页网站视频| 午夜精品九一唐人麻豆嫩草成人| av在线资源中文字幕| aⅴ精产国品一二三产品| 午夜精品亚洲精品五月色| 国产精品3p和黑人大战| 青娱乐最新视频在线| 三上悠亚和黑人665番号| 亚洲综合在线视频可播放| 久草视频在线免播放| 蜜桃色婷婷久久久福利在线| 日韩中文字幕在线播放第二页| 亚洲精品在线资源站| 99一区二区在线观看| 很黄很污很色的午夜网站在线观看 | 亚洲另类综合一区小说| 一区二区三区毛片国产一区| 亚洲精品国产综合久久久久久久久 | 天天日天天舔天天射进去| 国产视频一区二区午夜| 日本啪啪啪啪啪啪啪| 欧美亚洲国产成人免费在线| 中文字幕之无码色多多| 97精品成人一区二区三区| 黑人3p华裔熟女普通话| 播放日本一区二区三区电影| 青青青青操在线观看免费| www骚国产精品视频| 日本性感美女三级视频| www骚国产精品视频| 婷婷久久一区二区字幕网址你懂得 | 一区二区三区麻豆福利视频| 自拍偷拍,中文字幕| 久久久超爽一二三av| 亚洲一区制服丝袜美腿| 天天干天天操天天摸天天射| 亚洲嫩模一区二区三区| 91九色国产porny蝌蚪| 日本成人不卡一区二区| 精品视频一区二区三区四区五区| 国产伦精品一区二区三区竹菊| 久久久麻豆精亚洲av麻花| 国产亚洲欧美另类在线观看| 深田咏美亚洲一区二区| 最近中文2019年在线看| 在线视频这里只有精品自拍| 亚洲欧美一卡二卡三卡| 91国内视频在线观看| 亚洲另类图片蜜臀av| 2019av在线视频| av中文字幕电影在线看| 边摸边做超爽毛片18禁色戒 | 亚欧在线视频你懂的| 激情内射在线免费观看| 亚洲熟色妇av日韩熟色妇在线| 国产精品黄大片在线播放| 岛国毛片视频免费在线观看| 国产精品国产三级国产精东| 人妻熟女在线一区二区| 亚洲成人三级在线播放| 最新97国产在线视频| 天天日天天爽天天爽| 18禁污污污app下载| 国产熟妇乱妇熟色T区| 久久艹在线观看视频| 91精品一区二区三区站长推荐| 久草视频在线免播放| 国产无遮挡裸体免费直播视频| 国产在线一区二区三区麻酥酥| 毛片av在线免费看| 国产成人精品午夜福利训2021| 欧美美女人体视频一区| 国产高清精品极品美女| 日韩一个色综合导航| 一个人免费在线观看ww视频| 91国内精品自线在拍白富美| 啊啊好慢点插舔我逼啊啊啊视频| 亚洲欧美激情人妻偷拍| 国产午夜激情福利小视频在线| 一区二区三区美女毛片| 国产精品自拍在线视频| 亚洲变态另类色图天堂网| 88成人免费av网站| sw137 中文字幕 在线| 中文字幕人妻av在线观看| 久久久久91精品推荐99| 日韩剧情片电影在线收看| 97少妇精品在线观看| 日韩一区二区电国产精品| 韩国三级aaaaa高清视频| 亚洲av成人网在线观看| 国产精品久久久久久久久福交| 免费黄页网站4188| 免费av岛国天堂网站| 中国老熟女偷拍第一页| 久久人人做人人妻人人玩精品vr| 久久三久久三久久三久久| 国产精彩对白一区二区三区| 大陆av手机在线观看| 中文字幕亚洲中文字幕| 人人人妻人人澡人人| 色哟哟在线网站入口| 最新国产亚洲精品中文在线| 精品成人啪啪18免费蜜臀| 久久人人做人人妻人人玩精品vr| rct470中文字幕在线| 影音先锋女人av噜噜色| 欧美乱妇无乱码一区二区| 韩国爱爱视频中文字幕| www,久久久,com| 久草电影免费在线观看| 国产普通话插插视频| 亚洲精品久久视频婷婷| 红杏久久av人妻一区| 天天日天天干天天要| 亚洲国产欧美国产综合在线| 超碰公开大香蕉97| 狠狠嗨日韩综合久久| 国产大鸡巴大鸡巴操小骚逼小骚逼 | 午夜婷婷在线观看视频| 青春草视频在线免费播放| 亚洲一区二区激情在线| 一个人免费在线观看ww视频 | 91麻豆精品久久久久| 亚洲男人在线天堂网| 亚洲国产最大av综合| 绝色少妇高潮3在线观看| 国产熟妇人妻ⅹxxxx麻豆| 天天想要天天操天天干| 亚洲视频在线观看高清| 粉嫩av懂色av蜜臀av| 日韩特级黄片高清在线看| 18禁免费av网站| 摧残蹂躏av一二三区| 老熟妇xxxhd老熟女| 久久久制服丝袜中文字幕| 丝袜美腿视频诱惑亚洲无| 欧美80老妇人性视频| 欧美色婷婷综合在线| 激情五月婷婷综合色啪| 亚洲精品成人网久久久久久小说| 色伦色伦777国产精品| 青青青视频手机在线观看| 超级碰碰在线视频免费观看| 色天天天天射天天舔| 99精品国自产在线人| 2018在线福利视频| 91久久精品色伊人6882| 不卡精品视频在线观看| 国产亚洲视频在线观看| 亚洲午夜电影在线观看| 成年人免费看在线视频| 国产性生活中老年人视频网站| 成人av天堂丝袜在线观看| 国产美女午夜福利久久| 美女视频福利免费看| 日本黄在免费看视频| 91老师蜜桃臀大屁股| 黄色在线观看免费观看在线| 日韩欧美一级精品在线观看| 天天干天天啪天天舔| 亚洲男人的天堂a在线| 青娱乐在线免费视频盛宴| 亚洲av香蕉一区区二区三区犇| 天天射,天天操,天天说| 亚洲女人的天堂av| 93视频一区二区三区| 亚洲老熟妇日本老妇| av老司机精品在线观看| 五月激情婷婷久久综合网| 中文字幕一区二区亚洲一区| 最新激情中文字幕视频| 人人超碰国字幕观看97| 国产精品人妻一区二区三区网站| 日视频免费在线观看| 国产性生活中老年人视频网站| 亚洲一区二区三区久久受| 亚洲激情唯美亚洲激情图片| 天美传媒mv视频在线观看| av手机在线免费观看日韩av| 99热碰碰热精品a中文| 婷婷色中文亚洲网68| 综合激情网激情五月五月婷婷| 91试看福利一分钟| 端庄人妻堕落挣扎沉沦| 久草视频在线看免费| 欧美色呦呦最新网址| 国产精品人妻熟女毛片av久| 一级黄色片夫妻性生活| 国产又粗又硬又猛的毛片视频| 精品一区二区三区三区88| 天天日天天干天天舔天天射| 久久亚洲天堂中文对白| 亚洲人人妻一区二区三区| 大肉大捧一进一出好爽在线视频| 日韩三级黄色片网站| 欧美黄片精彩在线免费观看| 绝顶痉挛大潮喷高潮无码| 久久99久久99精品影院| 激情人妻校园春色亚洲欧美| 亚洲精品国品乱码久久久久| 亚洲男人让女人爽的视频| 亚洲熟妇无码一区二区三区| 成熟丰满熟妇高潮xx×xx| 国产欧美精品一区二区高清| 夜色17s精品人妻熟女| 亚洲自拍偷拍精品网| 亚洲精品中文字幕下载| 亚洲超碰97人人做人人爱| 亚洲福利午夜久久久精品电影网| 亚洲另类在线免费观看| 人妻丝袜av在线播放网址| 国产黄色a级三级三级三级| 久久这里只有精品热视频| 人妻凌辱欧美丰满熟妇| 欧美精品免费aaaaaa| 中字幕人妻熟女人妻a62v网| 1区2区3区4区视频在线观看| 福利视频广场一区二区| 中文字幕人妻熟女在线电影| 国产精品视频资源在线播放| 99re国产在线精品| 最近的中文字幕在线mv视频| 中文字幕 人妻精品| 日本午夜久久女同精女女| 韩国三级aaaaa高清视频 | 中文字幕av一区在线观看| 视频在线免费观看你懂得| 男女第一次视频在线观看| 亚洲最大黄了色网站| 日噜噜噜夜夜噜噜噜天天噜噜噜| 亚洲 人妻 激情 中文| 淫秽激情视频免费观看| 五十路丰满人妻熟妇| 美女 午夜 在线视频| 色婷婷久久久久swag精品| 国产精彩福利精品视频| 亚洲欧美综合在线探花| 日本少妇在线视频大香蕉在线观看| 国产一区二区神马久久| 啪啪啪啪啪啪啪啪av| 国产成人自拍视频播放| 黄色黄色黄片78在线| www久久久久久久久久久| 最新欧美一二三视频| 久久尻中国美女视频| 中文字幕乱码av资源| 老司机在线精品福利视频| 最新中文字幕乱码在线| 丝袜长腿第一页在线| 国产va在线观看精品| 欧美韩国日本国产亚洲| 久久久麻豆精亚洲av麻花| 日日操综合成人av| 精品一区二区三区欧美| 九色视频在线观看免费| 大尺度激情四射网站| 水蜜桃一区二区三区在线观看视频| 激情人妻校园春色亚洲欧美| 福利午夜视频在线观看| 丁香花免费在线观看中文字幕| 熟女人妻在线观看视频| 日本啪啪啪啪啪啪啪| 93视频一区二区三区| 最近的中文字幕在线mv视频| 美女福利视频导航网站 | 一个人免费在线观看ww视频| 国产视频一区在线观看| 韩国女主播精品视频网站| 福利一二三在线视频观看| 日韩美女搞黄视频免费| 粉嫩小穴流水视频在线观看| 把腿张开让我插进去视频| 亚洲另类图片蜜臀av| 久久久久五月天丁香社区| 久久国产精品精品美女| 97人人模人人爽人人喊| 一区二区三区四区中文| 无码国产精品一区二区高潮久久4| 538精品在线观看视频| jiuse91九色视频| 日韩亚洲高清在线观看| 93精品视频在线观看| 在线观看操大逼视频| 沙月文乃人妻侵犯中文字幕在线| 男人操女人逼逼视频网站| 国产午夜无码福利在线看| 午夜成午夜成年片在线观看 | 亚洲 清纯 国产com| 国产va在线观看精品| 亚洲精品 日韩电影| 成年人该看的视频黄免费| 国产精品久久9999| 亚洲一区二区久久久人妻| 亚洲综合一区成人在线| 国产精彩福利精品视频| 国产精品久久久久久久女人18| 91社福利《在线观看| 午夜蜜桃一区二区三区| 天天日天天添天天爽| 精品区一区二区三区四区人妻| 欧美日韩熟女一区二区三区| 亚洲欧美成人综合在线观看| 国产密臀av一区二区三| 五十路熟女人妻一区二| 青青热久免费精品视频在线观看| 国产熟妇一区二区三区av | 国产精品亚洲在线观看| 美女视频福利免费看| 天天摸天天干天天操科普| 在线播放国产黄色av| 成人高清在线观看视频| 中文字幕日韩精品就在这里| 欧美色呦呦最新网址| 五十路息与子猛烈交尾视频| 成人av亚洲一区二区| 黑人3p华裔熟女普通话| 亚洲一级美女啪啪啪| 欧美精品免费aaaaaa| 国产欧美日韩在线观看不卡| 国产综合精品久久久久蜜臀| 天天操夜夜操天天操天天操 | 中文字幕午夜免费福利视频| 家庭女教师中文字幕在线播放| 美女视频福利免费看| 国产福利小视频免费观看| 国产普通话插插视频| 91国偷自产一区二区三区精品| 亚洲 中文字幕在线 日韩| 亚洲 欧美 精品 激情 偷拍 | 亚洲另类伦春色综合小| 国产成人精品久久二区91| 北条麻妃av在线免费观看| 亚洲av自拍偷拍综合| 97人妻无码AV碰碰视频| 88成人免费av网站| 精品乱子伦一区二区三区免费播| 中文字幕1卡1区2区3区| 春色激情网欧美成人| 99精品国自产在线人| 97国产在线av精品| 精品91自产拍在线观看一区| 欧美精品伦理三区四区| 日本美女成人在线视频| 人人人妻人人澡人人| 鸡巴操逼一级黄色气| 国产精品大陆在线2019不卡 | 日噜噜噜夜夜噜噜噜天天噜噜噜| 中文字幕人妻三级在线观看| 亚洲熟色妇av日韩熟色妇在线| 亚洲国产在人线放午夜| 国产精品久久久久久久久福交| 天天干天天日天天干天天操| 一区二区三区的久久的蜜桃的视频 | 亚洲精品国产久久久久久| 亚洲精品色在线观看视频| 五十路熟女av天堂| 亚洲的电影一区二区三区| 在线观看操大逼视频| 成人综合亚洲欧美一区| 中文字幕日韩无敌亚洲精品| 亚洲高清国产拍青青草原| 红杏久久av人妻一区| 偷拍自拍 中文字幕| 亚洲欧美福利在线观看| 亚洲成a人片777777| 岛国青草视频在线观看| 丝袜肉丝一区二区三区四区在线看| 久久久久久久一区二区三| 黑人3p华裔熟女普通话| 亚洲第17页国产精品| 欧美男人大鸡吧插女人视频| 黄片色呦呦视频免费看| 做爰视频毛片下载蜜桃视频1| av森泽佳奈在线观看| 亚洲一区二区三区在线高清| 一级黄色片夫妻性生活| 天天插天天色天天日| 热99re69精品8在线播放| 97黄网站在线观看| 91天堂精品一区二区| 大鸡吧插入女阴道黄色片| 第一福利视频在线观看| av老司机精品在线观看| 亚洲综合一区二区精品久久| 亚洲成人午夜电影在线观看| 久久一区二区三区人妻欧美| 国产极品精品免费视频| 国产高潮无码喷水AV片在线观看| 91超碰青青中文字幕| 美女骚逼日出水来了| 国产又大又黄免费观看| 一区二区三区久久中文字幕| 姐姐的朋友2在线观看中文字幕| 日韩在线中文字幕色| 日韩精品电影亚洲一区| 欧美日韩激情啪啪啪| 99精品免费久久久久久久久a| xxx日本hd高清| 爱有来生高清在线中文字幕| 98视频精品在线观看| 亚洲高清自偷揄拍自拍| 天天干天天操天天扣| 亚洲综合图片20p| 中文字幕高清在线免费播放| 亚洲嫩模一区二区三区| 国产精彩对白一区二区三区| 欧美黄色录像免费看的| 日韩亚国产欧美三级涩爱| 91精品视频在线观看免费| 国产精品国产精品一区二区| 国产又粗又硬又大视频| 天堂中文字幕翔田av| 91九色国产porny蝌蚪| 亚洲精品国品乱码久久久久| 初美沙希中文字幕在线| 福利午夜视频在线观看| 激情色图一区二区三区| 韩国亚洲欧美超一级在线播放视频| 亚洲熟妇无码一区二区三区| 动色av一区二区三区| 在线免费91激情四射 | 91麻豆精品久久久久| 五月精品丁香久久久久福利社| 最新黄色av网站在线观看| 偷拍自拍福利视频在线观看| 青青青青视频在线播放| 2020中文字幕在线播放| 欧美一级色视频美日韩| 青青青国产免费视频| 888亚洲欧美国产va在线播放| 五十路熟女人妻一区二| 久草视频福利在线首页| 亚洲国产成人最新资源| 国产在线拍揄自揄视频网站| 人妻3p真实偷拍一二区| 美女吃鸡巴操逼高潮视频| 国产剧情演绎系列丝袜高跟| 一区二区久久成人网| 人妻无码中文字幕专区| 啊啊好大好爽啊啊操我啊啊视频 | 1区2区3区4区视频在线观看| 天天干天天操天天插天天日| 直接能看的国产av| 9久在线视频只有精品| 少妇ww搡性bbb91| 天天爽夜夜爽人人爽QC| 又粗又硬又猛又黄免费30| 哥哥姐姐综合激情小说| 少妇人妻二三区视频| 日本免费午夜视频网站| 91免费放福利在线观看| 风流唐伯虎电视剧在线观看| 欧美xxx成人在线| 国产欧美日韩在线观看不卡| 日本成人不卡一区二区| 欧美一区二区三区激情啪啪啪| av线天堂在线观看| 亚洲国产美女一区二区三区软件| h国产小视频福利在线观看| 亚洲精品无码色午夜福利理论片| 亚洲av第国产精品| 亚洲图片欧美校园春色| 专门看国产熟妇的网站| 欧美xxx成人在线| 国产无遮挡裸体免费直播视频| 日本熟妇丰满厨房55| 91欧美在线免费观看| 农村胖女人操逼视频| 免费在线看的黄网站| 亚洲av色香蕉一区二区三区| 另类av十亚洲av| 久久艹在线观看视频| 国产免费av一区二区凹凸四季| 天天干天天插天天谢| 免费看美女脱光衣服的视频| 欧美精品国产综合久久| 五月激情婷婷久久综合网| 中国无遮挡白丝袜二区精品| 黑人乱偷人妻中文字幕| 91九色porny蝌蚪国产成人| 亚洲码av无色中文| 秋霞午夜av福利经典影视| 日韩美女精品视频在线观看网站| 97年大学生大白天操逼| 78色精品一区二区三区| 国产精品探花熟女在线观看| 日韩在线中文字幕色| 日本美女性生活一级片| 哥哥姐姐综合激情小说| 久久久久五月天丁香社区 | 国产一级麻豆精品免费| 国产自拍黄片在线观看| 欧美美女人体视频一区| 91精品一区二区三区站长推荐| 57pao国产一区二区| 中文字幕在线视频一区二区三区| 亚洲一区二区三区偷拍女厕91| 超碰97人人澡人人| 青青青青青青草国产| 日韩亚洲高清在线观看| 日本美女性生活一级片| 亚洲最大黄 嗯色 操 啊| 国产成人精品午夜福利训2021| 国产精品免费不卡av| 男生舔女生逼逼的视频| 视频啪啪啪免费观看| 好太好爽好想要免费| 一区二区久久成人网| 11久久久久久久久久久| 成人性爱在线看四区| 亚洲精品国品乱码久久久久 | 久久丁香婷婷六月天| 大屁股肉感人妻中文字幕在线| 经典av尤物一区二区| 国产使劲操在线播放| 五色婷婷综合狠狠爱| 经典亚洲伊人第一页| 亚洲在线一区二区欧美| 国产精品熟女久久久久浪潮| 欧美日韩在线精品一区二区三| 在线国产精品一区二区三区| 日韩北条麻妃一区在线| 成人资源在线观看免费官网| 国产激情av网站在线观看| 黄色成年网站午夜在线观看 | 亚洲欧美国产综合777| 亚洲一级美女啪啪啪| 最新日韩av传媒在线| 天天躁夜夜躁日日躁a麻豆| 久久农村老妇乱69系列| 精品国产污污免费网站入口自| 午夜毛片不卡在线看| japanese五十路熟女熟妇| 国产黄色高清资源在线免费观看| 天天日天天日天天射天天干| 日韩欧美国产一区不卡| 欧美精品免费aaaaaa| 99的爱精品免费视频| 亚洲欧美福利在线观看| 日本少妇精品免费视频| 国产三级影院在线观看| 久久久制服丝袜中文字幕| 亚洲天天干 夜夜操| 小穴多水久久精品免费看| 无忧传媒在线观看视频| 午夜毛片不卡在线看| 岳太深了紧紧的中文字幕| 久久久久久性虐视频| 男女之间激情网午夜在线| 亚洲精品在线资源站| 大黑人性xxxxbbbb| 久久这里只有精品热视频| 狠狠地躁夜夜躁日日躁| 自拍偷拍,中文字幕| 精品国产成人亚洲午夜| 99热这里只有精品中文| 精品日产卡一卡二卡国色天香| 日韩一区二区电国产精品| 人妻少妇精品久久久久久| 成人性黑人一级av| 久草极品美女视频在线观看| 2012中文字幕在线高清| 97青青青手机在线视频 | 最近的中文字幕在线mv视频| 欧美国产亚洲中英文字幕| 天天日天天玩天天摸| 18禁网站一区二区三区四区 | 99精品国产免费久久| 天天日天天舔天天射进去| 午夜国产福利在线观看| 91精品国产91青青碰| 婷婷五月亚洲综合在线| 亚洲精品 欧美日韩| 国产女人露脸高潮对白视频| 超黄超污网站在线观看| 亚洲av成人免费网站| 中国产一级黄片免费视频播放| 国产视频一区在线观看| 天天日天天日天天擦| 成人蜜桃美臀九一一区二区三区| 日韩成人综艺在线播放| 亚洲va国产va欧美精品88| 午夜精品福利91av| 91老师蜜桃臀大屁股| 色婷婷久久久久swag精品| 天天操天天干天天日狠狠插 | 国产又粗又黄又硬又爽| 日本在线不卡免费视频| 国产精品久久综合久久| 91中文字幕免费在线观看| 国产精品国产三级国产精东 | 午夜精品久久久久久99热| 日本在线一区二区不卡视频| 一区二区三区另类在线| 人妻av无码专区久久绿巨人| 日韩一个色综合导航| 欧美亚洲一二三区蜜臀| av中文在线天堂精品| 91精品国产91青青碰| 3D动漫精品啪啪一区二区下载| 自拍偷拍日韩欧美亚洲| 91麻豆精品91久久久久同性| 视频一区二区在线免费播放| 色哟哟在线网站入口| 亚洲伊人色一综合网| 91精品国产综合久久久蜜| 欧美性受xx黑人性猛交| 一级黄片大鸡巴插入美女| 男女之间激情网午夜在线| 精品久久婷婷免费视频| 亚洲欧美成人综合视频| 一区二区三区日本伦理| 亚洲欧美国产麻豆综合| 国产精品人妻66p| 亚洲精品 欧美日韩| 日韩影片一区二区三区不卡免费 | 欧美精品一区二区三区xxxx| 色偷偷伊人大杳蕉综合网| 国产精品成久久久久三级蜜臀av| 国产男女视频在线播放| 大屁股熟女一区二区三区| 一级a看免费观看网站| 在线免费观看国产精品黄色| 亚洲免费国产在线日韩| 福利视频一区二区三区筱慧| 男人操女人的逼免费视频| 日韩精品激情在线观看| 男生用鸡操女生视频动漫| h国产小视频福利在线观看| 天天干天天操天天摸天天射| 国产美女午夜福利久久| 日韩欧美高清免费在线| 中文字幕中文字幕人妻| 亚洲国产中文字幕啊啊啊不行了| 人妻少妇一区二区三区蜜桃| 天天想要天天操天天干| 国产九色91在线观看精品| 精品首页在线观看视频| 在线新三级黄伊人网| 欧美美女人体视频一区| rct470中文字幕在线| 少妇人妻真实精品视频| 天天干天天日天天谢综合156| 午夜免费体验区在线观看| 欧美一区二区三区在线资源 | 国产精品久久久黄网站| 777奇米久久精品一区| 丰满少妇人妻xxxxx| 欧美日韩一区二区电影在线观看| 久久久久久久久久性潮| 日本高清在线不卡一区二区| av网址国产在线观看| 激情国产小视频在线| 欧美在线一二三视频| 久久丁香花五月天色婷婷| 亚洲成人熟妇一区二区三区| 国产一区二区久久久裸臀| 国产片免费观看在线观看| 中文字幕亚洲久久久| 超碰97免费人妻麻豆| 国产黄色高清资源在线免费观看| 在线播放一区二区三区Av无码 | 亚洲一区二区久久久人妻| 黄色黄色黄片78在线| 日韩精品中文字幕播放| 国产老熟女伦老熟妇ⅹ| 成人免费公开视频无毒| 97a片免费在线观看| 亚洲中文字幕综合小综合| 天天综合天天综合天天网| 人人妻人人人操人人人爽| 天天日天天做天天日天天做| 干逼又爽又黄又免费的视频| 国产性色生活片毛片春晓精品 | 2020久久躁狠狠躁夜夜躁| 偷拍自拍福利视频在线观看| 啪啪啪啪啪啪啪啪啪啪黄色| 黑人巨大的吊bdsm| 久精品人妻一区二区三区| 涩爱综合久久五月蜜臀| 91在线免费观看成人| 93人妻人人揉人人澡人人| av中文字幕电影在线看| 久草视频在线免播放| 国产精彩福利精品视频| 亚洲黄色av网站免费播放| 天天干狠狠干天天操| 国产黄色大片在线免费播放| 色综合色综合色综合色| 欧美色呦呦最新网址| 99婷婷在线观看视频| 97精品成人一区二区三区 | 午夜精品九一唐人麻豆嫩草成人| 加勒比视频在线免费观看| av在线观看网址av| 高潮喷水在线视频观看| 97国产在线av精品| 亚洲福利精品福利精品福利| 亚洲午夜高清在线观看| 国产亚洲精品视频合集| 免费在线观看污污视频网站| 91麻豆精品传媒国产黄色片| 日本少妇的秘密免费视频| 少妇与子乱在线观看| 18禁免费av网站| 搡老妇人老女人老熟女| 在线观看操大逼视频| 国产亚洲视频在线二区| 老鸭窝日韩精品视频观看| 美女av色播在线播放| 东京干手机福利视频| 性感美女福利视频网站| 日本黄色特一级视频| 一区二区三区 自拍偷拍| 久久久久久久亚洲午夜综合福利| 伊人成人在线综合网| 91社福利《在线观看| 国产精品中文av在线播放| 91麻豆精品久久久久| 欧美爆乳肉感大码在线观看 | 精品suv一区二区69| 懂色av之国产精品| 久久精品国产亚洲精品166m| 中文字幕午夜免费福利视频| 91久久综合男人天堂| 三级av中文字幕在线观看| 懂色av之国产精品| 中文字母永久播放1区2区3区| 51精品视频免费在线观看| 精品成人啪啪18免费蜜臀| 欧洲精品第一页欧洲精品亚洲| 一区二区视频视频视频| 美女日逼视频免费观看| 亚洲av第国产精品| 青青青青青青青在线播放视频| 亚洲精品久久综合久| 5528327男人天堂| 亚洲一区二区三区精品乱码| 福利一二三在线视频观看| 亚洲一区二区人妻av| 51国产偷自视频在线播放| 亚洲 中文 自拍 另类 欧美| 五十路息与子猛烈交尾视频| 午夜久久香蕉电影网| 一区二区三区日本伦理| 欧美日韩激情啪啪啪| 综合精品久久久久97| 日视频免费在线观看| 午夜精品一区二区三区更新| 2022天天干天天操| 欧美偷拍自拍色图片| 日本韩国免费一区二区三区视频| 国产黄色大片在线免费播放| 日本脱亚入欧是指什么| 狠狠操操操操操操操操操| 激情人妻校园春色亚洲欧美| 和邻居少妇愉情中文字幕| 偷拍自拍 中文字幕| 66久久久久久久久久久| 大屁股熟女一区二区三区| 日美女屁股黄邑视频| 亚洲图片偷拍自拍区| 欧亚乱色一区二区三区| 天天射夜夜操狠狠干| 青青青aaaa免费| 日本特级片中文字幕| 93人妻人人揉人人澡人人| 夫妻在线观看视频91| 中文字幕欧美日韩射射一| 成人24小时免费视频| 国产精品亚洲在线观看| 日韩美女福利视频网| 黑人乱偷人妻中文字幕| av破解版在线观看| 亚洲福利精品视频在线免费观看| 日本女大学生的黄色小视频| 国产福利在线视频一区| 欧美一级视频一区二区| 欧美中国日韩久久精品| 欧美黑人性暴力猛交喷水| 中国熟女@视频91| 亚洲另类伦春色综合小| 韩国黄色一级二级三级| 中文字幕一区二区亚洲一区| 男人在床上插女人视频| 91综合久久亚洲综合| AV无码一区二区三区不卡| 在线观看的黄色免费网站| 2012中文字幕在线高清| 亚洲国产美女一区二区三区软件| 熟女人妻在线中出观看完整版| 日本xx片在线观看| 晚上一个人看操B片| 又大又湿又爽又紧A视频| 五月色婷婷综合开心网4438| 亚洲av日韩av第一区二区三区| 99国产精品窥熟女精品| 中文字幕一区二区三区人妻大片| 中文字幕日韩精品就在这里| 97成人免费在线观看网站| 中文字幕乱码人妻电影| av中文字幕网址在线| 国产麻豆乱子伦午夜视频观看| 午夜精品九一唐人麻豆嫩草成人 | 日本www中文字幕| av无限看熟女人妻另类av| 久精品人妻一区二区三区| 日美女屁股黄邑视频| 韩国男女黄色在线观看| 中文字幕之无码色多多| 国产第一美女一区二区三区四区| 青青青青青手机视频| 亚洲一级特黄特黄黄色录像片| 中文字幕人妻熟女在线电影| 99亚洲美女一区二区三区| 天天操天天射天天操天天天| 91人妻精品久久久久久久网站| 人妻少妇精品久久久久久| 亚洲精品无码色午夜福利理论片| 中字幕人妻熟女人妻a62v网| 欧美80老妇人性视频| 免费69视频在线看| 中文字幕无码日韩专区免费| brazzers欧熟精品系列| 欧美麻豆av在线播放| 亚洲午夜电影之麻豆 | 五十路息与子猛烈交尾视频| 亚洲精品麻豆免费在线观看| 亚洲一区二区激情在线| 欧美国产亚洲中英文字幕| 日本免费午夜视频网站| 中文字幕成人日韩欧美| 性感美女诱惑福利视频| 大陆av手机在线观看| 亚洲av日韩高清hd| 日本熟妇色熟妇在线观看| 久久丁香花五月天色婷婷| 午夜精彩视频免费一区| 青青青青青免费视频| 天天操天天干天天艹| 亚洲激情唯美亚洲激情图片| 日本男女操逼视频免费看| 97精品视频在线观看| 国产日本精品久久久久久久| 国产精品视频资源在线播放| 国内精品在线播放第一页| 日韩加勒比东京热二区| 北条麻妃av在线免费观看| 午夜精品一区二区三区福利视频| 日视频免费在线观看| 亚洲av日韩精品久久久| 黑人借宿ntr人妻的沦陷2| 91在线免费观看成人| 摧残蹂躏av一二三区| 久久香蕉国产免费天天| 亚洲欧美在线视频第一页| 欧美日韩v中文在线| 黄色成年网站午夜在线观看| 一本一本久久a久久精品综合不卡| 99热久久这里只有精品| 99精品视频在线观看婷婷| 中文字幕乱码人妻电影| 在线免费91激情四射| 巨乳人妻日下部加奈被邻居中出| 自拍偷拍亚洲欧美在线视频| 蜜桃视频在线欧美一区| 青娱乐最新视频在线| 一色桃子人妻一区二区三区| 亚欧在线视频你懂的| 久久香蕉国产免费天天| 天天干天天日天天谢综合156| 亚洲 欧美 自拍 偷拍 在线| 亚洲第一伊人天堂网| 亚洲福利天堂久久久久久 | 国产女人被做到高潮免费视频| 午夜激情精品福利视频| 日韩一个色综合导航| 特一级特级黄色网片| 中文字幕在线欧美精品| 午夜精品一区二区三区更新| 专门看国产熟妇的网站| 日韩美av高清在线| 亚洲免费在线视频网站| 骚逼被大屌狂草视频免费看| 99的爱精品免费视频| 欧美va亚洲va天堂va| 亚洲国产免费av一区二区三区 | 日日夜夜大香蕉伊人| 任我爽精品视频在线播放| 最近的中文字幕在线mv视频| 内射久久久久综合网| 国产综合精品久久久久蜜臀| 午夜av一区二区三区| 亚洲欧美一区二区三区爱爱动图| 国产精品入口麻豆啊啊啊| 91精品国产高清自在线看香蕉网| 国产aⅴ一线在线观看| 最近的中文字幕在线mv视频| 成年美女黄网站18禁久久| 亚洲国产中文字幕啊啊啊不行了| 人妻少妇av在线观看| 最新激情中文字幕视频| 在线观看911精品国产| 最新欧美一二三视频| 中文字幕高清资源站| 11久久久久久久久久久| 欧美日韩精品永久免费网址| 人妻少妇中文有码精品| 91免费黄片可看视频| 18禁无翼鸟成人在线| 日本性感美女视频网站| 日韩欧美国产一区不卡| 快插进小逼里大鸡吧视频| 亚洲中文字字幕乱码| 欧亚日韩一区二区三区观看视频 | 可以免费看的www视频你懂的 | 91亚洲国产成人精品性色| 欧美精品亚洲精品日韩在线| 人妻丝袜诱惑我操她视频| 99热久久极品热亚洲| 色婷婷精品大在线观看| 91天堂精品一区二区| 大香蕉大香蕉在线有码 av| 五月天色婷婷在线观看视频免费| 午夜青青草原网在线观看| 天天想要天天操天天干| 超污视频在线观看污污污| 可以免费看的www视频你懂的| 久草极品美女视频在线观看| 欧美激情精品在线观看| 国产麻豆剧传媒精品国产av蜜桃| 天天插天天色天天日| 天天色天天操天天透| 岛国青草视频在线观看| 亚洲一区二区三区精品视频在线| 日本av熟女在线视频| 日本在线不卡免费视频| 青青在线视频性感少妇和隔壁黑丝 | 欧美精产国品一二三区| 狠狠嗨日韩综合久久| 黄页网视频在线免费观看| 天天干天天操天天扣| av天堂中文字幕最新| 国产视频在线视频播放| 国产乱子伦精品视频潮优女| 最近的中文字幕在线mv视频| 中文字幕一区二区人妻电影冢本| 成人网18免费视频版国产| 老熟妇凹凸淫老妇女av在线观看| 成人30分钟免费视频| 亚洲一区二区三区av网站| 国产精品精品精品999| 在线观看av观看av| 亚洲视频在线观看高清| 自拍偷拍日韩欧美一区二区| 国产实拍勾搭女技师av在线| 中文字幕在线观看国产片| 欧美成一区二区三区四区| 少妇一区二区三区久久久| 女警官打开双腿沦为性奴| 丝袜美腿视频诱惑亚洲无| 97超碰人人搞人人| 亚洲精品午夜久久久久| 超碰在线观看免费在线观看| 国产白袜脚足J棉袜在线观看| 亚洲精品一区二区三区老狼| 日本性感美女写真视频| 日本男女操逼视频免费看| 亚洲精品国品乱码久久久久 | 中文字幕一区二区三区人妻大片| 人人妻人人澡人人爽人人dvl| 男女啪啪啪啪啪的网站| 亚洲午夜电影之麻豆| 91传媒一区二区三区| 97精品综合久久在线| 欧美激情电影免费在线| 天天操天天插天天色| 亚洲一区二区激情在线| 久草视频首页在线观看| 亚洲图片欧美校园春色| 成人av亚洲一区二区| 97国产在线av精品| 亚洲天堂有码中文字幕视频| 骚货自慰被发现爆操| 日韩欧美一级黄片亚洲| 中文亚洲欧美日韩无线码| 亚洲国产精品黑丝美女| 国产综合高清在线观看| 婷婷久久久久深爱网| 亚洲欧美人精品高清| 日日操夜夜撸天天干| 人妻另类专区欧美制服| 在线观看日韩激情视频| 一区二区三区在线视频福利| 久久久久久9999久久久久| 少妇人妻100系列| 亚洲成人情色电影在线观看| 巨乳人妻日下部加奈被邻居中出| 成人蜜臀午夜久久一区| 天天摸天天干天天操科普| 91人妻精品一区二区在线看| 五十路老熟女码av| 亚洲欧美激情中文字幕| 大胆亚洲av日韩av| 中文字幕亚洲久久久| 大香蕉大香蕉在线有码 av| 伊人开心婷婷国产av| 成人免费做爰高潮视频| av手机在线免费观看日韩av| 午夜精品久久久久麻豆影视| 91桃色成人网络在线观看| 国产精品一区二区av国| 在线观看视频 你懂的| 白白操白白色在线免费视频 | 人妻丝袜精品中文字幕| 日本午夜久久女同精女女| 中文字幕亚洲中文字幕| 亚洲精品久久综合久| 51国产成人精品视频| 欧美精品亚洲精品日韩在线| 国产精品精品精品999| 青草亚洲视频在线观看| 国产在线91观看免费观看| 国产又色又刺激在线视频 | 中文字幕第三十八页久久| 偷拍自拍亚洲美腿丝袜| 一区二区在线视频中文字幕| 日韩欧美一级精品在线观看| 熟女人妻在线观看视频| 成人福利视频免费在线| 欧美另类z0z变态| 亚洲精品色在线观看视频| 香蕉av影视在线观看| 免费福利av在线一区二区三区| 一区二区视频视频视频| 成人av天堂丝袜在线观看 | 这里只有精品双飞在线播放| 亚洲偷自拍高清视频| 青春草视频在线免费播放| 欧美亚洲少妇福利视频| 亚洲成人午夜电影在线观看| 亚洲在线一区二区欧美| 美日韩在线视频免费看| 亚洲精品午夜aaa久久| 精品亚洲在线免费观看| 不卡精品视频在线观看| 伊人综合免费在线视频| 婷婷久久久久深爱网| 亚洲成人激情av在线| 91久久精品色伊人6882| 黄色三级网站免费下载| 国产精品久久久久久久久福交| 欧美老妇精品另类不卡片| 欧美交性又色又爽又黄麻豆| 午夜蜜桃一区二区三区| 国产亚洲国产av网站在线| 丝袜美腿视频诱惑亚洲无| 另类av十亚洲av| 日韩欧美在线观看不卡一区二区| 好太好爽好想要免费| 亚洲美女美妇久久字幕组| 亚洲国产精品中文字幕网站| 97超碰最新免费在线观看| 免费成人av中文字幕| 无码中文字幕波多野不卡| 在线视频免费观看网| 天天日天天干天天干天天日| 无套猛戳丰满少妇人妻| 91人妻精品久久久久久久网站 | 亚洲男人在线天堂网| 久久久久久久精品老熟妇| 精品91自产拍在线观看一区| 懂色av之国产精品| 欧美成一区二区三区四区| 国产综合视频在线看片| 老司机午夜精品视频资源| 在线视频精品你懂的| 亚洲视频在线视频看视频在线| 精品人人人妻人人玩日产欧| 亚洲一区二区三区久久受 | 黑人乱偷人妻中文字幕| 亚洲av成人免费网站| 欧美性受xx黑人性猛交| 大鸡吧插逼逼视频免费看| 天天干天天爱天天色| 亚洲国产免费av一区二区三区| 护士小嫩嫩又紧又爽20p| 天天干夜夜操天天舔| 99热这里只有国产精品6| av乱码一区二区三区| 成人性爱在线看四区| 在线观看免费av网址大全| 久久久久久久精品成人热| 国产精品入口麻豆啊啊啊| 精品成人啪啪18免费蜜臀| 国际av大片在线免费观看| 91麻豆精品传媒国产黄色片| 超级碰碰在线视频免费观看| 国产在线免费观看成人| 老司机在线精品福利视频| gav成人免费播放| 福利午夜视频在线合集| 无套猛戳丰满少妇人妻| 久久久久久9999久久久久| 国产一区二区火爆视频| 老司机在线精品福利视频| 日本啪啪啪啪啪啪啪| 超pen在线观看视频公开97| 精品少妇一二三视频在线| 蜜桃专区一区二区在线观看| 亚洲午夜伦理视频在线| 韩国女主播精品视频网站| 亚洲护士一区二区三区| 在线观看免费av网址大全| 久久久精品精品视频视频| 亚洲最大免费在线观看| 国产麻豆乱子伦午夜视频观看| 99精品一区二区三区的区| 亚洲激情,偷拍视频| 2018在线福利视频| 亚洲国产第一页在线观看| 欧美va亚洲va天堂va| 在线可以看的视频你懂的| 欧美亚洲一二三区蜜臀| 五十路人妻熟女av一区二区| h国产小视频福利在线观看| 亚洲av可乐操首页| 成人av亚洲一区二区| 日本精品一区二区三区在线视频。| 超pen在线观看视频公开97| 国产精品黄色的av| 久久人人做人人妻人人玩精品vr| 日韩av有码一区二区三区4| 肏插流水妹子在线乐播下载| av视屏免费在线播放| 美女小视频网站在线| 亚洲1区2区3区精华液| 亚洲综合自拍视频一区| 午夜美女少妇福利视频| rct470中文字幕在线| 熟妇一区二区三区高清版| 玖玖一区二区在线观看| 免费观看成年人视频在线观看| 精品亚洲中文字幕av| 爱爱免费在线观看视频| 婷婷六月天中文字幕| 成人精品视频99第一页| 爱有来生高清在线中文字幕| 精品一区二区三区在线观看| 欧美一区二区三区四区性视频| 中文字幕一区二区自拍| 秋霞午夜av福利经典影视| 视频二区在线视频观看| 激情伦理欧美日韩中文字幕| 日韩美女精品视频在线观看网站| 久久久久只精品国产三级| 中文字幕日韩91人妻在线| 人妻少妇亚洲精品中文字幕| 国产三级精品三级在线不卡| 日韩在线中文字幕色| 在线制服丝袜中文字幕| 中文字幕免费在线免费| 色哟哟在线网站入口| 人妻熟女中文字幕aⅴ在线| 2020中文字幕在线播放| 欧美国产亚洲中英文字幕| 日韩中文字幕福利av| 国产极品美女久久久久久| 天天干天天操天天爽天天摸| 国产一区二区神马久久| 精品一区二区亚洲欧美| 青青青青青手机视频| 美女张开两腿让男人桶av| 搞黄色在线免费观看| 国产午夜福利av导航| 1024久久国产精品| 男人的天堂在线黄色| 55夜色66夜色国产精品站| 粗大的内捧猛烈进出爽大牛汉子| 成年午夜影片国产片| 久久丁香花五月天色婷婷| 最近的中文字幕在线mv视频| 55夜色66夜色国产精品站| 人妻久久久精品69系列| 欧美男同性恋69视频| 黑人进入丰满少妇视频| 亚洲变态另类色图天堂网| 可以免费看的www视频你懂的 | 中文字幕第三十八页久久 | av俺也去在线播放| 亚洲欧洲av天堂综合| 亚洲av午夜免费观看| 国产欧美日韩第三页| 成年人免费看在线视频| 亚洲 国产 成人 在线| 国产精品黄色的av| 91亚洲国产成人精品性色| 999九九久久久精品| 青青青爽视频在线播放| 亚洲av色图18p| 成人亚洲精品国产精品| 抽查舔水白紧大视频| 男人天堂av天天操| 熟女人妻三十路四十路人妻斩| 天天摸天天干天天操科普| 视频一区 视频二区 视频| 成人av天堂丝袜在线观看| 在线免费视频 自拍| 偷拍自拍 中文字幕| 亚洲自拍偷拍综合色| 真实国模和老外性视频| 在线免费视频 自拍| 亚洲中文字幕乱码区| 欧美国产亚洲中英文字幕| 日本少妇精品免费视频| 扒开腿挺进肉嫩小18禁视频| 成人免费公开视频无毒| 97人妻夜夜爽二区欧美极品| 久久久噜噜噜久久熟女av| 78色精品一区二区三区| 污污小视频91在线观看| 精品国产污污免费网站入口自 | 亚洲欧美激情人妻偷拍| 青娱乐极品视频青青草|