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

MySQL 分布式鎖:設(shè)計(jì)與實(shí)現(xiàn)

數(shù)據(jù)庫 MySQL
如果想鎖住多份數(shù)據(jù)該怎么實(shí)現(xiàn)?比如說,某個(gè)庫存操作,既要修改物理庫存,又要修改虛擬庫存,想鎖住物理庫存的同時(shí),又鎖住虛擬庫存。

概述

以前參加過一個(gè)庫存系統(tǒng),由于其業(yè)務(wù)復(fù)雜性,搞了很多個(gè)應(yīng)用來支撐。這樣的話一份庫存數(shù)據(jù)就有可能同時(shí)有多個(gè)應(yīng)用來修改庫存數(shù)據(jù)。

比如說,有定時(shí)任務(wù)域xx.cron,和SystemA域和SystemB域這幾個(gè)JAVA應(yīng)用,可能同時(shí)修改同一份庫存數(shù)據(jù)。如果不做協(xié)調(diào)的話,就會有臟數(shù)據(jù)出現(xiàn)。

對于跨JAVA進(jìn)程的線程協(xié)調(diào),可以借助外部環(huán)境,例如DB或者Redis。下文介紹一下如何使用DB來實(shí)現(xiàn)分布式鎖。

設(shè)計(jì)

本文設(shè)計(jì)的分布式鎖的交互方式如下:

  • 根據(jù)業(yè)務(wù)字段生成transaction_id,線程安全的創(chuàng)建鎖資源
  • 根據(jù)transaction_id申請鎖
  • 釋放鎖

動態(tài)創(chuàng)建鎖資源

在使用synchronized關(guān)鍵字的時(shí)候,必須指定一個(gè)鎖對象。

synchronized(obj) {

}

進(jìn)程內(nèi)的線程可以基于obj來實(shí)現(xiàn)同步。obj在這里可以理解為一個(gè)鎖對象。如果線程要進(jìn)入synchronized代碼塊里,必須先持有obj對象上的鎖。這種鎖是JAVA里面的內(nèi)置鎖,創(chuàng)建的過程是線程安全的。那么借助DB,如何保證創(chuàng)建鎖的過程是線程安全的呢?

可以利用DB中的UNIQUE KEY特性,一旦出現(xiàn)了重復(fù)的key,由于UNIQUE KEY的唯一性,會拋出異常的。在JAVA里面,是SQLIntegrityConstraintViolationException異常。

create table distributed_lock
(
 id BIGINT UNSIGNED PRIMARY KEY AUTO_INCREMENT COMMENT '自增主鍵',
 transaction_id varchar(128) NOT NULL DEFAULT '' COMMENT '事務(wù)id',
 last_update_time TIMESTAMP DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP NOT NULL COMMENT '最后更新時(shí)間',
 create_time TIMESTAMP DEFAULT '0000-00-00 00:00:00' NOT NULL COMMENT '創(chuàng)建時(shí)間',
 UNIQUE KEY `idx_transaction_id` (`transaction_id`)
)

transaction_id是事務(wù)Id,可以用

倉庫 + 條碼 + 銷售模式

來組裝一個(gè)transaction_id,表示某倉庫某銷售模式下的某個(gè)條碼資源。不同條碼,當(dāng)然就有不同的transaction_id。如果有兩個(gè)應(yīng)用,拿著相同的transaction_id來創(chuàng)建鎖資源的時(shí)候,只能有一個(gè)應(yīng)用創(chuàng)建成功。

一條distributed_lock記錄插入成功了,就表示一份鎖資源創(chuàng)建成功了。

DB連接池列表設(shè)計(jì)

在寫操作頻繁的業(yè)務(wù)系統(tǒng)中,通常會進(jìn)行分庫,以降低單數(shù)據(jù)庫寫入的壓力,并提高寫操作的吞吐量。如果使用了分庫,那么業(yè)務(wù)數(shù)據(jù)自然也都分配到各個(gè)數(shù)據(jù)庫上了。

