精品欧美一区二区三区在线观看 _久久久久国色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)

国产精品色哟哟| 亚洲免费激情视频| 蜜桃视频在线观看www社区| 亚洲无毛电影| 亚洲欧洲日产国产网站| 成年人深夜视频| 四虎影视2018在线播放alocalhost| 日韩欧美中文字幕视频| 亚洲一区二区免费在线观看| 午夜久久福利影院| 欧美重口乱码一区二区| 一级特黄aaa| 91精品久久久久久久蜜月| 亚洲第一天堂无码专区| 久久久久久久久久久久91| 成人看av片| 国产亚洲一区二区在线观看| 97se国产在线视频| 黄色av一级片| 国内精品嫩模av私拍在线观看| 日韩av中文字幕在线免费观看| 看欧美ab黄色大片视频免费 | 亚洲香蕉中文网| 欧洲性视频在线播放| 欧美激情中文不卡| 久久99久久精品国产| 一二三区在线播放| 国产欧美成人| 久久天堂av综合合色| 在线 丝袜 欧美 日韩 制服| 九九热最新地址| 欧美激情办公室videoshd| 日本中文字幕二区| 自拍网站在线观看| 亚洲国产成人精品视频| 四虎永久国产精品| 天天摸夜夜添狠狠添婷婷| 国产一区在线观看视频| 欧美一级免费看| 亚洲一区视频在线播放| 超碰成人免费| 欧美一区二区成人6969| 亚洲精品久久久久久久蜜桃臀| 黄色av网站在线| 91免费观看国产| 亚洲精品免费av| 国产精品视频一二区| 秋霞电影网一区二区| 欧美亚洲第一区| 日本视频免费在线| 日韩一区二区久久| 97视频免费看| 久久久精品国产sm调教网站| 91精品国产自产在线观看永久∴| 色悠悠久久久久| a级大片在线观看| 你懂的一区二区三区| 亚洲精品国精品久久99热 | 久久发布国产伦子伦精品| 欧美色999| 色天天综合久久久久综合片| 亚洲欧美日韩精品综合在线观看| av在线日韩国产精品| 亚洲国产成人一区二区三区| 色乱码一区二区三在线看| yiren22亚洲综合伊人22| 欧美激情一区二区三区蜜桃视频| 日韩成人在线资源| 五月天久久久久久| 久久亚区不卡日本| 亚洲精品成人三区| 黄网页免费在线观看| 亚洲精品国产a| 欧美精品在欧美一区二区| √天堂8资源中文在线| 亚洲国产精品影院| 黄色片久久久久| 欧美xxxx做受欧美护士| 欧美中文一区二区三区| 中国黄色片免费看| 亚洲综合资源| 制服丝袜亚洲色图| 制服丝袜在线第一页| 亚洲人成精品久久久| 日韩国产激情在线| 国产18无套直看片| 欧美a级在线| 97久久超碰福利国产精品…| 黄色一级片免费在线观看| 美腿丝袜亚洲一区| 国产精品二区三区| 视频二区在线| 亚洲视频免费看| 国产日韩亚洲欧美在线| segui88久久综合9999| 色综合久久中文综合久久97| 自拍亚洲欧美老师丝袜| 不卡视频观看| 欧美性生活大片免费观看网址| 国产精品69久久| 一区二区三区av| 在线看av的网址| 亚洲精品老司机| 国产一区视频免费观看| 精品一区91| 日韩色视频在线观看| 国产精品嫩草69影院| 免费成人蒂法| 日韩在线观看免费高清| 在线观看亚洲欧美| 国产麻豆日韩欧美久久| 欧美激情视频一区二区三区| 久cao在线| 一本久久精品一区二区| 四虎国产精品免费| 爽成人777777婷婷| 欧美一区二区三区四区在线| 国产成人精品a视频| 国产亚洲va综合人人澡精品| 国产一区 在线播放| 国产私拍福利精品视频二区| 亚洲国产91精品在线观看| 99自拍偷拍视频| 欧美日韩国产一区精品一区| 国产欧美日韩中文字幕| 毛片在线能看| 亚洲国产婷婷综合在线精品| 亚洲成色www.777999| 国产精品sss在线观看av| 久久综合伊人77777尤物| 亚洲视屏在线观看| 国产视频一区二区在线| 欧美大片在线播放| 日韩精品亚洲专区在线观看| 亚洲毛茸茸少妇高潮呻吟| 日本五十路女优| 成人永久看片免费视频天堂| 小说区视频区图片区| 亚洲国产尤物| 国产亚洲欧洲黄色| 中文字幕亚洲精品在线| 国产·精品毛片| www国产无套内射com| 日本三级一区| 日韩精品免费综合视频在线播放| 国产在线拍揄自揄拍| 狠狠色丁香久久婷婷综| 成年人黄色在线观看| 四虎影视国产精品| 伊人久久免费视频| 久草热在线观看| 久久夜色精品一区| 国产精品少妇在线视频| 国产一区2区| 97精品久久久中文字幕免费| 国产在线高清精品| 日本私人网站在线观看| 欧美日韩加勒比精品一区| 无码成人精品区在线观看| 国内视频精品| 久久国产主播精品| 超碰激情在线| 亚洲精品一区二区久| 免费又黄又爽又猛大片午夜| 国产亚洲福利社区一区| 色一情一区二区| 欧美国产免费| 国产一区二区精品在线| 亚洲性色av| 伊人激情综合网| 国产美女无遮挡永久免费| 一区二区三区在线视频观看| 国产午夜在线一区二区三区| 国产日韩亚洲| 亚洲精品第一区二区三区| 懂色av色香蕉一区二区蜜桃| 久久中文精品视频| 日本精品一二区| 在线亚洲一区二区| 国产av 一区二区三区| 成人免费毛片app| 欧美性大战久久久久xxx| 日韩精品免费一区二区在线观看| 91久久国产精品91久久性色| 男人天堂亚洲| 亚洲视频精品在线| 国产乱子伦精品无码码专区| 午夜视频一区二区三区| 手机看片福利视频| 国产美女久久久久| 日本a在线免费观看| 盗摄系列偷拍视频精品tp| 欧美日韩成人在线观看| 久青草国产在线| 欧美狂野另类xxxxoooo| 日韩成人免费在线视频| 国产精品不卡在线| 永久免费未满蜜桃| 免费av网站大全久久| 伊人再见免费在线观看高清版 | 亚洲情侣在线| 久久精品一二三区| 992tv人人草| 日韩mv欧美mv国产网站| 欧美专区日韩视频| 少妇av在线| 综合欧美国产视频二区| 三级视频在线看| 欧美视频在线一区二区三区| 国产真人真事毛片| 久久综合资源网| av污在线观看| 欧美暴力喷水在线| 四虎一区二区| 性人久久久久| 国产女人水真多18毛片18精品 | av老司机在线观看| 日韩中文字幕亚洲| 午夜影院免费视频| 欧美一级淫片007| 伊人网综合在线| 欧美视频一二三| caoporn91| 国产精品毛片无遮挡高清| 双性尿奴穿贞c带憋尿| 高清久久久久久| 久久久久亚洲av片无码v| 丝袜亚洲另类丝袜在线| 99爱视频在线| 亚洲福利国产| 中文字幕一区二区中文字幕| 精品高清久久| 日本成人看片网址| 蜜乳av综合| 欧美精品中文字幕一区二区| 国产厕拍一区| 岛国视频一区免费观看| 国产免费av国片精品草莓男男| 国产精品亚洲精品| 国产精品原创视频| 国产日韩综合一区二区性色av| av免费在线一区| 国产欧美久久一区二区| 九九热这里有精品| 国产欧美日韩视频| 亚洲伊人伊成久久人综合网| 成人网在线观看| 精品视频一区二区三区在线观看| 91免费观看网站| 精品一区二区三区中文字幕在线 | 亚洲成人网在线观看| 国产精品探花视频| 色综合天天综合给合国产| 国产美女激情视频| 五月婷在线视频| 四虎精品永久免费| 成人动漫免费在线观看| 亚洲free性xxxx护士白浆| 97精品资源在线观看| 91精品国产综合久久香蕉的用户体验 | 亚洲成人高清| 亚洲综合成人婷婷小说| 国产精品视频一区二区三区综合| 国产精品久久久久久久久借妻| 免费在线成人激情电影| 国产欧美一区二区白浆黑人| 精品亚洲a∨| a级国产乱理论片在线观看99| 91成人在线精品视频| 精品一区二区久久久久久久网站| 香蕉久久精品| 欧美日韩一区在线视频| 色97色成人| 椎名由奈jux491在线播放| 好看的日韩av电影| 欧美三级一级片| 免费久久99精品国产| 日本黄色三级网站| 91色乱码一区二区三区| 亚洲图片综合网| 久久久久久久久蜜桃| 精品在线观看一区| 亚洲一区二区在线视频| 天天干天天操天天爱| 在线看国产一区| 国产乱淫av片免费| 亚洲美腿欧美激情另类| 暖暖日本在线观看| 国模吧一区二区| 狠狠久久综合| 国产亚洲精品美女久久久m| 欧美丝袜激情| 久久久99精品视频| 日韩在线观看一区二区| 中文字幕久久久久久久| 久久―日本道色综合久久| 久久精品在线观看视频| 亚洲成a人在线观看| 中文字幕一区二区三区精品| 日韩欧美在线网址 | 成人国产精品入口免费视频| 国产原创欧美精品| 久久精品97| 国产欧美一区二区| 亚洲三级精品| 日本一区二区三区视频免费看| 中文字幕一区二区三三| 国产精品69页| 成人永久免费视频| 神马久久精品综合| 日本精品视频一区二区三区| 精品黑人一区二区三区在线观看| 亚洲一级免费视频| 男女视频在线| 亚洲在线视频福利| 欧美激情在线免费| 成人国产一区二区三区| 免费看日韩精品| 免费看黄色av| 欧美视频在线看| 熟妇人妻av无码一区二区三区| 久久亚洲影音av资源网| 欧美影视资讯| 免费在线成人av电影| 伊人久久大香线蕉av超碰演员| 国产野外作爱视频播放| 91社区在线播放| 久久艹免费视频| 亚洲精品国产品国语在线| 日本欧美电影在线观看| 国产一区二区丝袜| 久久国产小视频| 婷婷免费在线观看| 97久久久精品综合88久久| 国产性xxxx| 91官网在线免费观看| 欧美女子与性| 日韩免费av一区二区| 国产丝袜一区| 成人网站免费观看入口| 狠狠色丁香婷综合久久| 亚洲欧美小视频| 日韩欧美国产电影| 成人看片免费| 91在线视频免费| 综合天堂av久久久久久久| jizzzz日本| 1000部国产精品成人观看| 91亚洲视频在线观看| x99av成人免费| 青青热久免费精品视频在线18| 国产日韩欧美精品| 亚洲区第一页| av在线播放网址| 欧美性猛交xxxx免费看漫画| 精品视频二区| 国产欧美久久久久久| 久久久久久久久久久9不雅视频| 99视频在线观看视频| 亚洲精品免费播放| 性一交一乱一透一a级| 超薄丝袜一区二区| 成人在线视频你懂的| 女人被男人躁得好爽免费视频| 成人小视频在线观看| 国产一级二级三级| 日韩风俗一区 二区| av激情成人网| 无码毛片aaa在线| www.亚洲色图.com| 天天操天天爽天天干| 日韩电影网在线| av亚洲一区| 免费看av软件| 不卡电影免费在线播放一区| 久久精品www| 精品成人佐山爱一区二区| 久久久久久久| 日韩欧美一区二区视频在线播放 | 污视频网站免费在线观看| 青青精品视频播放| 欧美wwwww| a级片在线观看视频| 欧美亚洲禁片免费| 国产后进白嫩翘臀在线观看视频| 九九九久久久| 激情五月婷婷综合| 国产成人免费观看视频 | 日韩视频在线观看一区二区| 182在线视频观看| 在线观看日韩片| 久久99久久久久| 国产亚洲精品成人| 精品亚洲一区二区三区| 国产91在线精品| 999在线观看视频| 亚洲视频一二区| 二区三区在线播放| 999热视频在线观看| 秋霞电影网一区二区|