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

SQL ON MongoDB實(shí)現(xiàn)原理

運(yùn)維 數(shù)據(jù)庫(kù)運(yùn)維 MongoDB
可能你在看到這個(gè)標(biāo)題會(huì)小小的吃驚,MyCAT 能使用 MongoDB 做數(shù)據(jù)節(jié)點(diǎn)。是的,沒(méi)錯(cuò),確實(shí)可以。

1. 概述

可能你在看到這個(gè)標(biāo)題會(huì)小小的吃驚,MyCAT 能使用 MongoDB 做數(shù)據(jù)節(jié)點(diǎn)。是的,沒(méi)錯(cuò),確實(shí)可以。

吼吼吼,讓我們開(kāi)啟這段神奇的“旅途”。

本文主要分成四部分:

  1. 總體流程,讓你有個(gè)整體的認(rèn)識(shí)
  2. 查詢操作
  3. 插入操作
  4. 彩蛋,😈彩蛋,🙂彩蛋

建議你看過(guò)這兩篇文章(_非必須_):

  1. 《MyCAT源碼分析 —— 【單庫(kù)單表】插入》
  2. 《MyCAT源碼分析 —— 【單庫(kù)單表】查詢》

2. 主流程 

  1. MyCAT Server 接收 MySQL Client 基于 MySQL協(xié)議 的請(qǐng)求,翻譯 SQL 成 MongoDB操作 發(fā)送給 MongoDB Server。
  2. MyCAT Server 接收 MongoDB Server 返回的 MongoDB數(shù)據(jù),翻譯成 MySQL數(shù)據(jù)結(jié)果 返回給 MySQL Client。

這樣一看,MyCAT 連接 MongoDB 是不是少神奇一點(diǎn)列。 

 

 

 

Java數(shù)據(jù)庫(kù)連接,(Java Database Connectivity,簡(jiǎn)稱JDBC)是Java語(yǔ)言中用來(lái)規(guī)范客戶端程序如何來(lái)訪問(wèn)數(shù)據(jù)庫(kù)的應(yīng)用程序接口,提供了諸如查詢和更新數(shù)據(jù)庫(kù)中數(shù)據(jù)的方法。JDBC也是Sun Microsystems的商標(biāo)。JDBC是面向關(guān)系型數(shù)據(jù)庫(kù)的。

MyCAT 使用 JDBC 規(guī)范,抽象了對(duì) MongoDB 的訪問(wèn)。通過(guò)這樣的方式,MyCAT 也抽象了 SequoiaDB 的訪問(wèn)。可能這樣說(shuō)法有些抽象,看個(gè)類(lèi)圖壓壓驚。 

 

 

 

是不是熟悉的味道。不得不說(shuō) JDBC 規(guī)范的精妙。

3. 查詢操作

  1. SELECT id, name FROM user WHERE name > '' ORDER BY _id DESC 

 

 

 

看順序圖已經(jīng)很方便的理解整體邏輯,我就不多廢話啦。我們來(lái)看幾個(gè)核心的代碼邏輯。