在這種水平切分的多數(shù)據(jù)庫上使用DB分布式鎖,可以自定義一個(gè)DataSouce列表。并暴露一個(gè)getConnection(String transactionId)方法,按照transactionId找到對應(yīng)的Connection。

實(shí)現(xiàn)代碼如下:

package dlock;

import com.alibaba.druid.pool.DruidDataSource;
import org.springframework.stereotype.Component;

import javax.annotation.PostConstruct;
import java.io.FileInputStream;
import java.io.IOException;
import java.sql.Connection;
import java.util.ArrayList;
import java.util.List;
import java.util.Properties;

@Component
public class DataSourcePool {
    private List<DruidDataSource> dlockDataSources = new ArrayList<>();

    @PostConstruct
    private void initDataSourceList() throws IOException {
        Properties properties = new Properties();
        FileInputStream fis = new FileInputStream("db.properties");
        properties.load(fis);

        Integer lockNum = Integer.valueOf(properties.getProperty("DLOCK_NUM"));
        for (int i = 0; i < lockNum; i++) {
            String user = properties.getProperty("DLOCK_USER_" + i);
            String password = properties.getProperty("DLOCK_PASS_" + i);
            Integer initSize = Integer.valueOf(properties.getProperty("DLOCK_INIT_SIZE_" + i));
            Integer maxSize = Integer.valueOf(properties.getProperty("DLOCK_MAX_SIZE_" + i));
            String url = properties.getProperty("DLOCK_URL_" + i);

            DruidDataSource dataSource = createDataSource(user,password,initSize,maxSize,url);
            dlockDataSources.add(dataSource);
        }
    }

    private DruidDataSource createDataSource(String user, String password, Integer initSize, Integer maxSize, String url) {
        DruidDataSource dataSource = new DruidDataSource();
        dataSource.setDriverClassName("com.mysql.jdbc.Driver");
        dataSource.setUsername(user);
        dataSource.setPassword(password);
        dataSource.setUrl(url);
        dataSource.setInitialSize(initSize);
        dataSource.setMaxActive(maxSize);

        return dataSource;
    }

    public Connection getConnection(String transactionId) throws Exception {
        if (dlockDataSources.size() <= 0) {
            return null;
        }

        if (transactionId == null || "".equals(transactionId)) {
            throw new RuntimeException("transactionId是必須的");
        }

        int hascode = transactionId.hashCode();
        if (hascode < 0) {
            hascode = - hascode;
        }

        return dlockDataSources.get(hascode % dlockDataSources.size()).getConnection();
    }
}

首先編寫一個(gè)initDataSourceList方法,并利用Spring的PostConstruct注解初始化一個(gè)DataSource 列表。相關(guān)的DB配置從db.properties讀取。

DLOCK_NUM=2

DLOCK_USER_0="user1"
DLOCK_PASS_0="pass1"
DLOCK_INIT_SIZE_0=2
DLOCK_MAX_SIZE_0=10
DLOCK_URL_0="jdbc:mysql://localhost:3306/test1"

DLOCK_USER_1="user1"
DLOCK_PASS_1="pass1"
DLOCK_INIT_SIZE_1=2
DLOCK_MAX_SIZE_1=10
DLOCK_URL_1="jdbc:mysql://localhost:3306/test2"

DataSource使用阿里的DruidDataSource。

接著最重要的一個(gè)實(shí)現(xiàn)getConnection(String transactionId)方法。實(shí)現(xiàn)原理很簡單,獲取transactionId的hashcode,并對DataSource的長度取模即可。

連接池列表設(shè)計(jì)好后,就可以實(shí)現(xiàn)往distributed_lock表插入數(shù)據(jù)了。

package dlock;

import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Component;

import java.sql.*;

@Component
public class DistributedLock {

    @Autowired
    private DataSourcePool dataSourcePool;

