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

數據庫中間件 MyCAT 源碼分析 —— SQL ON MongoDB

數據庫 其他數據庫 MongoDB
Java數據庫連接,(Java Database Connectivity,簡稱JDBC)是Java語言中用來規范客戶端程序如何來訪問數據庫的應用程序接口,提供了諸如查詢和更新數據庫中數據的方法。JDBC也是Sun Microsystems的商標。JDBC是面向關系型數據庫的。
  • 1. 概述
  • 2. 主流程
  • 3. 查詢操作
  • 4. 插入操作

1. 概述

可能你在看到這個標題會小小的吃驚,MyCAT 能使用 MongoDB 做數據節點。是的,沒錯,確實可以。

吼吼吼,讓我們開啟這段神奇的“旅途”。

本文主要分成四部分:

  1. 總體流程,讓你有個整體的認識
  2. 查詢操作
  3. 插入操作

2. 主流程

MyCAT Server 接收 MySQL Client 基于 MySQL協議 的請求,翻譯 SQL 成 MongoDB操作 發送給 MongoDB Server。

MyCAT Server 接收 MongoDB Server 返回的 MongoDB數據,翻譯成 MySQL數據結果 返回給 MySQL Client。

這樣一看,MyCAT 連接 MongoDB 是不是少神奇一點列。

Java數據庫連接,(Java Database Connectivity,簡稱JDBC)是Java語言中用來規范客戶端程序如何來訪問數據庫的應用程序接口,提供了諸如查詢和更新數據庫中數據的方法。JDBC也是Sun Microsystems的商標。JDBC是面向關系型數據庫的。

MyCAT 使用 JDBC 規范,抽象了對 MongoDB 的訪問。通過這樣的方式,MyCAT 也抽象了 SequoiaDB 的訪問??赡苓@樣說法有些抽象,看個類圖壓壓驚。

是不是熟悉的味道。不得不說 JDBC 規范的精妙。

3. 查詢操作

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

看順序圖已經很方便的理解整體邏輯,我就不多廢話啦。我們來看幾個核心的代碼邏輯。

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; 

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}); 

3、解析 MongoDB 數據

  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.        // 設置 fields 類型 
  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.    } 
  • 當使用 SELECT * 查詢字段時,fields 使用***條數據返回的 fields。即使,后面的數據有其他 fields,也不返回。

4、返回數據給 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(); 

當返回字段值是 Object 時,返回該對象.toString()。例如:

  1. mysql> select * from user order by _id asc
  2. +--------------------------+------+-------------------------------+ 
  3. | _id                      | name | profile                       | 
  4. +--------------------------+------+-------------------------------+ 
  5. | 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; 
責任編輯:武曉燕 來源: oschina博客
相關推薦

2017-07-18 17:35:16

數據庫MyCATPreparedSta

2017-07-18 17:07:40

數據庫 MyCATJoin

2017-12-01 05:04:32

數據庫中間件Atlas

2017-11-27 05:36:16

數據庫中間件TDDL

2017-11-27 05:06:42

數據庫中間件cobar

2018-02-24 19:37:33

Java8數據庫中間件

2009-01-20 10:45:55

Oracle數據庫中間件

2011-08-10 13:03:58

CJDBC數據庫集群

2017-05-23 18:55:05

mysql-proxy數據庫架構

2017-11-27 06:01:37

數據庫中間件中間層

2017-12-01 05:40:56

數據庫中間件join

2017-12-11 13:30:49

Go語言數據庫中間件

2020-04-10 17:00:33

Mycat分庫分表SpringBoot

2017-11-03 11:02:08

數據庫中間件

2017-11-30 08:56:14

數據庫中間件架構師

2024-12-06 08:29:29

2019-05-13 15:00:14

MySQLMyCat數據庫

2021-07-27 05:49:59

MySQL數據庫中間件

2020-10-15 08:34:32

數據庫中間件漫談

2018-11-07 15:30:19

數據庫NewSQLNoSQL
點贊
收藏

51CTO技術棧公眾號

