精品欧美一区二区三区在线观看 _久久久久国色av免费观看性色_国产精品久久在线观看_亚洲第一综合网站_91精品又粗又猛又爽_小泽玛利亚一区二区免费_91亚洲精品国偷拍自产在线观看 _久久精品视频在线播放_美女精品久久久_欧美日韩国产成人在线

徹底學會 gRPC:用 Go 實現一個迷你考試服務

開發 后端
本文將通過用 Go 編寫一個最簡化的考試服務,一步步演示 gRPC 支持的四種 RPC 形式:一元調用、服務器端流、客戶端流和雙向流。

lack、Netflix 甚至 Kubernetes 這類現代系統,都能高效地完成實時通信。它們龐大的后端被拆分為一個個微服務。那么,這些服務之間到底如何通信?大概率就是 gRPC 在背后發揮魔法。

本文將通過用 Go 編寫一個最簡化的考試服務,一步步演示 gRPC 支持的四種 RPC 形式:一元調用(Unary)、服務器端流(Server Streaming)、客戶端流(Client Streaming)和雙向流(Bidirectional Streaming)。

在此之前,我們先快速掃清一些概念和行話,做好充電。如果你已經很熟悉并且只想要代碼,可以在這里查看相關倉庫:https://github.com/pixperk/grpc_exam。

一、RPC 是啥?

RPC(Remote Procedure Call,遠程過程調用)乍聽高大上,其實就是 “在另一臺機器上調用一個函數”。

想象你是客戶端,要問服務器 “學生 42 的成績是多少?”。借助 RPC,就像本地函數調用一樣簡單,雖然函數真正執行的地方在遠端。

早期 RPC 系統在今天看來問題多多:

  • 數據格式混亂:XML、私有格式,臃腫又慢;
  • 很難流式通信:實時或長鏈接幾乎做不了;
  • 語言綁定有限:經常綁定特定生態(Java RMI、CORBA 等);
  • 自動化差:代碼生成寥寥,樣板代碼海量;
  • 安全自己管:TLS、認證要開發者手寫;
  • 擴展性差:不復用連接、無多路復用,高并發直接跪;

gRPC 則把這些痛點一次性解決:更快、更安全、開發體驗更爽。

二、gRPC — Google 出品的現代化 RPC 框架

gRPC 是 Google 開發的開源 RPC 框架。它旨在讓服務間通信變得更:

  • 快速(得益于 HTTP/2);
  • 類型安全(通過 Protocol Buffers);
  • 流式友好(支持實時通信)。

1. HTTP/2 小科普

gRPC 在底層使用 HTTP/2。與 HTTP/1.1 相比,它具有:

  • 多路復用:在同一連接上并行處理多個請求;
  • 支持雙向流:客戶端和服務器可同時發送和接收數據;
  • 內置頭部壓縮:加快數據傳輸速度。

得益于此,gRPC 能在低延遲下搞定實時通信。

2. gRPC vs REST

特性

REST

gRPC

數據格式

JSON / XML

Protocol Buffers

流式

罕見 / 需自己造輪子

內置

傳輸層

HTTP/1.1

HTTP/2

性能

冗長

緊湊 & 快

開發體驗

手動寫文檔

自動生成代碼

三、gRPC 支持的四種 RPC 通信方式

gRPC 在客戶端與服務器之間支持四種通信方式。下面我們逐一進行介紹。

1. Unary(單次請求-響應)

客戶端發送一次請求 → 服務器返回一次響應。類似普通函數:

rpc GetMarks(StudentRequest) returns (MarksResponse);

這是最常見的類型,非常適合 CRUD 風格的操作。

2. Server Streaming(服務器流)

客戶端一次請求,服務器持續推送多條響應。

rpc StreamSemesterResults(SemesterRequest) returns (stream MarksResponse);

當服務器需要發送大量數據時,這種方式特別有用——結果一準備好就會實時推送給你。

3. Client Streaming(客戶端流)

客戶端持續發送一連串請求 → 服務器最終返回一個匯總響應。也就是說,客戶端批量推送數據,服務器全部處理完畢后一次性給出結果。

rpc UploadAttendance(stream AttendanceEntry) returns (UploadStatus);

非常適合用于發送日志、監控指標或進行批量上傳。

4. Bidirectional Streaming(雙向流)

客戶端和服務器可以同時向對方持續發送數據流。就像實時聊天一樣,雙方能夠一邊說話一邊收聽彼此的消息。