    /**
     * 根據(jù)transactionId創(chuàng)建鎖資源
     */
    public String createLock(String transactionId) throws Exception{
        if (transactionId == null) {
            throw new RuntimeException("transactionId是必須的");
        }
        Connection connection = null;
        Statement statement = null;
        try {
            connection = dataSourcePool.getConnection(transactionId);
            connection.setAutoCommit(false);
            statement = connection.createStatement();
            statement.executeUpdate("INSERT INTO distributed_lock(transaction_id) VALUES ('" + transactionId + "')");
            connection.commit();
            return transactionId;
        }
        catch (SQLIntegrityConstraintViolationException icv) {
            //說明已經(jīng)生成過了。
            if (connection != null) {
                connection.rollback();
            }
            return transactionId;
        }
        catch (Exception e) {
            if (connection != null) {
                connection.rollback();
            }
            throw  e;
        }
        finally {
            if (statement != null) {
                statement.close();
            }

            if (connection != null) {
                connection.close();
            }
        }
    }
}

根據(jù)transactionId鎖住線程

接下來利用DB的select for update特性來鎖住線程。當(dāng)多個(gè)線程根據(jù)相同的transactionId并發(fā)同時(shí)操作select for update的時(shí)候,只有一個(gè)線程能成功,其他線程都block住,直到select for update成功的線程使用commit操作后,block住的所有線程的其中一個(gè)線程才能開始干活。

我們在上面的DistributedLock類中創(chuàng)建一個(gè)lock方法。

public boolean lock(String transactionId) throws Exception {
        Connection connection = null;
        PreparedStatement preparedStatement = null;
        ResultSet resultSet = null;
        try {
            connection = dataSourcePool.getConnection(transactionId);
            preparedStatement = connection.prepareStatement("SELECT * FROM distributed_lock WHERE transaction_id = ? FOR UPDATE ");
            preparedStatement.setString(1,transactionId);
            resultSet = preparedStatement.executeQuery();
            if (!resultSet.next()) {
                connection.rollback();
                return false;
            }
            return true;
        } catch (Exception e) {
            if (connection != null) {
                connection.rollback();
            }
            throw  e;
        }
        finally {
            if (preparedStatement != null) {
                preparedStatement.close();
            }

            if (resultSet != null) {
                resultSet.close();
            }

            if (connection != null) {
                connection.close();
            }
        }
    }

實(shí)現(xiàn)解鎖操作

當(dāng)線程執(zhí)行完任務(wù)后,必須手動的執(zhí)行解鎖操作,之前被鎖住的線程才能繼續(xù)干活。在我們上面的實(shí)現(xiàn)中,其實(shí)就是獲取到當(dāng)時(shí)select for update成功的線程對應(yīng)的Connection,并執(zhí)行commit操作即可。

那么如何獲取到呢?我們可以利用ThreadLocal。首先在DistributedLock類中定義

private ThreadLocal<Connection> threadLocalConn = new ThreadLocal<>();

每次調(diào)用lock方法的時(shí)候,把Connection放置到ThreadLocal里面。我們修改lock方法。

public boolean lock(String transactionId) throws Exception {
        Connection connection = null;
        PreparedStatement preparedStatement = null;
        ResultSet resultSet = null;
        try {
            connection = dataSourcePool.getConnection(transactionId);
            threadLocalConn.set(connection);
            preparedStatement = connection.prepareStatement("SELECT * FROM distributed_lock WHERE transaction_id = ? FOR UPDATE ");
            preparedStatement.setString(1,transactionId);
            resultSet = preparedStatement.executeQuery();
            if (!resultSet.next()) {
                connection.rollback();
                threadLocalConn.remove();
                return false;
            }
            return true;
        } catch (Exception e) {
            if (connection != null) {
                connection.rollback();
                threadLocalConn.remove();
            }
            throw  e;
        }
        finally {
            if (preparedStatement != null) {
                preparedStatement.close();
            }

            if (resultSet != null) {
                resultSet.close();
            }

            if (connection != null) {
                connection.close();
            }
        }
    }