国内av一区二区| 成人激情视频在线| 在线 丝袜 欧美 日韩 制服| 热色播在线视频| 国产香蕉久久精品综合网| 国产日韩精品在线| 久草视频在线免费看| 五月国产精品| 欧美高清性hdvideosex| 成人在线视频一区二区三区| 日本天堂影院在线视频| 蜜臀精品久久久久久蜜臀| 欧美极品xxxx| 91精品久久久久久久久久久久| 996久久国产精品线观看| 亚洲大片一区二区三区| 9999在线观看| 牛牛热在线视频| 国产成人精品亚洲午夜麻豆| 国产精品久久久久久久久久| 国产无遮挡裸体免费视频| 色97色成人| 精品亚洲国产视频| 野花视频免费在线观看| 成人在线视频免费| 欧美日韩国产激情| 4444在线观看| 92国产在线视频| 久久夜色精品国产噜噜av | 欧美大陆一区二区| 国产又黄又爽视频| 日韩激情一区二区| 18性欧美xxxⅹ性满足| 全网免费在线播放视频入口| 北条麻妃在线视频| 99视频在线观看免费| 丝袜诱惑制服诱惑色一区在线观看 | 久久精品一区二| 都市激情久久综合| 日韩久久一区二区| 中文字幕99| 97在线观看免费观看高清| 91日韩在线专区| 国产一区二区黄色| 亚洲精品人妻无码| 国产精品一区免费视频| 成人免费视频97| 在线免费av网| 免费人成精品欧美精品| 国产精品678| 亚洲欧美偷拍一区| 午夜一级久久| 亲爱的老师9免费观看全集电视剧| 久久久综合久久| 午夜精品网站| 九九综合九九综合| 欧美日韩综合一区二区| 欧美激情第8页| 欧美高清电影在线看| 青青青在线免费观看| 99国产**精品****| 久久亚洲综合国产精品99麻豆精品福利 | 久久综合伊人77777| 天海翼在线视频| 亚洲精品二区三区| 欧美久久精品午夜青青大伊人| 手机在线免费看毛片| 日韩精品免费一区二区三区| 中文字幕久久久| 任我爽在线视频| 欧美精品一区二区三区久久久竹菊| 免费91在线视频| 一区二区三区免费高清视频| 精品电影一区| 欧亚精品在线观看| 久久99精品久久久久久秒播放器| 欧美色图亚洲激情| 竹菊久久久久久久| 综合欧美国产视频二区| 疯狂试爱三2浴室激情视频| 欧美黄在线观看| 午夜剧场成人观在线视频免费观看| 日操夜操天天操| 媚黑女一区二区| 国产精品美女久久久久久免费| 国产精品久久久久久久免费看| 国产乱人伦精品一区二区在线观看| 国产v亚洲v天堂无码| 五月婷婷丁香网| 中文乱码免费一区二区 | 一本到12不卡视频在线dvd| 欧美国产精品人人做人人爱| 欧美一级片免费在线观看| 日韩国产欧美三级| 91传媒视频免费| 色鬼7777久久| 中文字幕一区二区不卡| 国产高清av在线播放| 精品三区视频| 欧美va亚洲va| 天天摸日日摸狠狠添| 欧美不卡视频| 国产不卡视频在线| 国内爆初菊对白视频| 国产亚洲欧美日韩在线一区| 91麻豆天美传媒在线| 超碰超碰人人人人精品| 91精品国产乱| 在线观看福利片| 欧美精品偷拍| 国产日产久久高清欧美一区| 全国男人的天堂网| 综合久久久久久久| 情侣黄网站免费看| 国产无遮挡裸体免费久久| 日韩在线免费观看视频| 欧美bbbbbbbbbbbb精品| 国产一区二区不卡在线| 日韩亚洲视频在线| 欧美激情20| 精品少妇一区二区三区视频免付费| jizz中文字幕| 国产精品美女久久久浪潮软件| 久久资源在线| 北条麻妃在线一区二区| 美日韩一二三区| 国产成人啪免费观看软件| 亚洲不卡1区| a级片在线免费观看| 91精品国产欧美一区二区18| 女人十八毛片嫩草av| 亚洲综合99| 国产日韩二区| 久久国产精品黑丝| 777亚洲妇女| 免费黄色国产视频| 日韩二区三区四区| 欧美日韩国产精品一区二区| 国产不卡人人| 亚洲精品一区在线观看| 久草视频免费在线播放| 国产精品一区二区在线看| 一区二区三区四区国产| 91成人在线| 影音先锋欧美精品| 波多野结衣电车痴汉| 国产亚洲欧美在线| 精品久久久久av| 久操成人av| 国产精品jizz在线观看麻豆| 欧美日韩免费做爰大片| 一本色道久久综合亚洲91| 特大黑人巨人吊xxxx| 99精品欧美| 精品免费国产| 成人美女视频| 亚洲色无码播放| 国产日韩在线免费观看| 国产女人18毛片水真多成人如厕| 日本成人在线免费视频| 国产成人ay| 国产精品人人做人人爽| 在线中文资源天堂| 91精品国产全国免费观看| 久久久久亚洲av无码专区体验| 国产精品一区三区| 日韩欧美国产综合在线| 欧美极品在线观看| 国产精品久久久久久久久久| 欧美三级理伦电影| 欧美一级欧美一级在线播放| 欧美成人三级在线观看| 成人黄页在线观看| 日本在线观看a| 欧美gay男男猛男无套| 亚洲一区二区三区777| 国产福利一区二区三区在线观看| 国产视频一区二区| 亚洲成人av在线| 一级成人黄色片| 国产精品久久久久三级| 下面一进一出好爽视频| 国产日韩一区二区三区在线| 日韩欧美一区二区在线观看 | av在线免费观看网站| 正在播放一区二区| 国产在线精品观看| 国产三级三级三级精品8ⅰ区| 久久国产精品国产精品| 欧美国产精品| 蜜桃麻豆www久久国产精品| 欧美高清影院| 久久久这里只有精品视频| 青青操在线视频| 91精品国产综合久久久久久久久久| 久久久久久久中文字幕| 国产三级一区二区| 天堂va欧美va亚洲va老司机| 日韩二区三区在线观看| 日韩精品一区二区三区四| 亚洲制服一区| 51国产成人精品午夜福中文下载| 97成人资源| 欧美国产精品va在线观看| www 日韩| 亚洲精品720p| 99久久久国产精品无码网爆| 色国产精品一区在线观看| 九九热视频精品| 国产欧美一区二区三区沐欲| 免费看毛片的网站| 久久99精品久久久久久动态图| 精品久久一二三| 一区二区在线| 亚洲精品第一区二区三区| 国产精品香蕉| 91在线直播亚洲| 成人黄色免费观看| 欧美一区第一页| 久草在线视频福利| 久久激情视频久久| www.国产精品.com| 亚洲午夜国产成人av电影男同| 高h放荡受浪受bl| 欧美一区午夜精品| 影音先锋国产在线| 一本一道久久a久久精品| 日韩av电影网址| 一区二区三区精品在线| 91麻豆精品成人一区二区| 国产农村妇女毛片精品久久麻豆| 国产精品无码永久免费不卡| 成人免费va视频| jjzz黄色片| 91最新在线| 疯狂欧美牲乱大交777| 欧美精品久久久久性色| 亚洲素人一区二区| 国产喷水在线观看| 国产精品久久久久国产精品日日| 国产jk精品白丝av在线观看| 91免费视频网址| 亚洲第一香蕉网| 久久综合久久久久88| av直播在线观看| 91小视频在线| 网站免费在线观看| 26uuu精品一区二区三区四区在线| 国产伦精品一区三区精东| www.视频一区| 中文字幕av观看| 久久蜜桃av一区二区天堂| 老司机福利av| 亚洲国产精品激情在线观看| 四虎成人免费影院| 国产精品国产成人国产三级| 极品色av影院| 亚洲精品乱码久久久久久久久 | 91影视免费在线观看| 黄色成人小视频| 国产精品美女网站| 在线免费成人| 国产精品二区在线| 久久久精品国产**网站| 国产欧美亚洲日本| 在线亚洲a色| 亚洲国产精品日韩| 亚洲第一天堂| 三上悠亚久久精品| 久久成人在线| 久热精品视频在线观看一区| 97久久人国产精品婷婷 | 国产男男gay体育生网站| 7777精品伊人久久久大香线蕉的| 国产ts变态重口人妖hd| 亚洲成人999| 成人免费在线视频网| 色偷偷88888欧美精品久久久 | 色棕色天天综合网| 午夜午夜精品一区二区三区文| 亚洲v在线看| 日韩欧美国产综合在线| 日本欧美一区二区三区乱码| 91网址在线观看精品| 91伊人久久大香线蕉| 欧美激情视频二区| 亚洲电影第三页| 又污又黄的网站| 精品99一区二区三区| 黄色软件在线| 欧美丰满老妇厨房牲生活| 欧美激情网站| 91精品国产99久久久久久红楼| 欧美日日夜夜| 韩国黄色一级大片| 国产亚洲精品久久久久婷婷瑜伽| 天天干天天玩天天操| bt欧美亚洲午夜电影天堂| 久久精品在线观看视频| 午夜精品久久久久久| 一级黄在线观看| 日韩国产一区三区| 最近中文字幕免费mv2018在线| 57pao成人永久免费视频| 久久久久久无码精品人妻一区二区| 亚洲精品乱码| 欧美美女一级片| 26uuu成人网一区二区三区| 日韩一区二区不卡视频| 日本久久一区二区| 人妻中文字幕一区| 久久韩国免费视频| 666av成人影院在线观看| 国产精品区一区二区三在线播放| 日韩成人精品一区| 色综合av综合无码综合网站| 国产v综合v亚洲欧| 欧美h片在线观看| 色婷婷综合久色| 国模私拍视频在线| 欧美精品免费在线观看| 色噜噜成人av在线| 日韩成人在线资源| 美女精品在线| 国产xxxx视频| 亚洲成人综合网站| 成人网站免费观看| 性欧美video高清bbw| 国产亚洲一级高清| 僵尸再翻生在线观看| 成人黄色在线免费观看| 在线电影一区二区| 亚洲国产高清av| 国产欧美日本一区视频| 亚洲婷婷综合网| 亚洲精品国产精品自产a区红杏吧| 亚洲国产精品精华素| 91亚洲国产成人久久精品网站| 欧美色蜜桃97| 欧美自拍小视频| 久久久久久久久一| 色av性av丰满av| 亚洲欧美日韩区| 成人免费影院| 日韩欧美精品一区二区| 久久婷婷av| 欧美剧情电影在线观看完整版免费励志电影| 99视频在线观看免费| 久久精品久久精品亚洲人| 色综合一区二区日本韩国亚洲| 最近2019中文免费高清视频观看www99| 蜜臀av在线播放| 豆国产97在线| 激情久久久久久久| 欧产日产国产精品98| 一区二区三区在线视频免费| 国产无套精品一区二区| 一本一道无码中文字幕精品热| 国产精品毛片高清在线完整版| 日本久久综合网| 亚洲午夜av久久乱码| 婷婷综合六月| 日韩三级电影免费观看| 久久99精品久久久久婷婷| 污污的视频在线免费观看| 欧美大胆人体bbbb| 成年女人在线看片| 欧美日韩高清免费| 99久久九九| 国产丝袜一区视频在线观看| 国产精品伦一区二区三区| 成人97在线观看视频| 一区二区三区在线资源| 国产69精品久久久久久久| 26uuu国产电影一区二区| 精品一区二三区| 精品久久久av| 国产在线播放精品| 无码人妻丰满熟妇区毛片| 亚洲欧洲三级电影| 欧美少妇bbw| 国产精品久久久久久久一区探花| 久久久9色精品国产一区二区三区| 下面一进一出好爽视频| 岛国精品视频在线播放| www.在线视频.com| 99电影在线观看| 久久亚洲欧美| 欧美国产在线看| 精品视频在线观看日韩| 欧美亚洲综合视频| 91麻豆swag| 国产一级在线观看视频| 亚洲精品一区中文字幕乱码| 亚洲国产91视频| 波多野结衣家庭教师在线| 成人欧美一区二区三区1314| 亚洲av电影一区| 91深夜福利视频| 视频一区二区不卡|