rpc LiveQuiz(stream QuizMessage) returns (stream QuizMessage);

這正是 gRPC 大顯身手的地方:實時多人游戲、協作工具、實時儀表盤——統統不在話下。

四、Protocol Buffers(Protobuf)簡介

Google 推出的跨語言、跨平臺序列化協議。

相較 JSON:

  • 體積更?。憾M制編碼;
  • 速度更快:序列化 / 反序列化省時;
  • 字段用編號:解析無需比對字符串;

Protobuf Buffers 工作流程:

  • 寫 .proto 描述文件;
  • protoc 編譯生成各語言代碼;
  • 在代碼里直接調用序列化 / 反序列化方法;

示例:

syntax = "proto3";

message StudentRequest {
  string student_id = 1;
}

五、動手實踐:Go 版 Exam Service

1. 倉庫初始化

首先,來初始化一個 Go 項目:

mkdir grpc_exam && cd grpc_exam
go mod init github.com/pixperk/grpc_exam

安裝必要的生成插件:

go install google.golang.org/protobuf/cmd/protoc-gen-go@latest
go install google.golang.org/grpc/cmd/protoc-gen-go-grpc@latest

如果你沒有安裝 protoc 編譯工具,可以參考 https://protobuf.dev/installation/ 進行安裝。

接下來,準備好目錄結構:

├── client/
│   ├── clients/
│   │   ├── unary.go
│   │   ├── server_stream.go
│   │   ├── client_stream.go
│   │   ├── bi_stream.go
│   └── main.go
├── proto/
│   ├── exam.proto
│   └── generated/exampb/
│       ├── exam.pb.go
│       └── exam_grpc.pb.go
├── server/
│   ├── main.go
│   └── servers/
│       ├── unary.go
│       ├── server_stream.go
│       ├── client_stream.go
│       ├── bi_stream.go
│       └── exam_service_server.go
├── utils/
│   └── logger.go
├── go.mod
├── go.sum
└── Makefile

接下來,編寫一個 Makefile,避免每次都執行很長的命令來執行操作:

proto:
    protoc \
        --proto_path=proto \
        --go_out=proto \
        --go-grpc_out=proto \
        proto/*.proto
    @echo "Proto files generated in the 'proto' directory."

server:
    go run server/main.go

client_unary:
    go run client/main.go unary

client_server:
    go run client/main.go server

client_client:
    go run client/main.go client

client_bidi:
    go run client/main.go bidi

.PHONY: proto server client_unary client_server client_client client_bidi

make proto 一鍵生成 Go 代碼。

2. 構建 Unary RPC

在 exam.proto 文件中編寫 Exam Service:

syntax = "proto3";

package exam;

option go_package = "generated/exampb";

service ExamService {
rpc GetExamResult(GetExamResultRequest) returns (GetExamResultResponse); //unary
}

message GetExamResultRequest {
string student_id = 1;
string exam_id = 2;
}

message GetExamResultResponse {
string student_name = 1;
string subject = 2;
int32 marks_obtained = 3;
int32 total_marks = 4;
string grade = 5;
}

執行 make proto 生成 Go 代碼,代碼會被放到 proto/generated 目錄。

在 server/servers/exam_service_server.go 中定義:

package servers

import"github.com/pixperk/grpc_exam/proto/generated/exampb"

type ExamServiceServer struct {
    exampb.UnimplementedExamServiceServer
    examData map[string]*exampb.GetExamResultResponse
}

func NewExamServiceServer() *ExamServiceServer {
    data := map[string]*exampb.GetExamResultResponse{
        "123_math101": {
            StudentName:   "John Doe",
            Subject:       "Math 101",
            MarksObtained: 95,
            TotalMarks:    100,
            Grade:         "A+",
        },
        "456_phy101": {
            StudentName:   "Jane Smith",
            Subject:       "Physics 101",
            MarksObtained: 88,
            TotalMarks:    100,
            Grade:         "A",
        },
    }

    return &ExamServiceServer{
        examData: data,
    }
}

接下來設計服務端和客戶端。先寫開發客戶端和服務端的 main.go。

  • server/main.go:
package main

import (
    "net"

    "log/slog"

    "github.com/pixperk/grpc_exam/proto/generated/exampb"
    "github.com/pixperk/grpc_exam/server/servers"

    "github.com/pixperk/grpc_exam/utils"
    "google.golang.org/grpc"
)

func main() {
    utils.InitLogger(true)
    //Spin up a TCP Server
    lis, err := net.Listen("tcp", ":50051")
    if err != nil {
        slog.Error("failed to listen", "error", err)
    }

    //New gRPC server instance
    s := grpc.NewServer()

    //Register services
    exampb.RegisterExamServiceServer(s, servers.NewExamServiceServer())

// Start serving gRPC requests
    if err := s.Serve(lis); err != nil {
        slog.Error("failed to serve", "error", err)
    }

}
  • client/main.go:
package main

import (
    "log/slog"
    "os"

    "github.com/pixperk/grpc_exam/client/clients"
    "github.com/pixperk/grpc_exam/proto/generated/exampb"
    "github.com/pixperk/grpc_exam/utils"
    "google.golang.org/grpc"
    "google.golang.org/grpc/credentials/insecure"
)

func main() {
    // Initialize logger (true = debug mode)
    utils.InitLogger(true)

    // Create a gRPC client connection to the server
    conn, err := grpc.Dial("localhost:50051", grpc.WithTransportCredentials(insecure.NewCredentials()))
    if err != nil {
        slog.Error("Failed to connect to server", "error", err)
        return
    }
    defer conn.Close()

    // Create a client for the ExamService
    client := exampb.NewExamServiceClient(conn)

    clients.Unary(client)
}

3. 編寫 Unary 服務端和客戶端

服務端 (server/servers/unary.go):

package servers

import (
    "context"
    "fmt"

    "github.com/pixperk/grpc_exam/proto/generated/exampb"
)

func (s *ExamServiceServer) GetExamResult(ctx context.Context, req *exampb.GetExamResultRequest) (*exampb.GetExamResultResponse, error) {
    key := fmt.Sprintf("%s_%s", req.StudentId, req.ExamId)
    if result, ok := s.examData[key]; ok {
        return result, nil
    } else {
        returnnil, fmt.Errorf("exam result not found for student ID %s and exam ID %s", req.StudentId, req.ExamId)
    }
}

客戶端 (client/clients/unary.go):

package clients

import (
    "context"
    "fmt"
    "time"

    "github.com/pixperk/grpc_exam/proto/generated/exampb"
)

func Unary(client exampb.ExamServiceClient) {

    fmt.Println("Enter student ID and exam ID (e.g., 123 math101):")
    var studentID, examID string
    fmt.Scanf("%s %s", &studentID, &examID)

    ctx, cancel := context.WithTimeout(context.Background(), 5*time.Second)
    defer cancel()

    resp, err := client.GetExamResult(ctx, &exampb.GetExamResultRequest{StudentId: studentID, ExamId: examID})
    if err != nil {
        fmt.Printf("Error: %v\n", err)
        return
    }

    fmt.Printf("Student Name: %s\n", resp.StudentName)
    fmt.Printf("Subject: %s\n", resp.Subject)
    fmt.Printf("Marks Obtained: %d out of %d\n", resp.MarksObtained, resp.TotalMarks)
    fmt.Printf("Grade: %s\n", resp.Grade)
    fmt.Println("Unary RPC call completed successfully.")

}

上面的代碼,就是一些基礎的函數調用與簡單的 Go 編程。在兩個終端分別運行 make server 和 make client 即可看到效果。

Server Streaming 和 Client Streaming 也很簡單,只要熟悉 Go 流式處理即可。下面直接跳到雙向流(Bidirectional Streaming)。

六、構建雙向 RPC

在 exam.proto 中增加:

rpc LiveExamQuery(stream GetExamResultRequest) returns (stream GetExamResultResponse); //bidi streaming

在 servers/bi_stream.go 文件中開發以下代碼:

package servers

import (
    "fmt"
    "io"

    "github.com/pixperk/grpc_exam/proto/generated/exampb"
)


func (s *ExamServiceServer) LiveExamQuery(stream exampb.ExamService_LiveExamQueryServer) error {
    for {
        // Receive a stream request from the client
        req, err := stream.Recv()
        if err != nil {
            // If the client closes the stream (EOF), stop the loop gracefully
            if err == io.EOF {
                returnnil
            }
            // If another error occurred, return it
            return err
        }

        key := fmt.Sprintf("%s_%s", req.StudentId, req.ExamId)

        result, ok := s.examData[key]

        // If result is not found, send a default "Not Found" response
        if !ok {
            err := stream.Send(&exampb.GetExamResultResponse{
                StudentName:   "N/A",
                Subject:       req.ExamId,
                MarksObtained: 0,
                TotalMarks:    0,
                Grade:         "Not Found",
            })
            if err != nil {
                return err // Stop on send error
            }
            continue
        }

        // If result is found, send it back to the client over the stream
        if err := stream.Send(result); err != nil {
            return err // Stop on send error
        }
    }
}

LiveExamQuery 函數邏輯如下:

  • 不斷從客戶端接收查詢;
  • 立即返回對應結果(若存在);
  • 直到客戶端結束(EOF);

七、Go Channel 簡介

Go Channel 像管道一樣在 goroutine 間傳遞數據,可安全同步,無需顯式 mutex。

  • 創建:done := make(chan struct{})
  • 發送:done <- struct{}{}
  • 接收:<-done

Go Channel 發送或接收前都會阻塞,非常適合協同。

八、回到 bi-dir 客戶端

package clients

import (
    "bufio"
    "context"
    "fmt"
    "io"
    "log"
    "os"
    "strings"

    "github.com/pixperk/grpc_exam/proto/generated/exampb"
)

func BiDirectional(client exampb.ExamServiceClient) {
    //body
}

讓我們逐步來看看這個函數體里都發生了什么:

stream, err := client.LiveExamQuery(context.Background())
done := make(chan struct{})

LiveExamQuery 會與服務器建立一個雙向流。

創建 done 通道是為了在接收方 goroutine 結束時發出信號。

go func() {
        for {
            res, err := stream.Recv() //receive stream from the server
            if err != nil {
                if err == io.EOF {
                    break
                }
                log.Fatalf("Error receiving response: %v", err)
                break
            }
            fmt.Printf("?? %s | %s: %d/%d (%s)\n",
                res.StudentName, res.Subject, res.MarksObtained, res.TotalMarks, res.Grade)

            fmt.Print("Enter student_id and exam_id (or 'exit'): ")
        }
        close(done)

    }()

    // Initial prompt
    fmt.Print("Enter student_id and exam_id (or 'exit'): ")

這個 goroutine 用來監聽服務器的響應并將其打印出來。

它在后臺運行,而主線程負責處理用戶輸入。

reader := bufio.NewReader(os.Stdin)
//Send data
    for {
        line, _ := reader.ReadString('\n')
        line = strings.TrimSpace(line)
        if line == "exit" {
            stream.CloseSend()
            break
        }
        parts := strings.Fields(line)
        iflen(parts) != 2 {
            fmt.Println("??  Usage: <student_id> <exam_id>")
            continue
        }
        req := &exampb.GetExamResultRequest{
            StudentId: parts[0],
            ExamId:    parts[1],
        }
        if err := stream.Send(req); err != nil {
            log.Printf("send error: %v", err)
            break
        }
    }
  • 讀取用戶輸入(student_id exam_id)。
  • 通過流將每個請求發送到服務器。
  • 如果輸入 exit,客戶端會關閉發送流。
<-done
fmt.Println("?? Session ended.")
  • 通過 done 通道等待接收端 goroutine 結束。
  • 確保所有通信完成后程序才退出。
  • 現在可以像之前那樣,使用 make 命令來測試這個雙向 RPC。

九、收獲總結

通過該項目你將學會了以下知識:

  • 編寫 .proto 并生成 Go 代碼;
  • 實現 Unary / Streaming / 雙向流 全套 RPC;
  • 在 Go 中用通道與協程管理并發;
  • 組織微服務項目結構,保持易維護、可擴展。

無論你是剛入門 RPC,還是想在微服務中實現高效通信,都希望這份項目能提供一個良好的起點。

責任編輯:趙寧寧 來源: 令飛編程
相關推薦

2020-07-03 10:21:48

Go框架Docker

2022-04-01 15:18:42

Web 框架網絡通信

2023-03-05 23:11:07

Go語言服務

2024-01-08 08:36:29

HTTPGo代理服務器

2024-01-02 12:17:44

Go傳統遠程

2023-06-10 23:01:41

GrpcProtobuf數據

2024-01-02 13:58:04

GoREST API語言

2021-07-26 10:14:38

Go語言工具

2014-04-14 15:54:00

print()Web服務器

2023-05-10 08:05:41

GoWeb應用

2023-02-26 01:37:57

goORM代碼

2020-12-08 08:46:07

GoJava工具

2021-07-29 07:55:19

Demo 工作池

2022-03-06 19:57:50

狀態機easyfsm項目

2022-05-17 08:02:55

GoTryLock模式

2025-02-11 00:00:00

2019-07-05 08:39:39

GoSQL解析器

2023-03-26 22:02:53

APMPR監控

2024-08-21 08:21:45

CNN算法神經網絡

2024-08-02 10:28:13

算法NLP模型
點贊
收藏

51CTO技術棧公眾號

www.51色.com| 另类小说综合网| 久久福利免费视频| 欧一区二区三区| 亚洲国产精品自拍| 欧美精品一区二区三区在线四季| 特级西西444www大胆免费看| 亚洲精品在线观看91| 欧美mv日韩mv国产| 蜜臀av午夜一区二区三区| 3p在线观看| 国产成人午夜精品5599 | 在线免费看黄色片| 韩国成人在线| 亚洲午夜羞羞片| 亚洲电影网站| 姝姝窝人体www聚色窝| 秋霞电影网一区二区| 色综合男人天堂| 欧美三级视频网站| 牛牛影视一区二区三区免费看| 欧美一a一片一级一片| 国产精品久久久久久久久电影网| 邻居大乳一区二区三区| 国产jizzjizz一区二区| 国产精品久久久久久久电影| 玖玖爱免费视频| 日韩欧美三级| 亚洲欧美日韩视频一区| 国产无套精品一区二区三区| 国产国产一区| 欧美性猛交xxxx免费看久久久| 老司机午夜免费福利视频| 黄色大片在线免费观看| 99热精品一区二区| 999久久久| 国产免费av电影| 青草av.久久免费一区| 欧美一乱一性一交一视频| 九九热国产精品视频| 亚洲九九视频| 久久久999精品免费| 免费看裸体网站| 西瓜成人精品人成网站| 精品国产凹凸成av人导航| 国产精品久久久久久久av福利| 四虎成人在线| 欧美自拍偷拍一区| 亚洲成熟丰满熟妇高潮xxxxx| 蜜桃视频m3u8在线观看| 亚洲电影中文字幕在线观看| 欧洲精品视频在线| 国产一区久久精品| 最新不卡av在线| 一区二区三区四区国产| a√在线中文网新版址在线| 国产午夜精品一区二区三区嫩草| 你懂的网址一区二区三区| 成人小说亚洲一区二区三区| 国产精品18久久久久久久网站| 成人av在线亚洲| 国产理论视频在线观看| 久久99国产精品麻豆| 成人黄色av网站| hs视频在线观看| 国产成人自拍网| 成人av资源| 欧美一级淫片免费视频魅影视频| 成人黄色777网| 国产一区二区无遮挡| 午夜在线视频免费| 国产亚洲综合在线| 一本一道久久a久久精品综合| 蜜桃av在线免费观看| 国产精品福利一区| 日本天堂免费a| 蜜桃在线视频| 在线观看91视频| 成年人网站av| 成人性生交大片免费看96| 亚洲第一综合天堂另类专| 国产 中文 字幕 日韩 在线| 国产免费av一区二区三区| 一区二区欧美亚洲| 亚洲国产美女视频| 99热这里只有精品8| 国产成人福利网站| 国产美女自慰在线观看| 成人黄色在线视频| 日韩欧美一区二区视频在线播放 | 69av成年福利视频| 波多野结衣视频在线观看| 精品综合久久久久久8888| 成人av免费看| av亚洲在线| 亚洲高清在线精品| 国产一级片黄色| 日韩精品视频一区二区三区| 日韩精品在线观| 日韩在线视频网址| 嫩草成人www欧美| 91午夜理伦私人影院| 五月婷婷六月色| 中文字幕制服丝袜一区二区三区 | 久久蜜桃资源一区二区老牛| 成人国产亚洲精品a区天堂华泰| 亚洲精品一级片| 欧美—级在线免费片| 99在线观看视频免费| 国产成人精品123区免费视频| 日韩午夜精品电影| 在哪里可以看毛片| 精品福利电影| 91精品在线影院| 你懂的在线看| 亚洲国产cao| 亚洲黄色片免费| 国产毛片一区二区三区| 高清欧美电影在线| 国产精品国产三级国产aⅴ| 26uuu国产日韩综合| 欧美日韩中文字幕在线播放| 欧洲成人一区| 亚洲欧洲日产国码av系列天堂| 全网免费在线播放视频入口 | 99九九精品视频| 免费视频国产一区| 久久久久久欧美| 国产精品久久久久久久成人午夜| 久久精品男人天堂av| 亚洲熟妇无码一区二区三区| 日韩欧美一级| 久久夜色精品国产亚洲aⅴ| 国产一区免费看| 久久这里只有精品视频网| 青春草国产视频| 日韩区欧美区| 超薄丝袜一区二区| 97超碰资源站| 国产精品成人免费在线| 午夜激情福利在线| 欧美欧美黄在线二区| 91成人天堂久久成人| 蜜臀av免费在线观看| 亚洲精品成人少妇| 污视频在线观看免费网站| 久久人体视频| 91免费精品国偷自产在线| 五月香视频在线观看| 欧美视频第二页| jizzjizzjizz国产| 久久国产精品99久久久久久老狼| 色一情一乱一伦一区二区三区丨 | 99在线观看免费视频精品观看| 99re视频在线播放| 色www永久免费视频首页在线| 欧美一区二区三区思思人| 免费在线观看a级片| 精品一区二区三区久久| 亚洲自拍偷拍一区二区三区| 91国产精品| 欧美久久久精品| 亚洲国产日韩在线观看| 亚洲v日本v欧美v久久精品| 日韩av无码一区二区三区不卡| 一本色道久久综合亚洲精品不卡 | 欧美第十八页| 成人淫片在线看| 色呦呦在线看| 亚洲精品美女免费| 日韩黄色一级视频| 国产精品不卡在线| 中文字幕18页| 亚洲永久视频| 亚洲欧洲一区二区福利| 国产aa精品| 久久久噜久噜久久综合| 午夜精品久久久久久久第一页按摩 | 热re99久久精品国99热蜜月| 国产精品高潮久久| 久久夜色精品国产| 香蕉视频免费在线看| 在线视频中文字幕一区二区| 影音先锋男人资源在线观看| 国产盗摄一区二区三区| 国产特级淫片高清视频| 精品在线观看入口| 成人精品一区二区三区电影黑人| 欧美人体视频xxxxx| 日韩精品视频在线| 中文字幕在线播放日韩| 亚洲一区二区四区蜜桃| xxxxx在线观看| 精品一区二区日韩| 国产真实老熟女无套内射| 国产真实有声精品录音| 91观看网站| 国产综合av| 欧美激情性做爰免费视频| 欧美少妇另类| 日韩欧美亚洲国产精品字幕久久久| 午夜毛片在线观看| 一区免费观看视频| 三级黄色片网站| 国产精品一区在线观看你懂的| 日韩欧美在线播放视频| 亚洲色图网站| 日韩免费电影一区二区| 一区视频网站| 91精品国产综合久久香蕉最新版 | 丰满少妇在线观看资源站| 美国欧美日韩国产在线播放| 欧美三级在线观看视频| 亚欧美无遮挡hd高清在线视频| 久久精品99| 涩爱av色老久久精品偷偷鲁| 国产精品精品视频一区二区三区| 欧洲成人综合网| 中文字幕亚洲欧美日韩在线不卡| 色窝窝无码一区二区三区成人网站| 欧美日高清视频| 日本黄色一级视频| 午夜精品久久久| 激情综合网五月天| 亚洲欧洲三级电影| 级毛片内射视频| 91视频在线观看免费| 日韩精品xxx| 狠狠色丁香久久婷婷综| 激情视频综合网| 久久电影一区| 精品视频免费在线播放| 国语对白精品一区二区| 一道本在线观看视频| 99成人超碰| 亚洲精品日韩精品| 国产一区99| 欧美一区亚洲二区| 免费精品国产| 欧美视频1区| 午夜a一级毛片亚洲欧洲| 国产精品一区二区不卡视频| 亚洲国产aⅴ精品一区二区| 成人激情免费在线| 综合久草视频| 亚洲一区二区三区在线免费观看| 精品三级在线| 国产色综合天天综合网| 欧美伊人亚洲伊人色综合动图| 国产精品成人一区二区| 欧美三区四区| 国产精品精品一区二区三区午夜版 | 自拍偷拍亚洲激情| av最新在线观看| 亚洲色图丝袜美腿| 欧美黑人猛猛猛| 一区2区3区在线看| 久久久久久久国产精品毛片| 亚洲国产乱码最新视频| 国产午夜免费视频| 欧美日韩午夜视频在线观看| 特黄视频免费看| 在线免费不卡电影| 一级做a爰片久久毛片16| 欧美精品高清视频| 成人激情四射网| 亚洲精品美女视频| 超碰国产在线观看| 久久视频免费观看| 国产第一页在线| 欧美在线www| 成人精品电影在线| 亚洲资源在线看| 欧美日韩一区二区三区四区不卡| 欧美精品一区二区三区在线看午夜| 精品国产美女| 国内外成人激情免费视频| 欧美午夜久久| 国产女女做受ⅹxx高潮| 蜜桃在线一区二区三区| 亚洲精品在线网址| 99精品视频在线观看免费| 波多野吉衣中文字幕| 最新国产成人在线观看| www.av麻豆| 欧美日韩日日骚| 亚洲精品综合久久| 亚洲欧美日韩精品久久奇米色影视| 日本高清中文字幕在线| 久久久久久久久久亚洲| 性欧美1819sex性高清| 91亚洲精品一区| 台湾亚洲精品一区二区tv| 亚洲蜜桃av| 伊人久久综合| 日日干夜夜操s8| 不卡高清视频专区| 天天操天天摸天天舔| 亚洲午夜久久久久中文字幕久| 四虎影院在线免费播放| 日韩一区二区精品葵司在线 | 亚洲亚洲一区二区三区| 日本a级片久久久| 国产精品草草| 五月天婷婷亚洲| www激情久久| 免看一级a毛片一片成人不卡| 欧洲精品在线观看| 蜜臀av在线观看| 萌白酱国产一区二区| 88xx成人永久免费观看| 成人精品一二区| 99re6这里只有精品| 欧美精品一区免费| 国产99精品国产| 天海翼在线视频| 欧美中文字幕亚洲一区二区va在线| 可以免费看毛片的网站| 久久久精品国产网站| 日韩成人影音| 久久久一本精品99久久精品| 激情综合自拍| 极品人妻一区二区| 中文字幕亚洲一区二区va在线| 久久青青草原亚洲av无码麻豆| 欧美va在线播放| 国产婷婷视频在线| 国产精品精品一区二区三区午夜版 | 欧美大电影免费观看| 国产91亚洲精品一区二区三区| 99热精品久久| 色悠悠久久综合网| 国产三区在线成人av| 国产黄色免费观看| 亚洲精品理论电影| av在线中出| 国产精品果冻传媒潘| 欧美+日本+国产+在线a∨观看| 中文字幕成人免费视频| 日本一区二区三区四区| 久久精品久久久久久久| 亚洲欧美国产另类| 美脚恋feet久草欧美| 精品视频在线观看| 一本久道久久综合婷婷鲸鱼| 秘密基地免费观看完整版中文| 亚洲影院免费观看| 成 人 免费 黄 色| 久久久久久免费精品| 九九热播视频在线精品6| 欧美视频在线观看视频| 成人精品高清在线| 男人的天堂一区二区| 亚洲精品成人网| 色在线视频观看| 欧美日韩在线精品| 肉色丝袜一区二区| 在线观看免费黄色网址| 欧美日韩成人综合在线一区二区| 亚洲视频tv| 91视频免费在线观看| 一区二区视频欧美| 少妇大叫太粗太大爽一区二区| 日本国产一区二区| 99re在线视频| 亚洲影院色无极综合| 国产精品vip| 鲁大师私人影院在线观看| 一本色道亚洲精品aⅴ| 国产高清视频在线观看| 国产日韩亚洲欧美| 女主播福利一区| 女人被狂躁c到高潮| 欧美亚洲愉拍一区二区| 国产写真视频在线观看| 国产丝袜不卡| 日本成人在线电影网| 婷婷社区五月天| 亚洲成人黄色在线| 桃子视频成人app| 懂色av粉嫩av蜜臀av| 成人久久视频在线观看| 丰满少妇xoxoxo视频| 日韩亚洲欧美中文在线| 日韩亚洲精品在线观看| 国产a级一级片| 亚洲丝袜另类动漫二区| 手机看片一区二区三区| 国产精品一区二区久久久| 国产精品啊啊啊| 调教驯服丰满美艳麻麻在线视频| 欧美一区二区观看视频| 日韩伦理在线一区| 亚洲综合av一区| 成人avav影音| 一区二区美女视频| 97在线精品国自产拍中文| 久久大综合网| 51调教丨国产调教视频| 日韩一级完整毛片|