這樣子,當(dāng)獲取到Connection后,將其設(shè)置到ThreadLocal中,如果lock方法出現(xiàn)異常,則將其從ThreadLocal中移除掉。

有了這幾步后,我們可以來實(shí)現(xiàn)解鎖操作了。我們在DistributedLock添加一個(gè)unlock方法。

public void unlock() throws Exception {
        Connection connection = null;
        try {
            connection = threadLocalConn.get();
            if (!connection.isClosed()) {
                connection.commit();
                connection.close();
                threadLocalConn.remove();
            }
        } catch (Exception e) {
            if (connection != null) {
                connection.rollback();
                connection.close();
            }
            threadLocalConn.remove();
            throw e;
        }
    }

缺點(diǎn)

畢竟是利用DB來實(shí)現(xiàn)分布式鎖,對DB還是造成一定的壓力。當(dāng)時(shí)考慮使用DB做分布式的一個(gè)重要原因是,我們的應(yīng)用是后端應(yīng)用,平時(shí)流量不大的,反而關(guān)鍵的是要保證庫存數(shù)據(jù)的正確性。對于像前端庫存系統(tǒng),比如添加購物車占用庫存等操作,最好別使用DB來實(shí)現(xiàn)分布式鎖了。

進(jìn)一步思考

如果想鎖住多份數(shù)據(jù)該怎么實(shí)現(xiàn)?比如說,某個(gè)庫存操作,既要修改物理庫存,又要修改虛擬庫存,想鎖住物理庫存的同時(shí),又鎖住虛擬庫存。其實(shí)也不是很難,參考lock方法,寫一個(gè)multiLock方法,提供多個(gè)transactionId的入?yún)ⅲ琭or循環(huán)處理就可以了。這個(gè)后續(xù)有時(shí)間再補(bǔ)上。

責(zé)任編輯:武曉燕 來源: 一安未來
相關(guān)推薦

2024-10-09 17:12:34

2019-02-26 09:51:52

分布式鎖RedisZookeeper

2021-10-25 10:21:59

ZK分布式鎖ZooKeeper

2023-08-21 19:10:34

Redis分布式

2022-01-06 10:58:07

Redis數(shù)據(jù)分布式鎖

2024-11-28 15:11:28

2021-10-09 11:34:59

MySQL分布式鎖庫存

2019-06-19 15:40:06

分布式鎖RedisJava

2023-03-06 08:14:48

MySQLRedis場景

2024-10-07 10:07:31

2021-02-28 07:49:28

Zookeeper分布式

2017-01-16 14:13:37

分布式數(shù)據(jù)庫

2018-04-03 16:24:34

分布式方式

2024-04-01 05:10:00

Redis數(shù)據(jù)庫分布式鎖

2024-01-02 13:15:00

分布式鎖RedissonRedis

2022-04-08 08:27:08

分布式鎖系統(tǒng)

2017-04-13 10:51:09

Consul分布式

2023-01-13 07:39:07

2025-03-25 10:29:52

2021-06-10 06:57:39

Redis存儲數(shù)據(jù)庫
點(diǎn)贊
收藏

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

