日韩av黄I国产麻豆传媒I国产91av视频在线观看I日韩一区二区三区在线看I美女国产在线I麻豆视频国产在线观看I成人黄色短片

歡迎訪問 生活随笔!

生活随笔

當前位置: 首頁 > 编程语言 > java >内容正文

java

Java并发编程中的若干核心技术,向高手进阶

發(fā)布時間:2025/3/21 java 32 豆豆
生活随笔 收集整理的這篇文章主要介紹了 Java并发编程中的若干核心技术,向高手进阶 小編覺得挺不錯的,現(xiàn)在分享給大家,幫大家做個參考.

來源:http://www.jianshu.com/p/5f499f8212e7

引言

本文試圖從一個更高的視角來總結(jié)Java語言中的并發(fā)編程內(nèi)容,希望閱讀完本文之后,可以收獲一些內(nèi)容,至少應該知道在Java中做并發(fā)編程實踐的時候應該注意什么,應該關(guān)注什么,如何保證線程安全,以及如何選擇合適的工具來滿足需求。

當然,更深層次的內(nèi)容就會涉及到JVM層面的知識,包括底層對Java內(nèi)存的管理,對線程的管理等較為核心的問題,當然,本文的定位在于抽象與總結(jié),更為具體而深入的內(nèi)容就需要自己去實踐,考慮到可能篇幅過長、重復描述某些內(nèi)容,以及自身技術(shù)深度等原因,本文將在深度和廣度上做一些權(quán)衡,某些內(nèi)容會做一些深入的分析,而有些內(nèi)容會一帶而過,點到為止。

總之,本文就當是對學習Java并發(fā)編程內(nèi)容的一個總結(jié),以及給那些希望快速了解Java并發(fā)編程內(nèi)容的讀者拋磚引玉,不足之處還望指正。

?

Java線程

一般來說,在java中實現(xiàn)高并發(fā)是基于多線程編程的,所謂并發(fā),也就是多個線程同時工作,來處理我們的業(yè)務,在機器普遍多核心的今天,并發(fā)編程的意義極為重大,因為我們有多個cpu供線程使用,如果我們的應用依然只使用單線程模式來工作的話,對極度浪費機器資源的。所以,學習java并發(fā)知識的首要問題是:如何創(chuàng)建一個線程,并且讓這個線程做一些事情?

這是java并發(fā)編程內(nèi)容的起點,下面將分別介紹多個創(chuàng)建線程,并且讓線程做一些事情的方法。

?

繼承Thread類

繼承Thread類,然后重寫run方法,這是第一種創(chuàng)建線程的方法。run方法里面就是我們要做的事情,可以在run方法里面寫我們想要在新的線程里面運行的任務,下面是一個小例子,我們繼承了Thread類,并且在run方法里面打印出了當然線程的名字,然后sleep1秒中之后就退出了:

/*** Created by hujian06 on 2017/10/31.*?* the demo of thread*/ public class ThreadDemo {public static void main(String ... args) {AThread aThread = new AThread();//start the threadaThread.start();}}class AThread extends Thread {@Overridepublic void run() {System.out.println("Current Thread Name:" +Thread.currentThread().getName());try {Thread.sleep(1000);} catch (InterruptedException e) {e.printStackTrace();}} }

如果我們想要啟動這個線程,只需要像上面代碼中那樣,調(diào)用Thread類的start方法就可以了。

?

實現(xiàn)Runnable接口

啟動一個線程的第二種方法是實現(xiàn)Runnable接口,然后實現(xiàn)其run方法,將你想要在新線程里面執(zhí)行的業(yè)務代碼寫在run方法里面,下面的例子展示了這種方法啟動線程的示例,實現(xiàn)的功能和上面的第一種示例是一樣的:

/*** Created by hujian06 on 2017/10/31.** the demo of Runnable*/ public class ARunnableaDemo {public static void main(String ... args) {ARunnanle aRunnanle = new ARunnanle();Thread thread = new Thread(aRunnanle);thread.start();}}class ARunnanle implements Runnable {@Overridepublic void run() {System.out.println("Current Thread Name:" +Thread.currentThread().getName());try {Thread.sleep(1000);} catch (InterruptedException e) {e.printStackTrace();}} }

在啟動線程的時候,依然還是使用了Thread這個類,只是我們在構(gòu)造函數(shù)中將我們實現(xiàn)的Runnable對象傳遞進去了,所以在我們執(zhí)行Thread類的start方法的時候,實際執(zhí)行的內(nèi)容是我們的Runnable的run方法。

?

使用FutureTask

啟動一個新的線程的第三種方法是使用FutureTask,下面來看一下FutureTask的類圖,就可以明白為什么可以使用FutureTask來啟動一個新的線程了:

FutureTask的類圖

從FutureTask的類圖中可以看出,FutureTask實現(xiàn)了Runnable接口和Future接口,所以它兼?zhèn)銻unnable和Future兩種特性,下面先來看看如何使用FutureTask來啟動一個新的線程:

import java.util.concurrent.Callable; import java.util.concurrent.ExecutionException; import java.util.concurrent.FutureTask;/*** Created by hujian06 on 2017/10/31.** the demo of FutureTask*/ public class FutureTaskDemo {public static void main(String ... args) {ACallAble callAble = new ACallAble();FutureTask<String> futureTask = new FutureTask<>(callAble);Thread thread = new Thread(futureTask);thread.start();do {}while (!futureTask.isDone());try {String result = futureTask.get();System.out.println("Result:" + result);} catch (InterruptedException | ExecutionException e) {e.printStackTrace();}}}class ACallAble implements Callable<String> {@Overridepublic String call() throws Exception {Thread.sleep(1000);return "Thread-Name:" +?Thread.currentThread().getName();} }

可以看到,使用FutureTask來啟動一個線程之后,我們可以監(jiān)控這個線程是否完成,上面的示例中主線程會一直等待這個新創(chuàng)建的線程直到它返回,其實只要是Future提供的接口,我們在FutureTask中都可以使用,這極大的方便了我們,Future在并發(fā)編程中的意義極為重要,Future代表一個未來會發(fā)生的東西,它是一種暗示,一種占位符,它示意我們它可能不會立即得到結(jié)果,因為它的任務還在運行,但是我們可以得到一個對這個線程的監(jiān)控對象。

我們可以對線程的執(zhí)行做一些判斷,甚至是控制,比如,如果我們覺得我們等了太久,并且我們覺得沒有必要再等待下去的時候,就可以將這個Task取消,還有一點需要提到的是,Future代表它可能正在運行,也可能已經(jīng)返回,當然Future更多的暗示你可以在等待這個結(jié)果的同時可以使用其他的線程做一些其他的事情,當你真的需要這個結(jié)果的時候再來獲取就可以了,這就是并發(fā),理解這一點非常重要。

本小節(jié)通過介紹三種創(chuàng)建并啟動一個新線程的方法,為進行并發(fā)編程開了一個頭,目前,我們還只是在能創(chuàng)建多個線程,然后讓多個線程做不同個的事情的階段,當然,這是學習并發(fā)編程最為基礎(chǔ)的,無論如何,現(xiàn)在,我們可以讓我們的應用運行多個線程了,下面的文章將會基于這個假設(shè)(一個應用開啟了多個線程)討論一些并發(fā)編程中值得關(guān)注的內(nèi)容。

?

線程模型

我們現(xiàn)在可以啟動多個線程,但是好像并沒有形成一種類似于模型的東西,非常混亂,并且到目前為止我們的多個線程依然只是各自做各自的事情,互不相干,多個線程之間并沒有交互(通信),這是最簡單的模型,也是最基礎(chǔ)的模型,本小節(jié)試圖介紹線程模型,一種指導我們的代碼組織的思想,線程模型確定了我們需要處理那些多線程的問題,在一個系統(tǒng)中,多個線程之間沒有通信是不太可能的,更為一般的情況是,多個線程共享一些資源,然后相互競爭來獲取資源權(quán)限,多個線程相互配合,來提高系統(tǒng)的處理能力。

正因為多個線程之間會有通信交互,所以本文接下來的討論才有了意義,如果我們的系統(tǒng)里面有幾百個線程在工作,但是這些線程互不相干,那么這樣的系統(tǒng)要么實現(xiàn)的功能非常單一,要么毫無意義(當然不是絕對的,比如Netty的線程模型)。

繼續(xù)來討論線程模型,上面說到線程模型是一種指導代碼組織的思想,這是我自己的理解,不同的線程模型需要我們使用不同的代碼組織,好的線程模型可以提高系統(tǒng)的并發(fā)度,并且可以使得系統(tǒng)的復雜度降低,這里需要提一下Netty 4的線程模型,Netty 4的線程模型使得我們可以很容易的理解Netty的事件處理機制,這種優(yōu)秀的設(shè)計基于Reactor線程模型,Reactor線程模型分為單線程模型、多線程模型以及主從多線程模型,Netty的線程模型類似于Reactor主從多線程模型。

當然線程模型是一種更高級別的并發(fā)編程內(nèi)容,它是一種編程指導思想,尤其在我們進行底層框架設(shè)計的時候特別需要注意線程模型,因為一旦線程模型設(shè)計不合理,可能會導致后面框架代碼過于復雜,并且可能因為線程同步等問題造成問題不可控,最終導致系統(tǒng)運行失控。類似于Netty的線程模型是一種好的線程模型,下面展示了這種模型:

Netty線程模型

簡單來說,Netty為每個新建立的Channel分配一個NioEventLoop,而每個NioEventLoop內(nèi)部僅使用一個線程,這就避免了多線程并發(fā)的同步問題,因為為每個Channel處理的線程僅有一個,所以不需要使用鎖等線程同步手段來做線程同步,在我們的系統(tǒng)設(shè)計的時候應該借鑒這種線程模型的設(shè)計思路,可以避免我們走很多彎路。

?

Java線程池

池化技術(shù)是一種非常有用的技術(shù),對于線程來說,創(chuàng)建一個線程的代價是很高的,如果我們在創(chuàng)建了一個線程,并且讓這個線程做一個任務之后就回收的話,那么下次要使用線程來執(zhí)行我們的任務的時候又需要創(chuàng)建一個新的線程,是否可以在創(chuàng)建完成一個線程之后一直緩沖,直到系統(tǒng)關(guān)閉的時候再進行回收呢?

線程池就是這樣的組件,使用線程池,就沒必要頻繁創(chuàng)建線程,線程池會為我們管理線程,當我們需要一個新的線程來執(zhí)行我們的任務的時候,就向線程池申請,而線程池會從池子里面找到一個空閑的線程返回給請求者,如果池子里面沒有可用的線程,那么線程池會根據(jù)一些參數(shù)指標來創(chuàng)建一個新的線程,或者將我們的任務提交到任務隊列中去,等待一個空閑的線程來執(zhí)行這個任務。

細節(jié)內(nèi)容在下文中進行分析,目前我們只需要明白,線程池里面有很多線程,這些線程會一直到系統(tǒng)關(guān)系才會被回收,否則一直會處于處理任務或者等待處理任務的狀態(tài)。

首先,如何使用線程池呢?下面的代碼展示了如何使用java線程池的例子:

import java.util.concurrent.ExecutorService; import java.util.concurrent.Executors; import java.util.concurrent.ScheduledExecutorService; import java.util.concurrent.ThreadFactory; import java.util.concurrent.TimeUnit; import java.util.concurrent.atomic.AtomicInteger;/*** Created by hujian06 on 2017/10/31.** the demo of Executors*/ public class ExecutorsDemo {public static void main(String ... args) {int cpuCoreCount = Runtime.getRuntime().availableProcessors();AThreadFactory threadFactory = new AThreadFactory();ARunnanle runnanle = new ARunnanle();ExecutorService fixedThreadPool=Executors.newFixedThreadPool(cpuCoreCount, threadFactory);ExecutorService cachedThreadPool =?Executors.newCachedThreadPool(threadFactory);ScheduledExecutorService newScheduledThreadPool =?Executors.newScheduledThreadPool(cpuCoreCount, threadFactory);ScheduledExecutorService singleThreadExecutor =?Executors.newSingleThreadScheduledExecutor(threadFactory);fixedThreadPool.submit(runnanle);cachedThreadPool.submit(runnanle);newScheduledThreadPool.scheduleAtFixedRate(runnanle, 0, 1, TimeUnit.SECONDS);singleThreadExecutor.scheduleWithFixedDelay(runnanle, 0, 100, TimeUnit.MILLISECONDS);try {TimeUnit.SECONDS.sleep(5);} catch (InterruptedException e) {e.printStackTrace();}fixedThreadPool.shutdownNow();cachedThreadPool.shutdownNow();newScheduledThreadPool.shutdownNow();singleThreadExecutor.shutdownNow();}}class ARunnable implements Runnable {@Overridepublic void run() {try {Thread.sleep(1000);} catch (InterruptedException e) {e.printStackTrace();}System.out.println("Current Thread Name:" +?Thread.currentThread().getName());} }/*** the thread factory*/ class AThreadFactory implements ThreadFactory {private final AtomicInteger threadNumber = new AtomicInteger(1);@Overridepublic Thread newThread(Runnable r) {return new Thread("aThread-" + threadNumber.incrementAndGet());} }

更為豐富的應用應該自己去探索,結(jié)合自身的需求來借助線程池來實現(xiàn),下面來分析一下Java線程池實現(xiàn)中幾個較為重要的內(nèi)容。

?

ThreadPoolExecutor和ScheduledThreadPoolExecutor

ThreadPoolExecutor和ScheduledThreadPoolExecutor是java實現(xiàn)線程池的核心類,不同類型的線程池其實就是在使用不同的構(gòu)造函數(shù),以及不同的參數(shù)來構(gòu)造出ThreadPoolExecutor或者ScheduledThreadPoolExecutor,所以,學習java線程池的重點也在于學習這兩個核心類。

前者適用于構(gòu)造一般的線程池,而后者繼承了前者,并且很多內(nèi)容是通用的,但是ScheduledThreadPoolExecutor增加了schedule功能,也就是說,ScheduledThreadPoolExecutor使用于構(gòu)造具有調(diào)度功能的線程池,在需要周期性調(diào)度執(zhí)行的場景下就可以使用ScheduledThreadPoolExecutor。

下面展示了ThreadPoolExecutor和ScheduledThreadPoolExecutor的類圖,可以看出他們的關(guān)系,以及他們的繼承關(guān)系:

ThreadPoolExecutor類圖 ScheduledThreadPoolExecutor類圖

關(guān)于較為細節(jié)的內(nèi)容不再本文的敘述范圍之內(nèi),如果想要了解這些內(nèi)容的詳細內(nèi)容,可以參考文章中給出的鏈接,這些文章較為深入的分析和總結(jié)了相關(guān)的內(nèi)容。

上文中提到,線程池會管理著一些線程,這些線程要么處于運行狀態(tài),要么處于等待任務的狀態(tài),當然這只是我們較為形象的描述,一個線程的狀態(tài)不僅有運行態(tài)與等待狀態(tài),還有其他的狀態(tài),但是對我我們來說,線程池里面的線程確實是要么處于運行狀態(tài),要么處于等待任務的狀態(tài),這體現(xiàn)在,當我們向一個線程池提交一個任務的時候,可能會被等待任務的線程立即執(zhí)行,但是可能線程池里面的線程都處于忙碌狀態(tài),那么我們提交的任務就會被加入到等待運行的任務隊列中去,當有空閑線程了,或者隊列也滿了,那么線程池就會采用一些策略來執(zhí)行任務,并且在某些時刻會拒絕提交的任務,這些細節(jié)都可以在ThreadPoolExecutor的實現(xiàn)中找到。

在線程池的實現(xiàn)中,有一個角色特別重要,那就是任務隊列,當線程池里面沒有空閑的線程來執(zhí)行我們的任務的時候,我們的任務就會被添加到任務隊列中去等待執(zhí)行,而這個任務隊列可能會被多個線程并發(fā)讀寫,所以需要支持多線程安全訪問,java提供了一類支持并發(fā)環(huán)境的隊列,稱為阻塞隊列,這是一類特殊的隊列,他們的使用時非常廣泛的,特別是在jdk自身的類庫建設(shè)上,當然在我們實際的工作中也是有很多使用場景的。

關(guān)于ThreadPoolExecutor是如何處理一個提交的任務的細節(jié),可以參考下面的代碼:

??

public void execute(Runnable command) {if (command == null)throw new NullPointerException();/** Proceed in 3 steps:** 1. If fewer than corePoolSize threads are running, try to* start a new thread with the given command as its first* task.? The call to addWorker atomically checks runState and* workerCount, and so prevents false alarms that would add* threads when it shouldn't, by returning false.** 2. If a task can be successfully queued, then we still need* to double-check whether we should have added a thread* (because existing ones died since last checking) or that* the pool shut down since entry into this method. So we* recheck state and if necessary roll back the enqueuing if* stopped, or start a new thread if there are none.** 3. If we cannot queue task, then we try to add a new* thread.? If it fails, we know we are shut down or saturated* and so reject the task.*/int c = ctl.get();if (workerCountOf(c) < corePoolSize) {if (addWorker(command, true))return;c = ctl.get();}if (isRunning(c) && workQueue.offer(command)) {int recheck = ctl.get();if (! isRunning(recheck) && remove(command))reject(command);else if (workerCountOf(recheck) == 0)addWorker(null, false);}else if (!addWorker(command, false))reject(command);}

下面來看一下java中借助ThreadPoolExecutor來構(gòu)造的幾個線程池的特性:

1、newFixedThreadPool

使用ThreadPoolExecutor構(gòu)造一個newCachedThreadPool的流程如下:

??? public static ExecutorService newFixedThreadPool(int nThreads) {return new ThreadPoolExecutor(nThreads, nThreads,0L, TimeUnit.MILLISECONDS,new LinkedBlockingQueue<Runnable>());}public ThreadPoolExecutor(int corePoolSize,int maximumPoolSize,long keepAliveTime,TimeUnit unit,BlockingQueue<Runnable> workQueue) {this(corePoolSize, maximumPoolSize, keepAliveTime, unit, workQueue,Executors.defaultThreadFactory(), defaultHandler);}public ThreadPoolExecutor(int corePoolSize,int maximumPoolSize,long keepAliveTime,TimeUnit unit,BlockingQueue<Runnable> workQueue,ThreadFactory threadFactory,RejectedExecutionHandler handler) {if (corePoolSize < 0 ||maximumPoolSize <= 0 ||maximumPoolSize < corePoolSize ||keepAliveTime < 0)throw new IllegalArgumentException();if (workQueue == null || threadFactory == null || handler == null)throw new NullPointerException();this.corePoolSize = corePoolSize;this.maximumPoolSize = maximumPoolSize;this.workQueue = workQueue;this.keepAliveTime = unit.toNanos(keepAliveTime);this.threadFactory = threadFactory;this.handler = handler;}????

在任意時刻,newFixedThreadPool構(gòu)造出來的線程池中最多只可能存活著nThreads個線程,如果所有的線程都在運行任務,那么這個時候提交的任務將會被添加到任務隊列中去等待執(zhí)行。

我們可以控制corePoolSize和maximumPoolSize來使得通過ThreadPoolExecutor構(gòu)造出來的線程池具有一些不一樣的特性,但是需要注意的是,當我們設(shè)置的maximumPoolSize大于corePoolSize的時候,如果當前線程池里面的線程數(shù)量已經(jīng)達到了corePoolSize了,并且當前所以線程都處于運行任務的狀態(tài),那么在這個時候提交的任務會被添加到任務隊列中去,只有在任務隊列滿了的時候,才會去創(chuàng)建新的線程,如果線程數(shù)量已經(jīng)達到了maximumPoolSize了,那么到此就會拒絕提交的任務,這些流程可以參考上面展示出來的execute方法的實現(xiàn)。該類型的線程池使用的任務隊列是LinkedBlockingQueue類型的阻塞隊列。

2、newCachedThreadPool

通過ThreadPoolExecutor構(gòu)造一個newCachedThreadPool線程池的流程如下:

??? public static ExecutorService newCachedThreadPool() {
??????? return new ThreadPoolExecutor(0, Integer.MAX_VALUE,
????????????????????????????????????? 60L, TimeUnit.SECONDS,
????????????????????????????????????? new SynchronousQueue<Runnable>());
??? }

??? public ThreadPoolExecutor(int corePoolSize,
????????????????????????????? int maximumPoolSize,
????????????????????????????? long keepAliveTime,
????????????????????????????? TimeUnit unit,
????????????????????????????? BlockingQueue<Runnable> workQueue) {
??????? this(corePoolSize, maximumPoolSize, keepAliveTime, unit, workQueue,
???????????? Executors.defaultThreadFactory(), defaultHandler);
??? }
?? ?
??? public ThreadPoolExecutor(int corePoolSize,
????????????????????????????? int maximumPoolSize,
????????????????????????????? long keepAliveTime,
????????????????????????????? TimeUnit unit,
????????????????????????????? BlockingQueue<Runnable> workQueue,
????????????????????????????? ThreadFactory threadFactory,
????????????????????????????? RejectedExecutionHandler handler) {
??????? if (corePoolSize < 0 ||
??????????? maximumPoolSize <= 0 ||
??????????? maximumPoolSize < corePoolSize ||
??????????? keepAliveTime < 0)
??????????? throw new IllegalArgumentException();
??????? if (workQueue == null || threadFactory == null || handler == null)
??????????? throw new NullPointerException();
??????? this.corePoolSize = corePoolSize;
??????? this.maximumPoolSize = maximumPoolSize;
??????? this.workQueue = workQueue;
??????? this.keepAliveTime = unit.toNanos(keepAliveTime);
??????? this.threadFactory = threadFactory;
??????? this.handler = handler;
??? }

newCachedThreadPool適合于類似秒殺系統(tǒng)中,它可以按需創(chuàng)建線程。每個線程在空閑了一段時間之后會被回收,然后需要創(chuàng)建的時候再創(chuàng)建出來,在使用的時候應該使用合適的構(gòu)造參數(shù)。

該類型使用的任務隊列是SynchronousQueue這種同步隊列,這是一種特別的隊列,每個線程都是有使命的,每個線程都會等待另外一個線程和自己交易,在交易完成之前都會阻塞住線程,他們之間有一種傳遞關(guān)系,數(shù)據(jù)是從一個線程直接傳遞到例外一個線程中去的,SynchronousQueue這種隊列不存儲實際的數(shù)據(jù),而是存儲著一些線程的信息,而SynchronousQueue管理著這些線程之間的交易,更為詳細的細節(jié)參考后面的文章。

上面提到,ScheduleThreadPoolExecutor是繼承自ThreadPoolExecutor的,而且從類圖中也可以看出來這種關(guān)系,所以其實ScheduleThreadPoolExecutor是對ThreadPoolExecutor的增強,它增加了schedule功能,使用與那些需要周期性調(diào)度執(zhí)行,或者是延時執(zhí)行的任務,在ScheduleThreadPoolExecutor中使用了一種阻塞隊列稱為延時阻塞隊列,這種隊列有能力持有一段時間數(shù)據(jù),我們可以設(shè)定這種時間,時間沒到的時候嘗試獲取數(shù)據(jù)的線程會被阻塞,直到設(shè)定的時間到了,線程才會被喚醒來消費數(shù)據(jù)。而關(guān)于ScheduleThreadPoolExecutor是如何運作的,包括他的周期性任務調(diào)度是如何工作的,可以參考上面提到的鏈接。

Future

Future代表一種未來某個時刻會發(fā)生的事情,在并發(fā)環(huán)境下使用Future是非常重要的,使用Future的前提是我們可以容許線程執(zhí)行一段時間來完成這個任務,但是需要在我們提交了任務的時候就返回一個Future,這樣在接下來的時間程序員可以根據(jù)實際情況來取消任務或者獲取任務,在多個任務沒有相互依賴關(guān)系的時候,使用Future可以實現(xiàn)多線程的并發(fā)執(zhí)行,多個線程可以執(zhí)行在不同的處理器上,然后在某個時間點來統(tǒng)一獲取結(jié)果就可以了。

上文中已經(jīng)提到了FutureTask,FutureTask既是一種Runnable,也是一種Future,并且結(jié)合了兩種類型的特性。下面展示了Future提供的一些方法,使用這些方法可以很方便的進行任務控制:

public interface Future<V> {boolean cancel(boolean mayInterruptIfRunning);boolean isCancelled();boolean isDone();V get() throws InterruptedException, ExecutionException;V get(long timeout, TimeUnit unit)throws InterruptedException, ExecutionException, TimeoutException; }

在java 8中增加了一個新的類CompletableFuture,這是對Future的極大增強,CompletableFuture提供了非常豐富的操作可以來控制我們的任務,并且可以根據(jù)多種規(guī)則來關(guān)聯(lián)多個Future。

Fork/Join框架

Fork/Join框架是一種并行框架,它可以將一個較大的任務切分成一些小任務來執(zhí)行,并且多個線程之間會相互配合,每個線程都會有一個任務隊列,對于某些線程來說它們可能很快完成了自己的任務隊列中的任務,但是其他的線程還沒有完成,那么這些線程就會去竊取那些還沒有完成任務執(zhí)行的線程的任務來執(zhí)行,這成為“工作竊取”算法,關(guān)于Fork/Join中的工作竊取,其實現(xiàn)還是較為復雜的,下面展示了Fork/Join框架的工作模式:

Fork/Join工作模式

可以從上面的圖中看出,一個較大的任務會被切分為一個小任務,并且小任務還會繼續(xù)切分,直到符合我們設(shè)定的執(zhí)行閾值,然后就會執(zhí)行,執(zhí)行完成之后會進行join,也就是將小任務的結(jié)果組合起來,組裝出我們提交的整個任務的結(jié)果,這是一種非常先進的工作模式,非常有借鑒意義。當然,使用Fork/Join框架的前提是我們的任務時可以拆分成小任務來執(zhí)行的,并且小人物的結(jié)果可以組裝出整個大任務的結(jié)果,歸并排序是一種可以借助Fork/Join框架來提供處理速度的算法,下面展示了使用Fork/Join框架來執(zhí)行歸并排序的代碼,可以試著調(diào)整參數(shù)來進行性能測試:

import java.util.Random; import java.util.concurrent.ForkJoinPool; import java.util.concurrent.RecursiveAction;/*** Created by hujian06 on 2017/10/23.** merge sort by fork/join*/ public class ForkJoinMergeSortDemo {public static void main(String ... args) {new Worker().runWork();}}class Worker {private static final boolean isDebug = false;public void runWork() {int[] array = mockArray(200000000, 1000000); // mock the dataforkJoinCase(array);normalCase(array);}private void printArray(int[] arr) {if (isDebug == false) {return;}for (int i = 0; i < arr.length; i ++) {System.out.print(arr[i] + " ");}System.out.println();}private void forkJoinCase(int[] array) {ForkJoinPool pool = new ForkJoinPool();MergeSortTask mergeSortTask = new MergeSortTask(array, 0, array.length - 1);long start = System.currentTimeMillis();pool.invoke(mergeSortTask);long end = System.currentTimeMillis();printArray(array);System.out.println("[for/join mode]Total cost: " + (end - start) / 1000.0 + " s, for " +array.length + " items' sort work.");}private void normalCase(int[] array) {long start = System.currentTimeMillis();new MergeSortWorker().sort(array, 0, array.length - 1);long end = System.currentTimeMillis();printArray(array);System.out.println("[normal mode]Total cost: " + (end - start) / 1000.0 + " s, for " +array.length + " items' sort work.");}private static final? int[] mockArray(int length, int up) {if (length <= 0) {return null;}int[] array = new int[length];Random random = new Random(47);for (int i = 0; i < length; i ++) {array[i] = random.nextInt(up);}return array;} }class MergeSortTask extends RecursiveAction {private static final int threshold = 100000;private final MergeSortWorker mergeSortWorker = new MergeSortWorker();private int[] data;private int left;private int right;public MergeSortTask(int[] array, int l, int r) {this.data = array;this.left = l;this.right = r;}@Overrideprotected void compute() {if (right - left < threshold) {mergeSortWorker.sort(data, left, right);} else {int mid = left + (right - left) / 2;MergeSortTask l = new MergeSortTask(data, left, mid);MergeSortTask r = new MergeSortTask(data, mid + 1, right);invokeAll(l, r);mergeSortWorker.merge(data, left, mid, right);}} }class MergeSortWorker {// Merges two subarrays of arr[].// First subarray is arr[l..m]// Second subarray is arr[m+1..r]void merge(int arr[], int l, int m, int r) {// Find sizes of two subarrays to be mergedint n1 = m - l + 1;int n2 = r - m;/* Create temp arrays */int L[] = new int[n1];int R[] = new int[n2];/*Copy data to temp arrays*/for (int i = 0; i < n1; ++i)L[i] = arr[l + i];for (int j = 0; j < n2; ++j)R[j] = arr[m + 1 + j];/* Merge the temp arrays */// Initial indexes of first and second subarraysint i = 0, j = 0;// Initial index of merged subarry arrayint k = l;while (i < n1 && j < n2) {if (L[i] <= R[j]) {arr[k ++] = L[i ++];} else {arr[k ++] = R[j ++];}}/* Copy remaining elements of L[] if any */while (i < n1) {arr[k ++] = L[i ++];}/* Copy remaining elements of R[] if any */while (j < n2) {arr[k ++] = R[j ++];}}// Main function that sorts arr[l..r] using// merge()void sort(int arr[], int l, int r) {if (l < r) {// Find the middle pointint m = l + (r - l) / 2;// Sort first and second halvessort(arr, l, m);sort(arr, m + 1, r);// Merge the sorted halvesmerge(arr, l, m, r);}} }

在jdk中,使用Fork/Join框架的一個典型案例是Streams API,Streams API試圖簡化我們的并發(fā)編程,可以使用很簡單的流式API來處理我們的數(shù)據(jù)流,在我們無感知的狀態(tài)下,其實Streams的實現(xiàn)上借助了Fork/Join框架來實現(xiàn)了并發(fā)計算,所以強烈建議使用Streams API來處理我們的流式數(shù)據(jù),這樣可以充分的利用機器的多核心資源,來提高數(shù)據(jù)處理的速度。鑒于Fork/Join框架的先進思想,理解并且學會使用Fork/Join框架來處理我們的實際問題是非常有必要的。

Java volatile關(guān)鍵字

volatile解決的問題是多個線程的內(nèi)存可見性問題,在并發(fā)環(huán)境下,每個線程都會有自己的工作空間,每個線程只能訪問各自的工作空間,而一些共享變量會被加載到每個線程的工作空間中,所以這里面就有一個問題,內(nèi)存中的數(shù)據(jù)什么時候被加載到線程的工作緩存中,而線程工作空間中的內(nèi)容什么時候會回寫到內(nèi)存中去。這兩個步驟處理不當就會造成內(nèi)存可加性問題,也就是數(shù)據(jù)的不一致,比如某個共享變量被線程A修改了,但是沒有回寫到內(nèi)存中去,而線程B在加載了內(nèi)存中的數(shù)據(jù)之后讀取到的共享變量是臟數(shù)據(jù),正確的做法應該是線程A的修改應該對線程B是可見的,更為通用一些,就是在并發(fā)環(huán)境下共享變量對多個線程是一致的。
對于內(nèi)存可見性的一點補充是,之所以會造成多個線程看到的共享變量的值不一樣,是因為線程在占用CPU時間的時候,cpu為了提高處理速度不會直接和內(nèi)存交互,而是會先將內(nèi)存中的共享內(nèi)容讀取到內(nèi)部緩存中(L1,L2),然后cpu在處理的過程中就只會和內(nèi)部緩存交互,在多核心的機器中這樣的處理方式就會造成內(nèi)存可見性問題。
volatile可以解決并發(fā)環(huán)境下的內(nèi)存可見性問題,只需要在共享變量前面加上volatile關(guān)鍵字就可以解決,但是需要說明的是,volatile僅僅是解決內(nèi)存可見性問題,對于像i++這樣的問題還是需要使用其他的方式來保證線程安全。使用volatile解決內(nèi)存可見性問題的原理是,如果對被volatile修飾的共享變量執(zhí)行寫操作的話,JVM就會向cpu發(fā)送一條Lock前綴的指令,cpu將會這個變量所在的緩存行(緩存中可以分配的最小緩存單位)寫回到內(nèi)存中去。但是在多處理器的情況下,將某個cpu上的緩存行寫回到系統(tǒng)內(nèi)存之后,其他cpu上該變量的緩存還是舊的,這樣再進行后面的操作的時候就會出現(xiàn)問題,所以為了使得所有線程看到的內(nèi)容都是一致的,就需要實現(xiàn)緩存一致性協(xié)議,cpu將會通過監(jiān)控總線上傳遞過來的數(shù)據(jù)來判斷自己的緩存是否過期,如果過期,就需要使得緩存失效,如果cpu再來訪問該緩存的時候,就會發(fā)現(xiàn)緩存失效了,這時候就會重新從內(nèi)存加載緩存。

總結(jié)一下,volatile的實現(xiàn)原則有兩條:

1、JVM的Lock前綴的指令將使得cpu緩存寫回到系統(tǒng)內(nèi)存中去
2、為了保證緩存一致性原則,在多cpu的情景下,一個cpu的緩存回寫內(nèi)存會導致其他的cpu上的緩存都失效,再次訪問會重新從系統(tǒng)內(nèi)存加載新的緩存內(nèi)容。

原子操作CAS

原子操作表達的意思是要么一個操作成功,要么失敗,中間過程不會被其他的線程中斷,這一點對于并發(fā)編程來說非常重要,在java中使用了大量的CAS來做并發(fā)編程,包括jdk的ConcurrentHsahMap的實現(xiàn),還有AtomicXXX的實現(xiàn)等其他一些并發(fā)工具的實現(xiàn)都使用了CAS這種技術(shù),CAS包括兩部分,也就是Compare and swap,首先是比較,然后再交互,這樣做的原因是,在并發(fā)環(huán)境下,可能不止一個線程想要來改變某個共享變量的值,那么在進行操作之前使用一個比較,而這個比較的值是當前線程認為(知道)該共享變量最新的值,但是可能其他線程已經(jīng)改變了這個值,那么此時CAS操作就會失敗,只有在共享變量的值等于線程提供的用于比較的值的時候才會進行原子改變操作。
java中有一個類是專門用于提供CAS操作支持的,那就是Unsafe類,但是我們不能直接使用Unsafe類,因為Unsafe類提供的一些底層的操作,需要非常專業(yè)的人才能使用好,并且Unsafe類可能會造成一些安全問題,所以不建議直接使用Unsafe類,但是如果想使用Unsafe類的話還是有方法的,那就是通過反射來獲取Unsafe實例,類似于下面的代碼:

class UnsafeHolder {private static Unsafe U = null;public static Unsafe getUnsafe() {if (U == null) {synchronized (UnsafeHolder.class) {if (U == null) {List<Exception> exception = null;try {Field field = Unsafe.class.getDeclaredField("theUnsafe");field.setAccessible(true);try {U = (Unsafe) field.get(null);} catch (IllegalAccessException e) {exception.add(e);}} catch (NoSuchFieldException e) {exception.add(e);} finally {if (exception != null) {reportException(exception);}}}}}return U;}/*** handler the exception in this method .* @param e The exception*/private static void reportException(List<Exception> e) {e.forEach(System.out::println);}}

如果想要了解Unsafe類到底提供了哪些較為底層的操作,可以直接參考Unsafe的源碼。CAS操作解決了原子操作問題,只要進行操作,CAS就會保證操作會成功,不會被中斷,這是一種非常好非常強大的特性,下面就java 8中的ConcurrentHashMap的size實現(xiàn)來談談CAS操作在并發(fā)環(huán)境下的使用案例。
在java 7中,ConcurrentHashMap的實現(xiàn)是基于分段鎖協(xié)議的實現(xiàn),本質(zhì)上還是使用了鎖,只是基于一種考慮,就是多個線程訪問哈希桶具有隨機性,基于這種考慮來將數(shù)據(jù)存儲在不同的哈希段上面,然后每一個段配有一把鎖,在需要寫某個段的時候需要加鎖,而在這個時候,其他訪問其他段的線程是不需要阻塞的,但是對于該段的線程訪問就需要等待,直到這個加鎖的線程釋放了鎖,其他線程才能進行訪問。在java 8中,ConcurrentHashMap的實現(xiàn)拋棄了這種復雜的架構(gòu)設(shè)計,但是繼承了這種分散線程競爭壓力的思想,其實就提高系統(tǒng)的并發(fā)度這一維度來說,分散競爭壓力是一種最為直接明了的解決方案,而java 8在實現(xiàn)ConcurrentHashMap的時候大量使用了CAS操作,減少了使用鎖的頻度來提高系統(tǒng)的響應度,其實使用鎖和使用CAS來做并發(fā)在復雜度上不是一個數(shù)量級的,使用鎖在很大程度上假設(shè)了多個線程的排斥性,并且使用鎖會將線程阻塞等待,也就是說使用鎖來做線程同步的時候,線程的狀態(tài)是會改變的,但是使用CAS是不會改變線程的狀態(tài)的(不太嚴謹?shù)恼f),所以使用CAS比起使用synchronized或者使用Lcok來說更為輕量級。
現(xiàn)在就ConcurrentHashMap的size方法來分析一下如何將線程競爭的壓力分散出去。在java 7的實現(xiàn)上,在調(diào)用size方法之后,ConcurrentHashMap會進行兩次對哈希桶中的記錄累加的操作,這兩次累加的操作是不加鎖的,然后判斷兩次結(jié)果是否一致,如果一致就說明目前的系統(tǒng)是讀多寫少的場景,并且可能目前沒有線程競爭,所以直接返回就可以,這就避免了使用鎖,但是如果兩次累加結(jié)果不一致,那就說明此時可能寫的線程較多,或者線程競爭較為嚴重,那么此時ConcurrentHashMap就會進行一個重量級的操作,對所有段進行加鎖,然后對每一個段進行記錄計數(shù),然后求得最終的結(jié)果返回。在最有情況下,size方法需要做兩次累加計數(shù),最壞情況需要三次,并且會涉及全局加鎖這種重量級的加鎖操作,性能肯定是不高的。而在java 8的實現(xiàn)上,ConcurrentHashMap的size方法實際上是與ConcurrentHashMap是解耦的,size方法更像是接入了一個額外的并發(fā)計數(shù)系統(tǒng),在進行size方法調(diào)用的時候是不會影響數(shù)據(jù)的存取的,這其實是一種非常先進的思想,就是一個系統(tǒng)模塊化,然后模塊可以進行更新,系統(tǒng)解耦,比如java 8中接入了并發(fā)計數(shù)組件Striped64來作為size方法的支撐,可能未來出現(xiàn)了比Striped64更為高效的算法來計數(shù),那么只需要將Striped64模塊換成新的模塊就可以了,對原來的核心操作是不影響的,這種模塊化系統(tǒng)設(shè)定的思想應該在我們的項目中具體實踐。
上面說到java 8在進行size方法的設(shè)計上引入了Striped64這種并發(fā)計數(shù)組件,這種組件的計數(shù)思想其實也是分散競爭,Striped64的實現(xiàn)上使用了volatile和CAS,在Striped64的實現(xiàn)中是看不到鎖的使用的,但是Striped64確實是一種高效的適用于并發(fā)環(huán)境下的計數(shù)組件,它會基于請求計數(shù)的線程,Striped64的計數(shù)會根據(jù)兩部分的內(nèi)容來得到最后的結(jié)果,類似于java 7中ConcurrentHashMap的size方法的實現(xiàn),在Striped64的實現(xiàn)上也是借鑒了這種思想的,Striped64會首先嘗試將某個線程的計數(shù)請求累加到一個base共享變量上,如果成功了,那么說明目前的競爭不是很激烈,也就沒必要后面的操作了,但是很多情況下,并發(fā)環(huán)境下的線程競爭是很激烈的,所以嘗試累加到base上的計數(shù)請求很大概率是會失敗的,那么Striped64會維護一個Cell數(shù)組,每個Cell是一個計數(shù)組件,Striped64會為每個請求計數(shù)的線程計算一個哈希值,然后哈希到Cell數(shù)組中的某個位置上,然后這個線程的計數(shù)就會累加到該Cell上面去。

并發(fā)同步框架AQS

AQS是java中實現(xiàn)Lock的基礎(chǔ),也是實現(xiàn)線程同步的基礎(chǔ),AQS提供了鎖的語義,并且支持獨占模式和共享模式,對應于悲觀鎖和樂觀鎖,獨占模式的含義是說同一時刻只能有一個線程獲取鎖,而其他試圖獲取鎖的線程都需要阻塞等待,而共享鎖的含義是說可以有多個線程獲得鎖,兩種模式在不同的場景下使用。
而鎖在并發(fā)編程中的地位不言而喻,多個線程的同步很多時候是需要鎖來做同步的,比如對于某些資源,我們希望可以有多個線程獲得鎖來讀取,但是只允許有一個線程獲得鎖來執(zhí)行寫操作,這種鎖稱為讀寫鎖,它的實現(xiàn)上結(jié)合了AQS的共享模式和獨占模式,共享模式對應于可以使得多個線程獲得鎖來進行讀操作,獨占模式對應于只允許有一個線程獲得鎖來進行寫操作。該文章詳細講述了多個Lock接口的實現(xiàn)類,以及他們是如何借助AQS來實現(xiàn)的具體細節(jié)。
某些時候,我們需要定制我們自己的線程同步策略,個性化的線程同步借助AQS可以很容易的實現(xiàn),比如我們的需求是允許限定個數(shù)的線程獲得鎖來進行一些操作,想要實現(xiàn)這樣的語義,只需要實現(xiàn)一個類,繼承AQS,然后重寫方法下面兩個方法:??

protected boolean tryAcquire(int arg) {throw new UnsupportedOperationException();}protected boolean tryRelease(int arg) {throw new UnsupportedOperationException();}????

還需要提到的一點是,鎖分為公平鎖和非公平鎖,java中大多數(shù)時候會使用隊列來實現(xiàn)公平鎖,而使用棧來實現(xiàn)非公平鎖,當然這是基于隊列和棧這兩種數(shù)據(jù)結(jié)構(gòu)的特點來實現(xiàn)的,直觀的來說,使用隊列的FIFO的特性就可以實現(xiàn)類似排隊的效果,也就保證了公平性,而棧是一個后進先出的數(shù)據(jù)結(jié)構(gòu),它的這種結(jié)構(gòu)造成的結(jié)果就是,最新進入的線程可能比那些等待過一段時間的線程更早的獲得鎖,更為具體的內(nèi)容可以參考上面的文章進行了解。

synchronized(同步鎖)

相對于volatile,synchronized就顯得比較重量級了。

首先,我們應該知道,在java中,所有的對象都可以作為鎖。可以分為下面三種情況:

1、普通方法同步,鎖是當前對象
2、靜態(tài)方法同步,鎖是當前類的Class對象
3、普通塊同步,鎖是synchronize里面配置的對象

當一個線程試圖訪問同步代碼時,必須要先獲得鎖,退出或者拋出異常時必須要釋放鎖。

JVM基于進入和退出Monitor對象來實現(xiàn)方法同步和代碼塊同步,可以使用monitorenter和monitorexit指令實現(xiàn)。monitorenter指令是在編譯后插入到同步代碼塊的開始位置,而monitorexit指令則插入到方法結(jié)束和異常處,JVM保證每個monitorenter都有一個monitorexit閾值相對應。線程執(zhí)行到monitorenter的時候,會嘗試獲得對象所對應的monitor的鎖,然后才能獲得訪問權(quán)限,synchronize使用的鎖保存在Java對象頭中。

并發(fā)隊列(阻塞隊列,同步隊列)

并發(fā)隊列,也就是可以在并發(fā)環(huán)境下使用的隊列,為什么一般的隊列不能再并發(fā)環(huán)境下使用呢?因為在并發(fā)環(huán)境下,可能會有多個線程同時來訪問一個隊列,這個時候因為上下文切換的原因可能會造成數(shù)據(jù)不一致的情況,并發(fā)隊列解決了這個問題,并且java中的并發(fā)隊列的使用時非常廣泛的,比如在java的線程池的實現(xiàn)上使用了多種不同特性的阻塞隊列來做任務隊列,對于阻塞隊列來說,它要解決的首要的兩個問題是:
1. 多線程環(huán)境支持,多個線程可以安全的訪問隊列
2. 支持生產(chǎn)和消費等待,多個線程之間互相配合,當隊列為空的時候,消費線程會阻塞等待隊列不為空;當隊列滿了的時候,生產(chǎn)線程就會阻塞直到隊列不滿。
Java中提供了豐富的并發(fā)隊列實現(xiàn),下面展示了這些并發(fā)隊列的概覽:

java并發(fā)隊列概覽

根據(jù)上面的圖可以將java中實現(xiàn)的并發(fā)隊列分為幾類:

1. 一般的阻塞隊列
2. 支持雙端存取的并發(fā)隊列
3. 支持延時獲取數(shù)據(jù)的延時阻塞隊列
4. 支持優(yōu)先級的阻塞隊列
這些隊列的區(qū)別就在于從隊列中存取數(shù)據(jù)時的具體表現(xiàn),比如對于延時隊列來說,獲取數(shù)據(jù)的線程可能被阻塞等待一段時間,也可能立刻返回,對于優(yōu)先級阻塞隊列,獲取的數(shù)據(jù)是根據(jù)一定的優(yōu)先級取到的。下面展示了一些隊列操作的具體表現(xiàn):

  • Throws Exception 類型的插入和取出在不能立即被執(zhí)行的時候就會拋出異常。

  • Special Value 類型的插入和取出在不能被立即執(zhí)行的情況下會返回一個特殊的值(true 或者 false)

  • Blocked 類型的插入和取出操作在不能被立即執(zhí)行的時候會阻塞線程直到可以操作的時候會被其他線程喚醒

  • Timed out 類型的插入和取出操作在不能立即執(zhí)行的時候會被阻塞一定的時候,如果在指定的時間內(nèi)沒有被執(zhí)行,那么會返回一個特殊值

總結(jié)

本文總結(jié)了Java并發(fā)編程中的若干核心技術(shù),并且對每一個核心技術(shù)都做了一些分析,并給出了參考鏈接,可以在參考鏈接中查找到更為具體深入的分析總結(jié)內(nèi)容。

Java并發(fā)編程需要解決一些問題,比如線程間同步問題,如何保證數(shù)據(jù)可見性問題,以及如何高效的協(xié)調(diào)多個線程工作等內(nèi)容,本文在這些維度上都有所設(shè)計。

本文作為對閱讀java.util.Concurrent包的源碼閱讀的一個總結(jié),同時本文也作為一個起點,一個開始更高層次分析總結(jié)的起點,之前的分析都是基于JDK源碼來進行的,并且某些細節(jié)的內(nèi)容還沒有完全搞明白,其實在閱讀了一些源碼之后就會發(fā)現(xiàn)。

如果想要深入分析某個方面的內(nèi)容,就需要一些底層的知識,否則很難完整的分析總結(jié)出來,但是這種不徹底的分析又是很有必要的,至少可以對這些內(nèi)容有一些大概的了解,并且知道自己的不足,以及未來需要了解的底層內(nèi)容。

對于Java并發(fā)包的分析研究,深入到底層就是對JVM如何管理內(nèi)容,如何管理線程的分析,在深入下去,就是操作系統(tǒng)對內(nèi)存的管理,對線程的管理等內(nèi)容,從操作系統(tǒng)再深入下去,就是去理解CPU的指令系統(tǒng),學習磁盤知識等內(nèi)容。

當然,知識的關(guān)聯(lián)是無止境的,學習也是無止境的,目前來說,首要解決的問題是可以熟練的使用Java提供的并發(fā)包內(nèi)容來進行并發(fā)編程,在業(yè)務上提高并發(fā)處理能力,在出現(xiàn)問題的時候可以很快找到問題并且解決問題,在達到這個要求之后,可以去了解一些JVM層次的內(nèi)容,比如JVM的內(nèi)存模型,以及線程的實現(xiàn),并且可以與學習操作系統(tǒng)的相關(guān)內(nèi)容并行進行。

總結(jié)

以上是生活随笔為你收集整理的Java并发编程中的若干核心技术,向高手进阶的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。

如果覺得生活随笔網(wǎng)站內(nèi)容還不錯,歡迎將生活随笔推薦給好友。

综合激情av| 日本精品一区二区三区在线观看 | 久久久久福利视频 | 91精品视频在线 | 国产精品a成v人在线播放 | 欧美在线日韩在线 | 日韩av一区二区三区在线观看 | 免费成人黄色 | 91精品亚洲影视在线观看 | 久久久久久久久久国产精品 | 99久久99久久精品免费 | 友田真希x88av | 婷婷 综合 色 | 美女久久久久久久久久久 | 91亚洲精品国偷拍自产在线观看 | 久久av中文字幕片 | 人人爽人人射 | 一区二区视 | 久久久久久看片 | 91精品系列 | 国产在线精品二区 | 手机av电影在线 | 91在线中字 | 久二影院 | 日韩高清av | 中文字幕之中文字幕 | 97国产在线播放 | 日本黄色免费网站 | 免费视频你懂的 | 久草com | 欧美精品一区二区三区四区在线 | 精品国产伦一区二区三区观看说明 | 国产免费av一区二区三区 | 欧美一级艳片视频免费观看 | 九九热免费精品视频 | 天堂av高清 | 好看av在线 | 91免费在线播放 | 国产色女| 精品在线观看一区二区 | 国产成人三级 | 免费观看9x视频网站在线观看 | 亚洲国产精品va在线看黑人动漫 | 久久综合九色欧美综合狠狠 | 国产美女视频免费 | 久久久亚洲影院 | 国产午夜精品理论片在线 | 97av影院 | 国产精品黄色影片导航在线观看 | 在线观看亚洲精品 | 国产九九热视频 | 成人h视频在线播放 | 日韩免费观看一区二区 | 一区中文字幕电影 | 国内外激情视频 | 亚洲va欧美va人人爽春色影视 | 国产精品每日更新 | 2019精品手机国产品在线 | av看片在线观看 | 免费观看成人av | 91成人国产 | 日日夜夜艹 | 久久精品视频免费 | www.亚洲激情.com | 美女视频黄免费网站 | 国产成人在线播放 | 麻花豆传媒mv在线观看 | 玖操| 人人干人人搞 | 最近中文字幕免费视频 | av在线免费观看黄 | 国产中文字幕视频在线 | 精品欧美乱码久久久久久 | 欧美久久久久久久 | 日韩手机在线 | 天天综合网在线观看 | 热re99久久精品国产66热 | 国产精品伦一区二区三区视频 | 国产尤物一区二区三区 | 成年人视频在线免费观看 | 亚洲天堂自拍视频 | 欧美巨大荫蒂茸毛毛人妖 | 免费看国产精品 | 久久精品一二区 | 国产一区自拍视频 | 91伊人久久大香线蕉蜜芽人口 | www四虎影院| 91在线视频一区 | 成人观看 | 成人av在线网址 | 成人夜晚看av | 国产99视频在线观看 | 久久久久高清毛片一级 | 婷婷色站 | 免费成人在线观看 | 久草视频免费在线观看 | 日韩在线免费不卡 | 一色屋精品视频在线观看 | 超碰av在线播放 | 国产黄色一级片 | 亚洲成人在线免费 | 日韩久久视频 | 色综合中文字幕 | 成人97视频一区二区 | 99精彩视频在线观看免费 | 一区二区视频欧美 | 黄免费在线观看 | 久久理论视频 | 国产黄在线 | 人人超碰97 | 久草观看 | 免费日韩电影 | 国产91综合一区在线观看 | 久久久久免费看 | 五月天堂色 | 麻豆久久一区 | 涩涩网站在线看 | 欧美日韩伦理在线 | 亚州视频在线 | 精品一区二区亚洲 | 久久精品黄 | 综合亚洲视频 | 91经典在线| 精品国产激情 | 欧美激情视频在线观看免费 | 婷婷久久婷婷 | 国产精品久久久久久久久久久久久 | 粉嫩一二三区 | 免费国产在线精品 | 婷婷开心久久网 | 亚洲日本黄色 | 天天干天天色2020 | 国产亚洲精品中文字幕 | 久久精品一区二 | 在线日韩精品视频 | 日本中文不卡 | 日韩三级不卡 | 亚洲国产一区av | 久久大片网站 | 国产精品麻豆视频 | 成人精品视频久久久久 | 在线国产能看的 | 精品久久久久久久 | 亚洲视频免费在线观看 | 日韩欧美高清一区二区 | 人人澡人人模 | 精品国模一区二区 | 精品久久一级片 | 成人av一区二区兰花在线播放 | 69av视频在线 | av在线免费网站 | 亚洲视频久久久 | 免费av网站观看 | 成人国产电影在线观看 | 日韩免费视频观看 | 久久精品视频免费观看 | 精品99免费| 天天操天天射天天爱 | av资源免费在线观看 | 香蕉视频在线视频 | av网在线观看 | 久久久久国产精品午夜一区 | 精品国产一区二区三区四区vr | 一区二区三区免费网站 | 国产理伦在线 | 在线观看日本高清mv视频 | 欧美一二三在线 | 狠狠干2018| 精品国产_亚洲人成在线 | 中文字幕一区二区三区久久 | 日韩精品一区二区三区第95 | 婷婷色站| 天天躁天天狠天天透 | 免费一级片视频 | 不卡av在线免费观看 | 精品国产免费观看 | 91精品欧美一区二区三区 | 在线观看黄网站 | 久久不卡日韩美女 | 综合铜03| 午夜国产福利在线观看 | 色哟哟国产精品 | 久久精品亚洲一区二区三区观看模式 | av电影中文 | 欧美淫aaa免费观看 日韩激情免费视频 | 在线播放亚洲 | 成年人视频在线免费 | 日韩av片无码一区二区不卡电影 | 黄色一级大片在线免费看产 | 免费高清av在线看 | 五月天亚洲婷婷 | 久久免费成人精品视频 | 国产精品白丝jk白祙 | 精品一区二区在线看 | 国产精品资源网 | 免费中文字幕 | 免费亚洲黄色 | av免费观看在线 | 国产亚洲欧美日韩高清 | 在线不卡a| 免费色视频在线 | 黄色特一级 | 很污的网站 | 日韩高清不卡一区二区三区 | 成年人免费观看国产 | 亚洲国产精品va在线看黑人动漫 | 96亚洲精品久久久蜜桃 | 日韩深夜在线观看 | 亚洲伊人成综合网 | 五月天堂网| 亚洲精品久久激情国产片 | 成人av在线影院 | 久久精品视频网址 | 成人91av | 伊人资源视频在线 | 久久视频免费在线观看 | 香蕉网址 | 国产a精品 | 日韩高清免费无专码区 | 亚洲美女免费精品视频在线观看 | 999久久久久久久久久久 | 亚洲黄色在线观看 | 国产一线二线三线在线观看 | 中文字幕视频 | 国产高清在线视频 | 国色天香av| 亚洲激情av | 精品视频123区在线观看 | 亚洲国产天堂av | 国产中文字幕视频在线观看 | av在线精品| 亚州人成在线播放 | 韩国一区二区三区在线观看 | 国产91影视 | 天天狠狠 | 亚洲人成网站精品片在线观看 | 成人黄色小说在线观看 | av不卡网站 | 久久精品一区二 | av黄色免费网站 | 五月婷婷影视 | 91日韩精品视频 | 久久成人午夜视频 | 国产精品一区电影 | 亚洲精品久久久久999中文字幕 | 美女视频又黄又免费 | 国产人在线成免费视频 | 国产麻豆果冻传媒在线观看 | 国内精品久久久久久久久久清纯 | 99久久精品免费看国产 | 国产精品观看视频 | 国产乱对白刺激视频在线观看女王 | 久久九九影视 | 欧美一二三在线 | 天天爽天天爽夜夜爽 | 国产成人久久77777精品 | 一区三区在线欧 | 在线免费高清视频 | 中文字幕一二三区 | 亚洲成人家庭影院 | 91视频黄色 | 日日爽天天操 | 青青久草在线 | 日韩激情视频 | 日韩啪啪小视频 | 国产在线色站 | 日韩电影久久 | 天天拍天天爽 | 国产精品久久久久久吹潮天美传媒 | 夜夜视频资源 | 国产视频 亚洲视频 | 亚洲精品国产日韩 | 久久婷婷视频 | 综合伊人久久 | 国产一级二级在线 | 国产精品毛片一区视频播 | 久久网站最新地址 | 国产精品岛国久久久久久久久红粉 | 色橹橹欧美在线观看视频高清 | 又黄又刺激的网站 | 成人动漫一区二区三区 | 日日躁天天躁 | 久久久久久久久久久精 | 久久久18| 国产精品99久久久久久宅男 | 丁香综合av | 亚洲天堂色婷婷 | 久久亚洲美女 | 精选久久| 91理论片午午伦夜理片久久 | 在线精品视频免费观看 | 国内精品久久久精品电影院 | 精品国产乱码一区二区三区在线 | 亚洲精品国偷拍自产在线观看蜜桃 | 日本久久综合视频 | 国产高清视频免费在线观看 | 亚洲黄色一级大片 | 欧美日韩一区三区 | 在线观看完整版 | 欧美在线观看小视频 | 久久综合五月天婷婷伊人 | 国产精品99久久久精品 | 激情久久伊人 | 中文字幕在线专区 | 久久伦理电影网 | 91视频这里只有精品 | 欧美久久综合 | 亚洲国产精品电影在线观看 | 国产一二区免费视频 | 久久免费视频8 | 欧美日韩久久 | 欧美精品久久久久性色 | 欧美日韩天堂 | 天天操夜| 日日干激情五月 | 精品久久久久免费极品大片 | 最近最新mv字幕免费观看 | 免费碰碰 | 国产91在线观看 | 日韩精品久久中文字幕 | 成人午夜电影久久影院 | 狠狠五月天 | 久久免费精彩视频 | 国产精品久久久久永久免费观看 | 久久久国产一区二区三区四区小说 | 成人精品一区二区三区中文字幕 | 久久久久黄 | 婷婷激情在线 | 蜜桃传媒一区二区 | 国产一级久久久 | 夜夜躁狠狠躁日日躁 | 视频在线观看91 | 国产亚洲无 | 日韩视频免费观看高清完整版在线 | 久久久免费少妇 | 五月婷婷久久丁香 | 天天色天天色天天色 | 西西www4444大胆在线 | 国产福利专区 | 在线观看视频99 | 久久久国产影院 | 成人午夜性影院 | 超碰在线观看av.com | 国产日韩在线看 | 国产一级性生活视频 | 亚洲最新视频在线播放 | 99久久日韩精品免费热麻豆美女 | 久久99国产精品久久 | 69亚洲视频 | 超碰人人射 | 91污在线| 免费三级骚 | 91黄色在线看 | 亚洲精品1区2区3区 超碰成人网 | 欧美久久久久久久久久久 | 天天天干天天射天天天操 | 国产一区免费观看 | 天堂av在线网址 | 成人超碰97 | 中文字幕在线观看你懂的 | 六月丁香在线观看 | 国产成人一二三 | 国产亚洲视频中文字幕视频 | 国产在线无 | 美女视频黄色免费 | av不卡免费在线观看 | 免费高清看电视网站 | 国产精品久久久久久久久久免费看 | 中文字幕二区在线观看 | 久久精品99国产精品日本 | 成人黄色电影在线观看 | 日本黄色a级大片 | 欧美一级艳片视频免费观看 | 亚洲精品美女久久久久 | 超碰在线亚洲 | 国产一级久久久 | 久久人人爽爽人人爽人人片av | 99久国产 | 国产精品96久久久久久吹潮 | 一色屋精品视频在线观看 | 精品国产一区二区三区蜜臀 | 国模精品一区二区三区 | 91亚洲精品乱码久久久久久蜜桃 | 中文字幕在线观看一区二区 | 黄色大全免费网站 | 五月激情丁香 | 久久高清av | x99av成人免费| 韩国av一区二区三区在线观看 | 久久精品国产亚洲a | 久久精品视频4 | 久久国产精品第一页 | 一本一道久久a久久精品蜜桃 | 午夜精品久久久久久久99热影院 | 久色免费视频 | 久久久久亚洲精品男人的天堂 | 久久国产精品色婷婷 | 国产 日韩 欧美 中文 在线播放 | 久草免费在线 | 激情av综合 | 天天干天天干天天干天天干天天干天天干 | 国产91精品久久久久久 | 亚洲最大成人网4388xx | 九九久久久久久久久激情 | 69精品在线 | 免费观看黄色12片一级视频 | 国产黑丝袜在线 | 国产精品久久久久久欧美 | 亚洲精品在线二区 | 在线观看免费av网 | 国产人成在线观看 | 欧美日产一区 | 丁香综合av | 欧美日韩亚洲第一页 | 人人看人人爱 | 国产成人精品亚洲a | 午夜影视av | 99精品国产一区二区 | 狠狠干狠狠色 | 91九色综合 | 色偷偷网站视频 | 亚洲国产精品成人女人久久 | 91在线资源 | 日韩试看 | 国产一区不卡在线 | 国产精品正在播放 | 亚洲精品国产精品99久久 | av在线最新| 婷婷丁香导航 | 中文字幕av在线不卡 | 奇米影视777影音先锋 | 蜜臀久久99精品久久久酒店新书 | 免费特级黄色片 | 九七视频在线 | 国产精品不卡 | 欧美一级性视频 | 色综合激情久久 | 亚洲人精品午夜 | 国产午夜精品一区二区三区欧美 | 亚洲高清久久久 | 免费男女羞羞的视频网站中文字幕 | 日韩一区二区三免费高清在线观看 | 日日日网| 主播av在线 | 韩国av电影网 | 男女靠逼app | 国产在线欧美日韩 | 欧美一级日韩三级 | av网站大全免费 | 视频一区二区国产 | 久草在线资源观看 | 黄色成人av在线 | 美女视频又黄又免费 | 精品国内| 在线视频1卡二卡三卡 | 久久在线视频精品 | 亚洲一级二级 | 狠狠gao | 国内成人精品视频 | 黄色毛片电影 | 在线之家免费在线观看电影 | 美女又爽又黄 | 婷婷色伊人 | 91亚色免费视频 | 国产精品久久久久免费a∨ 欧美一级性生活片 | 狠狠操狠狠干天天操 | 欧美一区二视频在线免费观看 | 六月丁香六月婷婷 | 亚洲国产人午在线一二区 | 国产不卡免费 | 天堂av一区二区 | 精品久久久久国产免费第一页 | 亚洲精品久久在线 | 久草精品视频 | www.黄色| 久久不色 | av福利第一导航 | 99久热在线精品视频观看 | 久久草草热国产精品直播 | 国产在线永久 | 国产成人精品亚洲a | 精品福利在线视频 | 毛片网站免费 | 日韩av电影免费观看 | 99久高清在线观看视频99精品热在线观看视频 | www.狠狠操.com | 在线有码中文字幕 | 国产精品12 | 毛片久久久 | 国产五月| 国产不卡精品视频 | 人人讲下载 | 国产 在线 日韩 | 精品二区视频 | 国产精品久久在线 | 亚洲a成人v| 欧美一区二区三区免费观看 | 国产一级免费观看视频 | 国产91勾搭技师精品 | 久热爱| 亚洲一二区精品 | 狠狠色婷婷丁香六月 | 久久国产视频网站 | 天天操天 | 久草网站| 久久久久久网站 | 天天射天天艹 | 日韩国产精品一区 | 五月开心色 | 免费无遮挡动漫网站 | 欧美日韩综合在线 | 久久久久国产精品一区 | 精品福利片 | 人人爱人人射 | 久久国产99 | 免费看日韩片 | 亚洲国产精品小视频 | 国产私拍在线 | 成人免费亚洲 | 男女视频91| 成人av高清在线观看 | 在线看免费 | 九九涩涩av台湾日本热热 | 国产成人亚洲在线电影 | 日韩久久精品一区二区 | 国产黄色看片 | 国产麻豆精品久久一二三 | 婷婷丁香激情综合 | 国产精品二区在线观看 | 亚洲精品一区中文字幕乱码 | 中文字幕国产精品一区二区 | 国产麻豆精品免费视频 | 久久综合九色综合97_ 久久久 | 五月婷香蕉久色在线看 | 人人干人人上 | 日韩视频免费 | 亚洲极色 | 免费看的黄色小视频 | 日日综合网 | 亚洲专区中文字幕 | 天天插天天狠天天透 | 久久精品高清 | 91麻豆精品国产 | 中文字幕在线播放一区 | 国产999精品久久久 免费a网站 | 久久久午夜剧场 | 日本99久久| 五月婷婷六月丁香 | 99精品视频免费 | 国产中文字幕一区二区三区 | 亚洲爱爱视频 | 麻豆视频免费播放 | 国产成人黄色 | a级国产乱理论片在线观看 特级毛片在线观看 | 六月婷婷久香在线视频 | 久久资源总站 | 激情 一区二区 | 久久视频6| 欧美在线资源 | 在线观看 国产 | 97视频入口免费观看 | 日韩精品免费一区二区在线观看 | 久久免费av| 丁香电影小说免费视频观看 | 亚洲精品久久久久中文字幕m男 | 国产精品久久久久久久久蜜臀 | 国产精品视频一二三 | 人人澡超碰碰 | 欧美一级日韩免费不卡 | 日韩午夜小视频 | 九九免费精品视频在线观看 | 天堂av在线 | 久要激情网 | 精品国产乱码一区二区三区在线 | 国产免费又爽又刺激在线观看 | 免费日韩电影 | 天天操天天干天天综合网 | 午夜影院在线观看18 | 2024国产在线 | 五月婷婷久久综合 | 久久久久久久影视 | 色999五月色| 免费看的黄色小视频 | 国产在线欧美 | 手机在线日韩视频 | 欧美日韩午夜爽爽 | 精品国产aⅴ麻豆 | 国产正在播放 | 91视频免费看网站 | a极黄色片 | 欧美午夜寂寞影院 | 国产明星视频三级a三级点| 欧美性生活免费 | 日韩欧美在线国产 | 91人人爽久久涩噜噜噜 | 国产精品18久久久久久不卡孕妇 | 综合久久影院 | 天天操操操操操操 | 麻豆国产精品视频 | 91精品在线视频观看 | 婷婷综合av | 91精品免费在线观看 | 国产午夜麻豆影院在线观看 | 91精品国自产在线观看欧美 | 91视频传媒 | 日韩精品一区二区三区电影 | 91麻豆精品久久久久久 | 中文字幕中文字幕中文字幕 | av 在线观看 | 国产精品成人免费精品自在线观看 | 激情图片qvod | 欧美日韩在线观看一区二区 | 亚洲黄色在线观看 | 成年人国产在线观看 | 国产精国产精品 | 亚洲视频在线播放 | 欧美日本一区 | 欧美日韩国产二区三区 | 久久精品这里热有精品 | 在线观看的a站 | 成人在线电影观看 | 久久久久国产成人精品亚洲午夜 | 爱射综合| 久久九精品 | 99成人免费视频 | 日本午夜在线亚洲.国产 | 日韩在线视频免费播放 | 日日操夜 | av在线网站观看 | 久久伊人国产精品 | 成人午夜剧场在线观看 | 久久久综合香蕉尹人综合网 | 精品99在线 | 97精品国产97久久久久久春色 | 亚洲黄色在线 | 99视频在线看 | 97国产| 91亚洲影院| 免费观看的av网站 | 色九九影院| 缴情综合网五月天 | 亚洲黄色高清 | 亚洲自拍偷拍色图 | 久久中国精品 | 91色在线观看 | 国产日韩av在线 | 干干操操 | 91超碰在线播放 | 西西4444www大胆艺术 | av成人在线播放 | 激情五月***国产精品 | 黄色影院在线播放 | 国产精品久久av | 最近中文字幕完整视频高清1 | 热久精品 | av中文在线观看 | 激情综合网天天干 | 国产高清一区二区 | 91福利视频久久久久 | 国产福利在线免费 | 欧美精品久久久久久久久久白贞 | 免费在线一区二区三区 | 欧美日一级片 | 国产91丝袜在线播放动漫 | 91精品国产一区 | av一级在线 | 91在线精品播放 | 丁香 久久 综合 | 日韩黄色免费看 | 日韩成人精品一区二区 | 波多野结衣一区二区三区中文字幕 | 精品黄色视 | 久久久www成人免费毛片 | 亚色视频在线观看 | 四虎影视www| 久久艹精品 | 久草精品视频在线观看 | 国产一区二区在线免费 | 九色精品免费永久在线 | 色噜噜噜噜 | 久久免费精品一区二区三区 | 在线观看一区二区视频 | 精品视频免费观看 | 天天操天天干天天操天天干 | 精品一区二区三区在线播放 | 免费看片成年人 | 国产伦精品一区二区三区在线 | 久久亚洲美女 | 久久精品伊人 | www.久久视频 | 亚洲精选99 | 国产精品伦一区二区三区视频 | 国产精品久久99综合免费观看尤物 | 欧美日韩高清不卡 | 成人在线播放视频 | 久久亚洲福利 | a级国产乱理论片在线观看 伊人宗合网 | 超碰人人av | 免费成人在线观看 | 国产亚洲精品久 | 色狠狠干 | 亚洲成人精品久久 | 国产色a在线观看 | 日韩免费视频线观看 | 在线观看成人小视频 | 成人影视免费看 | 成人免费在线视频观看 | 日韩av美女 | 一本一道久久a久久精品蜜桃 | 国产91探花 | 成人中文字幕在线观看 | 综合激情av | 国产成人精品电影久久久 | www.夜夜 | 国产在线色站 | 91视频午夜 | 涩涩网站在线观看 | 91豆花在线 | 国产精品麻豆免费版 | 午夜精品一区二区三区在线观看 | 欧美激情在线看 | 国产剧情一区二区 | 国产亚洲va综合人人澡精品 | 日日夜夜综合网 | 日韩综合一区二区三区 | 视频91 | 在线小视频你懂得 | 97超碰网| 成年人在线免费视频观看 | 国产 欧美 在线 | 狠狠色狠狠综合久久 | 成人av一区二区兰花在线播放 | 亚洲 精品在线视频 | 久操中文字幕在线观看 | 精品主播网红福利资源观看 | 国产精品99久久久久的智能播放 | 欧美激情精品一区 | 不卡电影免费在线播放一区 | 99久久久国产精品免费观看 | 国产一区二区观看 | 久久电影网站中文字幕 | 九九视频网 | 综合久久网站 | 欧美片网站yy | av中文国产 | 久久久久久久久久网站 | 国产精品永久免费视频 | 插久久| 亚洲在线资源 | 在线免费黄色av | 在线观看亚洲电影 | 97碰碰精品嫩模在线播放 | 91麻豆传媒| 麻豆成人网 | 韩国一区二区三区视频 | 狠狠躁夜夜躁人人爽超碰97香蕉 | 四虎国产精品成人免费4hu | 99视频+国产日韩欧美 | 久久久久免费电影 | 久久国产网 | 亚洲丝袜一区二区 | 国产97免费 | 九9热这里真品2 | 日韩av中文| www.黄色片网站 | 亚洲精品久久久蜜桃直播 | 久久精品伊人 | 黄色影院在线免费观看 | 国产精品久久一区二区三区不卡 | 婷婷丁香狠狠爱 | 美女视频黄是免费的 | 国产特级毛片aaaaaa高清 | 欧洲高潮三级做爰 | 久久国产二区 | 深夜免费网站 | 丁香综合| 国产成人久久精品一区二区三区 | 亚洲免费在线视频 | 亚洲欧美成人网 | 中文字幕av影院 | 亚洲成人精品影院 | 九月婷婷色 | 91精品啪在线观看国产81旧版 | 国产一级h| 99热这里有精品 | 日韩在线播放欧美字幕 | 亚州精品一二三区 | 激情综合网婷婷 | 成人三级黄色 | 久草在线网址 | 久久精品成人欧美大片古装 | 欧美午夜视频在线 | 久久伊人精品一区二区三区 | 日韩精品视频网站 | 国产一区视频在线播放 | 亚洲午夜激情网 | 91大神精品视频在线观看 | 国产成人黄色 | 91福利试看 | 国内成人精品2018免费看 | 97超碰人人爱| www.伊人网 | 亚洲精品中文在线观看 | 中文字幕亚洲在线观看 | 精品国产一区二区三区久久久久久 | 久久九九免费视频 | 国产在线色视频 | 麻豆91精品视频 | 偷拍区另类综合在线 | 国产高清福利在线 | 一区二区免费不卡在线 | av电影中文字幕在线观看 | 色综合天天综合网国产成人网 | 91精品免费看 | 国产精品久久久久一区二区国产 | 久久草视频 | 久久国内精品视频 | 免费色婷婷 | 精品在线免费视频 | 日韩高清不卡一区二区三区 | 91在线视频观看免费 | 日韩精品一区二区三区在线视频 | 国内精品免费 | 国产精品精品久久久久久 | 色www.| 免费在线观看视频a | 在线日韩精品视频 | 成人蜜桃视频 | 久久永久免费视频 | 欧美黄色免费 | 国产黄在线免费观看 | 国产精品a久久久久 | 日日操日日插 | 日韩欧美v| 九九视频免费观看视频精品 | 91夜夜夜 | 天天激情 | 亚洲视频免费在线观看 | 国产黄视频在线观看 | 天天干天天玩天天操 | 久久你懂的 | 久久一区91 | 特黄特色特刺激视频免费播放 | 日韩电影中文 | 免费在线播放视频 | 狠狠色丁香婷婷综合久久片 | 蜜臀一区二区三区精品免费视频 | 日韩区欠美精品av视频 | 人人干免费 | 欧美日韩不卡一区二区三区 | 国产精品系列在线播放 | www天天干com| 最近中文字幕在线播放 | 99精品视频一区 | 在线看片中文字幕 | 色大片免费看 | 国产护士在线 | 99久久婷婷国产综合精品 | 国产最新在线 | 欧洲不卡av | 黄色国产大片 | 久久久精品免费看 | 成人久久久久久久久久 | 久久视频99| 亚洲美女免费精品视频在线观看 | 中文字幕国产 | 日韩午夜在线观看 | 久久试看 | 国产一区二区在线观看免费 | 国产精品一区二区 91 | 国产精品美女久久 | 精品国产一区二区三区四区vr | 成 人 免费 黄 色 视频 | 欧美精品在线一区二区 | 麻豆传媒视频在线播放 | 精品福利片 | 日韩在线短视频 | 99免费看片 | 欧美激情操| 中文字幕一区二区三区在线观看 | 国产高清av在线播放 | 久久露脸国产精品 | 国产一区二区网址 | 在线国产视频 | 正在播放国产精品 | 成人av免费网站 | 国产一区二区免费看 | 久久久久久片 | 三级黄色大片在线观看 | 狠狠干夜夜爱 | 91亚洲夫妻 | 91精选 | 国产美女被啪进深处喷白浆视频 | 国产视频黄 | 毛片网站在线看 | 国内丰满少妇猛烈精品播放 | 中文字幕在线观看一区 | 亚洲精品乱码久久久久久9色 | 色综合天天综合 | 午夜影院在线观看18 | 久色免费视频 | 国产手机免费视频 | 激情久久网 | 最近字幕在线观看第一季 | 国产99久久久久 | 狠狠久久婷婷 | 黄色毛片网站在线观看 | 日韩视频精品在线 | 日本特黄特色aaa大片免费 | 亚洲特级毛片 | 免费美女久久99 | 激情综合五月天 | 色婷婷激情综合 | 99视频在线免费 | 激情中文在线 | 99久热在线精品 | 美女网站黄免费 | 色婷婷成人网 | 国产精品成| av电影 一区二区 | 中文字幕在线视频一区二区 | 日本婷婷色 | 欧美一级日韩三级 | 婷婷九月激情 | 伊人春色电影网 | 少妇bbb搡bbbb搡bbbb | 成人黄色影片在线 | 欧美日韩久| 久久精品视频免费播放 | 免费国产在线精品 | 日韩91在线| 日韩美女黄色片 | 9色在线视频 | 成人免费毛片aaaaaa片 | 久久免费在线观看视频 | 欧美午夜久久久 | 亚洲成人资源在线 | 国产精品 欧美 日韩 | 91精品国产91 | av黄色影院| 国产.精品.日韩.另类.中文.在线.播放 | 久久99爱视频 | 婷婷爱五月天 | 亚洲综合色激情五月 | 国产精品久久久久久久7电影 | 五月婷婷播播 | 日本成人免费在线观看 | 国产精品一区二区久久精品爱微奶 | 色五月激情五月 | 国产 日韩 中文字幕 | 狠狠色狠狠色合久久伊人 | 日日干天天干 | 久久国产精品99精国产 | 在线观看国产成人av片 | 国产明星视频三级a三级点| 日韩视频一区二区在线观看 | 亚洲视屏一区 | 久久久久久久亚洲精品 | 成人影片在线播放 | 久久久综合精品 | 国产永久网站 | 天天色天 | 久久五月婷婷丁香社区 | 91一区二区在线 | 日韩理论视频 | 99热官网 | 国产不卡免费视频 | 999久久国产精品免费观看网站 | 婷婷在线五月 | 久久精品视频99 | 91精品国产亚洲 | 国产精品18久久久 | 91视频 - 114av | 亚洲国产高清在线观看视频 | 婷婷色在线视频 | 亚洲欧美成人 | 色噜噜狠狠狠狠色综合久不 | 精品99免费视频 | av手机在线播放 | 91九色视频网站 | 97超碰超碰久久福利超碰 | 狠狠色丁香久久婷婷综合丁香 | 2023亚洲精品国偷拍自产在线 | 日本黄色片一区二区 | 91麻豆福利 | 久久伊人热 | 九九视频网 | 一级淫片在线观看 | 免费看的黄色的网站 | 日日操天天操狠狠操 | 久久桃花网 | 夜添久久精品亚洲国产精品 | 色av资源网| 亚洲精品午夜久久久久久久 | 一区二区视频播放 | 国产精品高潮呻吟久久av无 | 99热国产在线 | 国产亚洲婷婷免费 |