1)、查詢 MongoDB

  1. // MongoSQLParser.java 
  2. public MongoData query() throws MongoSQLException { 
  3.    if (!(statement instanceof SQLSelectStatement)) { 
  4.        //return null
  5.        throw new IllegalArgumentException("not a query sql statement"); 
  6.    } 
  7.    MongoData mongo = new MongoData(); 
  8.    DBCursor c = null
  9.    SQLSelectStatement selectStmt = (SQLSelectStatement) statement; 
  10.    SQLSelectQuery sqlSelectQuery = selectStmt.getSelect().getQuery(); 
  11.    int icount = 0; 
  12.    if (sqlSelectQuery instanceof MySqlSelectQueryBlock) { 
  13.        MySqlSelectQueryBlock mysqlSelectQuery = (MySqlSelectQueryBlock) selectStmt.getSelect().getQuery(); 
  14.  
  15.        BasicDBObject fields = new BasicDBObject(); 
  16.  
  17.        // 顯示(返回)的字段 
  18.        for (SQLSelectItem item : mysqlSelectQuery.getSelectList()) { 
  19.            //System.out.println(item.toString()); 
  20.            if (!(item.getExpr() instanceof SQLAllColumnExpr)) { 
  21.                if (item.getExpr() instanceof SQLAggregateExpr) { 
  22.                    SQLAggregateExpr expr = (SQLAggregateExpr) item.getExpr(); 
  23.                    if (expr.getMethodName().equals("COUNT")) { // TODO 待讀:count(*) 
  24.                        icount = 1; 
  25.                        mongo.setField(getExprFieldName(expr), Types.BIGINT); 
  26.                    } 
  27.                    fields.put(getExprFieldName(expr), 1); 
  28.                } else { 
  29.                    fields.put(getFieldName(item), 1); 
  30.                } 
  31.            } 
  32.  
  33.        } 
  34.  
  35.        // 表名 
  36.        SQLTableSource table = mysqlSelectQuery.getFrom(); 
  37.        DBCollection coll = this._db.getCollection(table.toString()); 
  38.        mongo.setTable(table.toString()); 
  39.  
  40.        // WHERE 
  41.        SQLExpr expr = mysqlSelectQuery.getWhere(); 
  42.        DBObject query = parserWhere(expr); 
  43.  
  44.        // GROUP BY 
  45.        SQLSelectGroupByClause groupby = mysqlSelectQuery.getGroupBy(); 
  46.        BasicDBObject gbkey = new BasicDBObject(); 
  47.        if (groupby != null) { 
  48.            for (SQLExpr gbexpr : groupby.getItems()) { 
  49.                if (gbexpr instanceof SQLIdentifierExpr) { 
  50.                    String name = ((SQLIdentifierExpr) gbexpr).getName(); 
  51.                    gbkey.put(nameInteger.valueOf(1)); 
  52.                } 
  53.            } 
  54.            icount = 2; 
  55.        } 
  56.  
  57.        // SKIP / LIMIT 
  58.        int limitoff = 0; 
  59.        int limitnum = 0; 
  60.        if (mysqlSelectQuery.getLimit() != null) { 
  61.            limitoff = getSQLExprToInt(mysqlSelectQuery.getLimit().getOffset()); 
  62.            limitnum = getSQLExprToInt(mysqlSelectQuery.getLimit().getRowCount()); 
  63.        } 
  64.        if (icount == 1) { // COUNT(*) 
  65.            mongo.setCount(coll.count(query)); 
  66.        } else if (icount == 2) { // MapReduce 
  67.            BasicDBObject initial = new BasicDBObject(); 
  68.            initial.put("num", 0); 
  69.            String reduce = "function (obj, prev) { " + "  prev.num++}"
  70.            mongo.setGrouyBy(coll.group(gbkey, query, initial, reduce)); 
  71.        } else { 
  72.            if ((limitoff > 0) || (limitnum > 0)) { 
  73.                c = coll.find(query, fields).skip(limitoff).limit(limitnum); 
  74.            } else { 
  75.                c = coll.find(query, fields); 
  76.            } 
  77.  
  78.            // order by 
  79.            SQLOrderBy orderby = mysqlSelectQuery.getOrderBy(); 
  80.            if (orderby != null) { 
  81.                BasicDBObject order = new BasicDBObject(); 
  82.                for (int i = 0; i < orderby.getItems().size(); i++) { 
  83.                    SQLSelectOrderByItem orderitem = orderby.getItems().get(i); 
  84.                    order.put(orderitem.getExpr().toString(), getSQLExprToAsc(orderitem.getType())); 
  85.                } 
  86.                c.sort(order); 
  87.                // System.out.println(order); 
  88.            } 
  89.        } 
  90.        mongo.setCursor(c); 
  91.    } 
  92.    return mongo; 
  93.  

2)、查詢條件

  1. // MongoSQLParser.java 
  2. private void parserWhere(SQLExpr aexpr, BasicDBObject o) { 
  3.    if (aexpr instanceof SQLBinaryOpExpr) { 
  4.        SQLBinaryOpExpr expr = (SQLBinaryOpExpr) aexpr; 
  5.        SQLExpr exprL = expr.getLeft(); 
  6.        if (!(exprL instanceof SQLBinaryOpExpr)) { 
  7.            if (expr.getOperator().getName().equals("=")) { 
  8.                o.put(exprL.toString(), getExpValue(expr.getRight())); 
  9.            } else { 
  10.                String op = ""
  11.                if (expr.getOperator().getName().equals("<")) { 
  12.                    op = "$lt"
  13.                } else if (expr.getOperator().getName().equals("<=")) { 
  14.                    op = "$lte"
  15.                } else if (expr.getOperator().getName().equals(">")) { 
  16.                    op = "$gt"
  17.                } else if (expr.getOperator().getName().equals(">=")) { 
  18.                    op = "$gte"
  19.                } else if (expr.getOperator().getName().equals("!=")) { 
  20.                    op = "$ne"
  21.                } else if (expr.getOperator().getName().equals("<>")) { 
  22.                    op = "$ne"
  23.                } 
  24.                parserDBObject(o, exprL.toString(), op, getExpValue(expr.getRight())); 
  25.            } 
  26.        } else { 
  27.            if (expr.getOperator().getName().equals("AND")) { 
  28.                parserWhere(exprL, o); 
  29.                parserWhere(expr.getRight(), o); 
  30.            } else if (expr.getOperator().getName().equals("OR")) { 
  31.                orWhere(exprL, expr.getRight(), o); 
  32.            } else { 
  33.                throw new RuntimeException("Can't identify the operation of  of where"); 
  34.            } 
  35.        } 
  36.    } 
  37.  
  38. private void orWhere(SQLExpr exprL, SQLExpr exprR, BasicDBObject ob) { 
  39.    BasicDBObject xo = new BasicDBObject(); 
  40.    BasicDBObject yo = new BasicDBObject(); 
  41.    parserWhere(exprL, xo); 
  42.    parserWhere(exprR, yo); 
  43.    ob.put("$or", new Object[]{xo, yo}); 
  44.  

3)、解析 MongoDB 數(shù)據(jù)

  1. // MongoResultSet.java 
  2. public MongoResultSet(MongoData mongo, String schema) throws SQLException { 
  3.    this._cursor = mongo.getCursor(); 
  4.    this._schema = schema
  5.    this._table = mongo.getTable(); 
  6.    this.isSum = mongo.getCount() > 0; 
  7.    this._sum = mongo.getCount(); 
  8.    this.isGroupBy = mongo.getType(); 
  9.  
  10.    if (this.isGroupBy) { 
  11.        dblist = mongo.getGrouyBys(); 
  12.        this.isSum = true
  13.    } 
  14.    if (this._cursor != null) { 
  15.        select = _cursor.getKeysWanted().keySet().toArray(new String[0]); 
  16.        // 解析 fields 
  17.        if (this._cursor.hasNext()) { 
  18.            _cur = _cursor.next(); 
  19.            if (_cur != null) { 
  20.                if (select.length == 0) { 
  21.                    SetFields(_cur.keySet()); 
  22.                } 
  23.                _row = 1; 
  24.            } 
  25.        } 
  26.        // 設(shè)置 fields 類(lèi)型 
  27.        if (select.length == 0) { 
  28.            select = new String[]{"_id"}; 
  29.            SetFieldType(true); 
  30.        } else { 
  31.            SetFieldType(false); 
  32.        } 
  33.    } else { 
  34.        SetFields(mongo.getFields().keySet());//new String[]{"COUNT(*)"}; 
  35.        SetFieldType(mongo.getFields()); 
  36.    } 
  37.  

當(dāng)使用 SELECT * 查詢字段時(shí),fields 使用***條數(shù)據(jù)返回的 fields。即使,后面的數(shù)據(jù)有其他 fields,也不返回。

4)、返回?cái)?shù)據(jù)給 MySQL Client

  1. // JDBCConnection.java 
  2. private void ouputResultSet(ServerConnection sc, String sql) 
  3.        throws SQLException { 
  4.    ResultSet rs = null
  5.    Statement stmt = null
  6.  
  7.    try { 
  8.        stmt = con.createStatement(); 
  9.        rs = stmt.executeQuery(sql); 
  10.  
  11.        // header 
  12.        List<FieldPacket> fieldPks = new LinkedList<>(); 
  13.        ResultSetUtil.resultSetToFieldPacket(sc.getCharset(), fieldPks, rs, this.isSpark); 
  14.        int colunmCount = fieldPks.size(); 
  15.        ByteBuffer byteBuf = sc.allocate(); 
  16.        ResultSetHeaderPacket headerPkg = new ResultSetHeaderPacket(); 
  17.        headerPkg.fieldCount = fieldPks.size(); 
  18.        headerPkg.packetId = ++packetId; 
  19.        byteBuf = headerPkg.write(byteBuf, sc, true); 
  20.        byteBuf.flip(); 
  21.        byte[] header = new byte[byteBuf.limit()]; 
  22.        byteBuf.get(header); 
  23.        byteBuf.clear(); 
  24.        List<byte[]> fields = new ArrayList<byte[]>(fieldPks.size()); 
  25.        for (FieldPacket curField : fieldPks) { 
  26.            curField.packetId = ++packetId; 
  27.            byteBuf = curField.write(byteBuf, sc, false); 
  28.            byteBuf.flip(); 
  29.            byte[] field = new byte[byteBuf.limit()]; 
  30.            byteBuf.get(field); 
  31.            byteBuf.clear(); 
  32.            fields.add(field); 
  33.        } 
  34.        // header eof 
  35.        EOFPacket eofPckg = new EOFPacket(); 
  36.        eofPckg.packetId = ++packetId; 
  37.        byteBuf = eofPckg.write(byteBuf, sc, false); 
  38.        byteBuf.flip(); 
  39.        byte[] eof = new byte[byteBuf.limit()]; 
  40.        byteBuf.get(eof); 
  41.        byteBuf.clear(); 
  42.        this.respHandler.fieldEofResponse(header, fields, eof, this); 
  43.  
  44.        // row 
  45.        while (rs.next()) { 
  46.            RowDataPacket curRow = new RowDataPacket(colunmCount); 
  47.            for (int i = 0; i < colunmCount; i++) { 
  48.                int j = i + 1; 
  49.                if (MysqlDefs.isBianry((byte) fieldPks.get(i).type)) { 
  50.                    curRow.add(rs.getBytes(j)); 
  51.                } else if (fieldPks.get(i).type == MysqlDefs.FIELD_TYPE_DECIMAL || 
  52.                        fieldPks.get(i).type == (MysqlDefs.FIELD_TYPE_NEW_DECIMAL - 256)) { // field type is unsigned byte 
  53.                    // ensure that do not use scientific notation format 
  54.                    BigDecimal val = rs.getBigDecimal(j); 
  55.                    curRow.add(StringUtil.encode(val != null ? val.toPlainString() : null, sc.getCharset())); 
  56.                } else { 
  57.                    curRow.add(StringUtil.encode(rs.getString(j), sc.getCharset())); 
  58.                } 
  59.            } 
  60.            curRow.packetId = ++packetId; 
  61.            byteBuf = curRow.write(byteBuf, sc, false); 
  62.            byteBuf.flip(); 
  63.            byte[] row = new byte[byteBuf.limit()]; 
  64.            byteBuf.get(row); 
  65.            byteBuf.clear(); 
  66.            this.respHandler.rowResponse(row, this); 
  67.        } 
  68.        fieldPks.clear(); 
  69.        // row eof 
  70.        eofPckg = new EOFPacket(); 
  71.        eofPckg.packetId = ++packetId; 
  72.        byteBuf = eofPckg.write(byteBuf, sc, false); 
  73.        byteBuf.flip(); 
  74.        eof = new byte[byteBuf.limit()]; 
  75.        byteBuf.get(eof); 
  76.        sc.recycle(byteBuf); 
  77.        this.respHandler.rowEofResponse(eof, this); 
  78.    } finally { 
  79.        if (rs != null) { 
  80.            try { 
  81.                rs.close(); 
  82.            } catch (SQLException e) { 
  83.            } 
  84.        } 
  85.        if (stmt != null) { 
  86.            try { 
  87.                stmt.close(); 
  88.            } catch (SQLException e) { 
  89.            } 
  90.        } 
  91.    } 
  92.  
  93. // MongoResultSet.java 
  94. @Override 
  95. public String getString(String columnLabel) throws SQLException { 
  96.    Object x = getObject(columnLabel); 
  97.    if (x == null) { 
  98.        return null
  99.    } 
  100.    return x.toString(); 
  101.  

當(dāng)返回字段值是 Object 時(shí),返回該對(duì)象.toString()。例如:

  1. mysql> select * from user order by _id asc
  2.  
  3. +--------------------------+------+-------------------------------+ 
  4.  
  5. | _id | name | profile | 
  6.  
  7. +--------------------------+------+-------------------------------+ 
  8.  
  9. | 1 | 123 | { "age" : 1 , "height" : 100} |  

4. 插入操作 

 

 

 

 

  1. // MongoSQLParser.java 
  2. public int executeUpdate() throws MongoSQLException { 
  3.    if (statement instanceof SQLInsertStatement) { 
  4.        return InsertData((SQLInsertStatement) statement); 
  5.    } 
  6.    if (statement instanceof SQLUpdateStatement) { 
  7.        return UpData((SQLUpdateStatement) statement); 
  8.    } 
  9.    if (statement instanceof SQLDropTableStatement) { 
  10.        return dropTable((SQLDropTableStatement) statement); 
  11.    } 
  12.    if (statement instanceof SQLDeleteStatement) { 
  13.        return DeleteDate((SQLDeleteStatement) statement); 
  14.    } 
  15.    if (statement instanceof SQLCreateTableStatement) { 
  16.        return 1; 
  17.    } 
  18.    return 1; 
  19.  
  20. private int InsertData(SQLInsertStatement state) { 
  21.    if (state.getValues().getValues().size() == 0) { 
  22.        throw new RuntimeException("number of  columns error"); 
  23.    } 
  24.    if (state.getValues().getValues().size() != state.getColumns().size()) { 
  25.        throw new RuntimeException("number of values and columns have to match"); 
  26.    } 
  27.    SQLTableSource table = state.getTableSource(); 
  28.    BasicDBObject o = new BasicDBObject(); 
  29.    int i = 0; 
  30.    for (SQLExpr col : state.getColumns()) { 
  31.        o.put(getFieldName2(col), getExpValue(state.getValues().getValues().get(i))); 
  32.        i++; 
  33.    } 
  34.    DBCollection coll = this._db.getCollection(table.toString()); 
  35.    coll.insert(o); 
  36.    return 1; 
  37.  

5. 彩蛋 

1)、支持多 MongoDB ,并使用 MyCAT 進(jìn)行分片。

MyCAT 配置:multi_mongodb

2)、支持 MongoDB + MySQL 作為同一個(gè) MyCAT Table 的數(shù)據(jù)節(jié)點(diǎn)。查詢時(shí),可以合并數(shù)據(jù)結(jié)果。

查詢時(shí),返回 MySQL 數(shù)據(jù)記錄字段要比 MongoDB 數(shù)據(jù)記錄字段全,否則,合并結(jié)果時(shí)會(huì)報(bào)錯(cuò)。

MyCAT 配置:single_mongodb_mysql

3)、MongoDB 作為數(shù)據(jù)節(jié)點(diǎn)時(shí),可以使用 MyCAT 提供的數(shù)據(jù)庫(kù)主鍵字段功能。

MyCAT 配置:single_mongodb 

責(zé)任編輯:龐桂玉 來(lái)源: segmentfault
相關(guān)推薦

2022-06-21 14:02:29

MongoDB數(shù)據(jù)庫(kù)存儲(chǔ)

2011-08-30 10:22:14

MongoDB

2025-08-01 07:07:18

2017-07-11 13:58:10

WebSocket

2019-10-31 10:43:01

MongoDB數(shù)據(jù)庫(kù)Java

2023-01-30 18:44:45

MVCC事務(wù)

2023-01-04 07:54:03

HashMap底層JDK

2021-02-07 09:36:20

LongAdderJDK8開(kāi)發(fā)

2020-11-26 15:10:20

Python代碼函數(shù)

2020-06-01 16:05:17

MongoDB復(fù)制集數(shù)據(jù)庫(kù)

2009-07-02 12:57:00

SQL Server視

2025-05-19 00:02:45

SQL窗口函數(shù)

2024-08-05 11:14:45

2015-11-12 09:39:28

微信紅包實(shí)現(xiàn)

2024-12-23 15:05:29

2015-07-10 12:23:05

JsPatch實(shí)現(xiàn)原理

2017-12-06 16:28:48

Synchronize實(shí)現(xiàn)原理

2014-06-06 09:01:07

DHCP

2021-05-27 09:57:55

Inotify監(jiān)控系統(tǒng)

2023-04-17 08:13:13

KubernetesPod
點(diǎn)贊
收藏

51CTO技術(shù)棧公眾號(hào)

国产 欧美 日本| 亚洲综合最新在线| 成人乱码一区二区三区av| 国产精品原创| 国产亚洲欧洲一区高清在线观看| 日韩美女免费线视频| 男女男精品视频网站| 久久久久毛片免费观看| 天天综合色天天| 涩涩涩999| 精品美女www爽爽爽视频| 99精品视频网| 在线亚洲欧美视频| 日本泡妞xxxx免费视频软件| 黑人巨大精品| 亚洲综合成人在线视频| 秋霞在线观看一区二区三区| 国产ts变态重口人妖hd| 性欧美暴力猛交另类hd| 少妇高潮 亚洲精品| 四虎成人免费视频| 偷拍自拍亚洲| 欧美日韩在线免费| 成人在线免费高清视频| 国产小视频免费在线观看| 国产酒店精品激情| 国产精品久久97| 国产午夜激情视频| 一区二区三区午夜探花| 亚洲图片欧美日产| 88av在线播放| 精品视频一区二区三区| 在线免费亚洲电影| 国产亚洲欧美在线视频| 超碰最新在线| 国产精品国产三级国产有无不卡| 久久99精品国产99久久| 精品黑人一区二区三区在线观看| 日本美女一区二区| 欧洲亚洲免费在线| 日本一区二区三区免费视频| 亚洲精品国产偷自在线观看| 中文字幕视频一区二区在线有码| 久久一区二区电影| 国产成人一二| 欧美精品自拍偷拍| www.com黄色片| 日本精品不卡| 色香色香欲天天天影视综合网| 人人妻人人澡人人爽欧美一区双 | 亚洲欧美99| 黄色毛片在线看| 久久中文娱乐网| 久久久久久a亚洲欧洲aⅴ| 六月婷婷综合网| 国产iv一区二区三区| 亚洲精品欧美日韩专区| 国产又大又黄又爽| 国内精品国产三级国产a久久| 国产精品一二区| 中文字幕一区二区免费| 日日骚欧美日韩| 国产精品www网站| 伊人久久中文字幕| 日本美女一区二区三区| 国产伦精品免费视频| 亚洲图片在线播放| 韩国三级在线一区| 亚洲最大福利网| 亚洲AV无码精品自拍| 国产不卡视频一区二区三区| 国产高清精品一区| 无码国产精品96久久久久| 99久久伊人精品| 欧美高清视频一区| 成人欧美亚洲| 亚洲欧美偷拍另类a∨色屁股| 好吊色这里只有精品| 在线观看的网站你懂的| 性做久久久久久免费观看欧美| 2018日日夜夜| 激情都市亚洲| 337p亚洲精品色噜噜狠狠| 人妻精品久久久久中文字幕69| 亚洲超碰在线观看| 精品无人区太爽高潮在线播放| 日韩在线免费观看av| 日韩精品dvd| 欧美激情视频网站| 9i精品福利一区二区三区| 免费看黄色91| 国产精品国产精品| 国产三级电影在线观看| 国产精品精品国产色婷婷| 欧美黄色免费网址| 欧美理论影院| 日韩欧美一二三四区| 亚洲 欧美 日韩在线| 精品国产乱码久久久久久1区2匹| 色偷偷av亚洲男人的天堂| 青娱乐国产盛宴| 久久综合导航| 99热在线国产| avtt亚洲| 一区二区三区日韩欧美| 可以免费观看av毛片| 国产区一区二| 亚洲丝袜一区在线| 久久久久久久久久久97| 日本在线播放一区二区三区| 国产精品露出视频| 91大神在线网站| 黄色成人在线播放| 亚洲高清av一区二区三区| 综合伊思人在钱三区| 欧美精品在线极品| 欧美性受xxx黑人xyx性爽| 国产**成人网毛片九色 | 三级毛片在线免费看| 自拍偷在线精品自拍偷无码专区 | 9191久久久久久久久久久| 亚洲一区二区三区四区五区六区| 久久久久久久久久久久久久久久久久 | 久久久久亚洲| 国产成人av网址| 全部免费毛片在线播放一个| 亚洲视频综合在线| 在线视频日韩一区| 九九久久精品| 久久久久久久久91| 国内精品久久久久久久久久久| 国产欧美日韩久久| 精品视频一区二区在线| 成人搞黄视频| 欧美激情久久久久| 国产精品-色哟哟| 国产精品理论片在线观看| 久久美女福利视频| 香蕉精品久久| 97国产一区二区精品久久呦| www久久久久久| 亚洲欧美二区三区| 在线视频一二区| 欧美xxxx中国| 国产欧美日韩中文| 在线日本视频| 欧美私人免费视频| 亚洲一级片在线播放| 久久久久中文| 日韩福利视频| 亚洲成av在线| 中文字幕欧美亚洲| 伊人精品在线视频| 国产精品久久影院| 99九九99九九九99九他书对| 日韩成人精品一区| 国产色综合天天综合网 | 91视频一区二区三区| 成人在线国产视频| 欧美久久精品| 日韩女优在线播放| h视频网站在线观看| 在线一区二区视频| 国产一区二区三区视频播放| 久久66热偷产精品| 奇米777四色影视在线看| 深夜激情久久| 亚洲2020天天堂在线观看| 手机看片福利在线| 色婷婷久久综合| 东方伊人免费在线观看| 久久精品国产亚洲一区二区三区| 综合久久国产| 中文在线综合| 欧美在线视频网站| 永久免费av在线| 日韩美一区二区三区| 日韩高清精品免费观看| 久久久高清一区二区三区| 最新中文字幕免费视频| 91精品国产福利在线观看麻豆| 99在线观看视频网站| 黄色激情在线播放| 中文字幕精品一区久久久久 | 日韩理论视频| 在线日韩精品视频| а√中文在线资源库| 黑人精品xxx一区一二区| 女人十八毛片嫩草av| 国产精品1024| 漂亮人妻被中出中文字幕| 青青草成人影院| yy111111少妇影院日韩夜片 | 精品日韩中文字幕| 一级片久久久久| 成人高清视频免费观看| 黄色片在线免费| 国产精品第十页| 欧美最大成人综合网| 在线观看亚洲精品福利片| 97视频在线观看免费| 日本天堂在线观看| 日韩精品极品在线观看播放免费视频| 亚洲手机在线观看| 精品福利在线视频| 老熟妇高潮一区二区三区| 99久久免费精品高清特色大片| 最新天堂在线视频| 国产精品一二| 日本人妻伦在线中文字幕| 国内成人精品| 国产日韩欧美一区二区三区四区| 亚洲电影有码| 欧美性受xxxx黑人猛交| 国产高清一区二区三区视频| 亚洲人永久免费| 亚洲精品一区二区三区区别| 欧美日韩aaaaaa| 国产性生活视频| 亚洲福利一二三区| 91嫩草丨国产丨精品| 欧美极品aⅴ影院| 美国黄色一级毛片| 成人丝袜18视频在线观看| 中文字幕久久av| 天堂久久一区二区三区| 丰满少妇大力进入| 牛牛国产精品| 在线播放 亚洲| 日韩精品诱惑一区?区三区| 久久久久久国产精品免费免费| 日韩精品一区二区三区中文字幕 | 免费观看国产精品视频| 中文在线日韩| 中文字幕一区二区三区在线乱码| 国产欧美日韩精品一区二区三区| 电影一区二区三| 久操成人av| 2019亚洲男人天堂| 日本性爱视频在线观看| zzijzzij亚洲日本成熟少妇| 国产永久免费高清在线观看视频| 日韩精品视频在线观看免费| 黄色成人一级片| 亚洲成年人影院在线| а√中文在线资源库| 日韩精品一区二区三区在线 | 久久99精品久久| 日韩中文字幕在线精品| 97超碰国产一区二区三区| 亚洲欧美制服综合另类| 日本福利片高清在线观看| 亚洲激情中文字幕| 污视频网站免费观看| 亚洲精品黄网在线观看| 黄色美女一级片| 亚洲成人黄色网| 亚洲 国产 欧美 日韩| 亚洲精品91美女久久久久久久| 午夜在线视频观看| 亚洲女人被黑人巨大进入al| 你懂的在线观看| 在线成人中文字幕| 91caoporn在线| 久久手机精品视频| 色yeye免费人成网站在线观看| 欧美第一页在线| 国产精选在线| 国产精品美女主播| 精品久久国产一区| 精品国产一区二区三| 中文字幕在线视频第一页| 日韩国产精品久久久| xxxx一级片| 激情五月激情综合网| 国产传媒免费观看| 东方aⅴ免费观看久久av| 蜜臀av一区二区三区有限公司| 久久亚洲免费视频| 精品在线观看一区| 亚洲国产精品久久一线不卡| 久久国产黄色片| 欧美日韩国产精品自在自线| 精品区在线观看| 日韩精品在线私人| 免费在线观看黄色| 性欧美亚洲xxxx乳在线观看| 欧美黑人粗大| 999视频在线免费观看| 伦理一区二区| 亚洲欧洲日韩综合二区| 欧美另类综合| 国产熟人av一二三区| 国产一区不卡精品| 波多野结衣av在线免费观看| 中文字幕日韩av资源站| 日韩av电影网| 在线播放中文一区| 神马亚洲视频| 久久综合国产精品台湾中文娱乐网| 国产又色又爽又黄刺激在线视频| 国产精品91久久| 亚洲精品在线播放| 日韩欧美在线观看强乱免费| 欧美三级午夜理伦三级中文幕| 激情网站五月天| 福利一区二区在线观看| 午夜精产品一区二区在线观看的| 一区二区三区四区激情| 亚洲高清在线看| 亚洲精品电影在线观看| 成人亚洲性情网站www在线观看| 久久男人av资源网站| 欧美黄页免费| 欧美一进一出视频| 亚洲天堂黄色| 午夜激情视频网| 欧美激情在线一区二区三区| 黄色小视频在线免费看| 欧美精选午夜久久久乱码6080| 九色网友自拍视频手机在线| 欧美精品xxx| 亚洲成人1区| 日韩理论片在线观看| 一本色道久久综合亚洲精品高清 | 亚洲成a人片| 国产乱码精品一区二区三区中文 | 日本中文字幕一级片| 久久xxxx精品视频| 国产黑丝在线观看| 亚洲自拍偷拍九九九| 国产精品嫩草影院精东| 正在播放欧美一区| 91av亚洲| 欧美成人蜜桃| 亚洲视频播放| 波多野结衣先锋影音| 亚洲午夜激情网站| 亚洲精品国产精品乱码不卡| 久久av中文字幕| 国产一区精品二区| 日韩视频在线免费播放| 美国十次了思思久久精品导航| mm131丰满少妇人体欣赏图| 欧美性20hd另类| 三级视频在线| 国产成人一区二区三区小说| 综合国产视频| 91av俱乐部| 国产欧美日韩另类一区| 进去里视频在线观看| 色香阁99久久精品久久久| 欧美成a人片免费观看久久五月天| 日本免费高清不卡| 蜜桃视频一区二区| 亚洲色偷偷综合亚洲av伊人| 欧美一区国产二区| caoporn免费在线视频| 97免费资源站| 亚洲神马久久| 黄色片网站免费| 欧美欧美欧美欧美| 成人在线播放免费观看| 91九色偷拍| 在线亚洲国产精品网站| 亚洲一区二区三区无码久久| 色婷婷激情综合| 日本精品在线| 亚洲中国色老太| 亚洲黄色影片| 精品人伦一区二区三电影| 欧美视频精品在线| 97caopron在线视频| 国产亚洲福利社区| 香蕉国产精品偷在线观看不卡| 我和岳m愉情xxxⅹ视频| 欧美性猛片aaaaaaa做受| 黄网址在线观看| 国产99在线免费| 久久久久久穴| 欧美卡一卡二卡三| 日韩国产高清污视频在线观看| 欧美日韩免费观看视频| 午夜啪啪福利视频| 成人福利视频网站| 成人小视频在线播放| 久久综合久久八八| 中文精品一区二区| 无码国产精品一区二区高潮| 欧美日韩一区二区在线播放| 国产在线1区| 久久天堂国产精品| 国产在线精品视频| 日韩一区二区视频在线| 日韩视频在线免费观看| 你懂的在线观看一区二区| 天天干天天玩天天操| 亚洲国产精品久久艾草纯爱| 日本高清中文字幕在线| 精品日本一区二区|