欧美中文字幕在线播放| 91精品国产91久久久久久一区二区 | 国产小视频你懂的| 亚洲日本va| 日本道在线观看一区二区| 一区二区三区我不卡| 国精产品一品二品国精品69xx | 性做久久久久久久免费看| 极品日韩久久| 97超碰中文字幕| 国产一区二区你懂的| xxxxx91麻豆| 熟女丰满老熟女熟妇| 亚洲欧洲日韩精品在线| 岛国av一区二区在线在线观看| 亚洲一区三区视频在线观看| 天天干视频在线观看| 精品无码三级在线观看视频| 91精品国产高清自在线| 99久久婷婷国产综合| 国产成人1区| 亚洲电影中文字幕| 午夜视频在线观| 国产亚洲一区二区手机在线观看| 一区二区三区四区五区视频在线观看 | 国产精品视频一二区| 欧美亚洲三级| 91青青国产在线观看精品| 日本国产一区| 综合在线观看色| 欧美韩国日本精品一区二区三区| 国产男男gay体育生白袜| 天堂影院一区二区| 69视频在线免费观看| 国产三级国产精品国产国在线观看| 国内成人自拍| 亚洲精品一区二区网址 | 国产高清不卡一区二区| 国产精品嫩草视频| 国产免费a视频| 亚洲男人影院| 日韩av手机在线| 日本韩国欧美中文字幕| 偷拍精品福利视频导航| 欧美丝袜丝nylons| 成人毛片视频网站| 2021天堂中文幕一二区在线观| 国产精品不卡在线| 亚洲人成网站在线播放2019| 国产精品秘入口| 国产精品午夜在线观看| 色噜噜狠狠一区二区三区| 男人的天堂在线| 国产色爱av资源综合区| 性欧美大战久久久久久久免费观看| 免费毛片在线| 国产午夜精品理论片a级大结局| 久久婷婷人人澡人人喊人人爽| 天堂网在线中文| 91亚洲国产成人精品一区二三| 国产伦精品一区二区三区高清版| 日本高清视频免费观看| 91视频观看免费| 日韩av电影免费播放| 第九色区av在线| 国产精品麻豆网站| 欧美一级中文字幕| 91九色国产在线播放| 欧美日韩一区二区在线| 国产三级日本三级在线播放| 国产精品原创视频| 日韩午夜激情视频| 制服丝袜第一页在线观看| 亚欧洲精品视频在线观看| 亚洲视频第一页| 亚洲激情图片网| 国产精品theporn| 欧美亚洲国产另类| 亚洲午夜激情视频| 国产一区二区美女诱惑| 国严精品久久久久久亚洲影视| 男女污视频在线观看| 中文字幕在线视频一区| 久操手机在线视频| 午夜无码国产理论在线| 欧美久久久影院| aaa黄色大片| 国产精品一区二区99| 免费不卡欧美自拍视频| 国产成人无码精品久在线观看 | 亚洲成人777777| 91丨porny丨首页| 亚洲永久一区二区三区在线| 久草在线视频资源| 欧美日韩精品综合在线| 成人区人妻精品一区二| 欧美一二区在线观看| 不卡av电影在线观看| 黄色一级片免费在线观看| 精品一区二区成人精品| 国产欧美一区二区三区不卡高清| 福利视频在线看| 亚洲一区日韩精品中文字幕| 99热手机在线| 美女视频亚洲色图| 久热国产精品视频| 在线观看国产区| 99视频精品全部免费在线| 一本久久a久久精品vr综合| av毛片午夜不卡高**水| 欧美一区二区三区小说| 国产一级久久久久毛片精品| 精品99视频| 国产欧美精品日韩精品| 色鬼7777久久| 亚洲成人精品影院| 欧美高清精品一区二区| 日韩激情图片| 日本精品视频在线观看| 日本韩国在线观看| 亚洲卡通欧美制服中文| 孩娇小videos精品| 蜜桃国内精品久久久久软件9| 欧美人交a欧美精品| 在线视频 中文字幕| 国产丝袜在线精品| 日韩有码免费视频| 老牛国内精品亚洲成av人片| 久久99国产精品自在自在app | 国内精品伊人久久久久av一坑| 欧美激情第六页| 欧美男男tv网站在线播放| 精品久久一二三区| 欧美成人一二三区| 国产又黄又大久久| 男人的天堂成人| 99re8精品视频在线观看| 国产一区av在线| 亚洲精品国产欧美在线观看| 久久久久久夜精品精品免费| 亚洲自偷自拍熟女另类| 91成人在线精品视频| 欧美激情视频三区| 亚洲精品成av人片天堂无码| 亚洲精品欧美二区三区中文字幕| www.夜夜爽| 小小影院久久| 亚洲xxxx18| av网站大全在线| 日韩视频在线你懂得| 麻豆精品一区二区三区视频| 国产高清不卡二三区| 日本一级黄视频| 精品日产乱码久久久久久仙踪林| 高清在线视频日韩欧美| 黑人操亚洲女人| 婷婷丁香激情综合| 欧美高清性xxxx| 日韩精品一卡二卡三卡四卡无卡| 日本午夜精品一区二区三区| 成人在线免费av| 久久国产精品久久精品| 六月丁香综合网| 欧美日韩亚洲系列| 欧美丰满美乳xxⅹ高潮www| 日韩av中文字幕一区二区| 一本久道久久综合狠狠爱亚洲精品 | 亚洲成av人片在线观看无码| 无码国产精品一区二区免费式直播| 亚洲精品一二| 日韩av在线电影观看| 成人短视频软件网站大全app| 九色精品免费永久在线| 天天操天天操天天干| 色综合色综合色综合色综合色综合| 国产精品国产三级国产专业不| 蜜桃久久av一区| 日本男女交配视频| 国产日产精品_国产精品毛片| 国产精品视频免费观看www| 中文字幕在线观看网站| 日韩激情第一页| 一级片视频免费| 午夜精品影院在线观看| 微拍福利一区二区| 国产传媒欧美日韩成人| 99色精品视频| 这里只有精品在线| 欧美一区二区在线| 日韩一区网站| 国产精品国产福利国产秒拍 | 欧美亚洲国产激情| 99中文字幕| 蜜桃视频成人m3u8| 欧美—级a级欧美特级ar全黄| 黄色片在线播放| 欧美zozozo| 中文字幕 亚洲视频| 午夜精品久久久久久久久| 女人裸体性做爰全过| 不卡视频一二三| 91网址在线观看精品| 久久精品麻豆| 加勒比成人在线| 久久在线免费| 欧美aaaaa喷水| 婷婷视频一区二区三区| 国产精品美女www| 麻豆网站免费在线观看| 欧美裸体xxxx极品少妇| 91精彩在线视频| 亚洲欧美激情四射在线日| 亚洲精品久久久蜜桃动漫 | 亚洲九九爱视频| 国产毛片欧美毛片久久久| www.亚洲人| 日韩一本精品| 这里只有精品国产| 国产亲近乱来精品视频 | ,亚洲人成毛片在线播放| 狠狠色狠色综合曰曰| 久久久久久免费观看| 国产精品国产三级国产普通话99 | 欧美精品导航| 一区二区三区视频| 欧美精品一区二区三区精品| 国产亚洲精品美女久久久m| 国产视频一区二| 国产三级精品网站| 久久人人视频| 国产精品99久久久久久白浆小说| 女厕盗摄一区二区三区| 91国产精品电影| 蜜桃视频在线观看播放| 韩国一区二区电影| 蜜桃视频在线网站| 日本久久久a级免费| 91av亚洲| 国产mv免费观看入口亚洲| 国产免费不卡| 国产aⅴ夜夜欢一区二区三区| 成人小电影网站| 国产精品激情av在线播放| 88xx成人永久免费观看| 国产精品高潮粉嫩av| 成人亚洲视频| 91美女片黄在线观看游戏| 95精品视频| 51蜜桃传媒精品一区二区| 91九色鹿精品国产综合久久香蕉| 5566av亚洲| 国产乱人伦精品一区| 91成人免费视频| 国产成人av毛片| 视频二区在线播放| 国产精品激情| 日韩亚洲欧美视频| 国产日本精品| 无码人妻丰满熟妇区毛片| 国产精品日本欧美一区二区三区| 日本wwww视频| 日韩va亚洲va欧美va久久| 国产九九热视频| 国产成人av福利| 亚洲第一黄色网址| 中文字幕乱码亚洲精品一区| jizz日本在线播放| 一区二区三区日本| 久久国产视频精品| 成人高清dvd| 亚洲综合色婷婷在线观看| 97人人香蕉| 精品人人人人| 视频一区二区三| 亚洲91视频| 国内成人精品2018免费看| 久久久国产精品一区二区三区| 清纯唯美亚洲经典中文字幕| 欧美一区二区三区四区夜夜大片| 精品美女视频| 最新视频 - x88av| 欧美视频一区| 可以免费观看av毛片| 美女诱惑一区二区| 视频一区亚洲| av久久网站| 91色视频在线导航| 黄色成人美女网站| 手机看片福利永久国产日韩| 一本到12不卡视频在线dvd| 日本a在线免费观看| 老司机久久99久久精品播放免费| 99九九99九九九99九他书对| 99国产一区二区三精品乱码| 国产精品1234| 国产麻豆久久| 999在线免费观看视频| 在线日韩网站| 大片在线观看网站免费收看| 久久一日本道色综合久久| 伊人五月天婷婷| 国产亚洲精品bt天堂精选| 久久一二三四区| 欧美亚洲动漫精品| 日本xxxxwww| 欧美xxxx综合视频| 成人精品国产| 精品伦精品一区二区三区视频 | 欧美成人免费网站| 成人不用播放器| 欧美专区在线播放| jizz性欧美23| 青青草视频国产| 九一九一国产精品| 国产一区二区三区四区五区六区 | 色悠悠久久88| 久久爱91午夜羞羞| 久久精品国产第一区二区三区最新章节 | 《视频一区视频二区| 真实乱视频国产免费观看| 亚洲一区二区视频在线| 在线免费看av的网站| 亚洲性夜色噜噜噜7777| 国产伦久视频在线观看| 粉嫩av一区二区三区免费观看| 9999国产精品| 伊人影院综合在线| 国产无遮挡一区二区三区毛片日本| 国产黄色片视频| 精品成人在线观看| 伊人春色在线观看| 5g影院天天爽成人免费下载| 天天射成人网| 亚洲老女人av| 国产精品―色哟哟| 正在播放亚洲精品| 国产性猛交xxxx免费看久久| 综合毛片免费视频| 欧美精品免费观看二区| 亚洲综合精品四区| 国产精品jizz| 91国偷自产一区二区开放时间| 欧美成人免费| 国产精品高潮在线| 红桃视频在线观看一区二区| 青青草av网站| 中文字幕二三区不卡| 最新中文字幕第一页| 色噜噜狠狠狠综合曰曰曰| 在线免费成人| 大地资源网在线观看免费官网| 国产成人精品www牛牛影视| 欧美极品aaaaabbbbb| 精品日韩在线观看| 超清av在线| 精品一区国产| 日韩电影在线观看网站| 污污视频网站在线免费观看| 欧美肥妇毛茸茸| 久久五月精品中文字幕| 精品免费国产| 日产欧产美韩系列久久99| 精品视频第一页| 日韩午夜激情视频| 九色porny丨国产首页在线| 欧美成ee人免费视频| 久久成人18免费观看| 欧美黄色免费观看| 亚洲精品久久久一区二区三区 | 亚洲91精品| 91精品啪在线观看国产| 一本一本久久a久久精品综合麻豆| 黄色片在线免费看| 91中文在线观看| 亚洲大黄网站| 五月天精品在线| 日韩三级.com| 亚洲最大网站| 青青草免费在线视频观看| 99久久精品免费| 亚洲综合精品国产一区二区三区| 色综合色综合网色综合| 国产亚洲精品美女久久久久久久久久| 色婷婷激情视频| 日韩欧美国产中文字幕| 国产福利视频在线| 久久久久久国产精品mv| 精品一区二区三区在线播放 | 免费观看在线午夜影视| 国内不卡一区二区三区| 另类综合日韩欧美亚洲| 国产一级一级片| 日韩中文字幕久久| 麻豆一区二区麻豆免费观看| 手机视频在线观看| 精品国产成人av| 黄色网在线播放| 亚洲高清精品中出| 99re成人精品视频| 精品国产va久久久久久久|