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

一口氣說出 6種 延時隊列的實現方法,面試官也得服

開發 架構
下邊會介紹多種實現延時隊列的思路,文末提供有幾種實現方式的 github地址。其實哪種方式都沒有絕對的好與壞,只是看把它用在什么業務場景中,技術這東西沒有最好的只有最合適的。

 五一期間原計劃是寫兩篇文章,看一本技術類書籍,結果這五天由于自律性過于差,禁不住各種誘惑,我連電腦都沒打開過,計劃完美宣告失敗。所以在這能看出和大佬之間的差距,人家沒白沒夜的更文,比你優秀的人比你更努力,難以望其項背,真是讓我自愧不如。

[[325171]]

知恥而后勇,這不逼著自己又學起來了,個人比較喜歡一些實踐類的東西,既學習到知識又能讓技術落地,能搞出個demo最好,本來不知道該分享什么主題,好在最近項目緊急招人中,而我有幸做了回面試官,就給大家整理分享一道面試題:“如何實現延時隊列?”。

下邊會介紹多種實現延時隊列的思路,文末提供有幾種實現方式的 github地址。其實哪種方式都沒有絕對的好與壞,只是看把它用在什么業務場景中,技術這東西沒有最好的只有最合適的。

一、延時隊列的應用

什么是延時隊列?顧名思義:首先它要具有隊列的特性,再給它附加一個延遲消費隊列消息的功能,也就是說可以指定隊列中的消息在哪個時間點被消費。

延時隊列在項目中的應用還是比較多的,尤其像電商類平臺:

1、訂單成功后,在30分鐘內沒有支付,自動取消訂單

2、外賣平臺發送訂餐通知,下單成功后60s給用戶推送短信。

3、如果訂單一直處于某一個未完結狀態時,及時處理關單,并退還庫存

4、淘寶新建商戶一個月內還沒上傳商品信息,將凍結商鋪等

。。。。

上邊的這些場景都可以應用延時隊列解決。

二、延時隊列的實現

我個人一直秉承的觀點:工作上能用JDK自帶API實現的功能,就不要輕易自己重復造輪子,或者引入三方中間件。一方面自己封裝很容易出問題(大佬除外),再加上調試驗證產生許多不必要的工作量;另一方面一旦接入三方的中間件就會讓系統復雜度成倍的增加,維護成本也大大的增加。

1、DelayQueue 延時隊列

JDK 中提供了一組實現延遲隊列的API,位于Java.util.concurrent包下DelayQueue。

DelayQueue是一個BlockingQueue(無界阻塞)隊列,它本質就是封裝了一個PriorityQueue(優先隊列),PriorityQueue內部使用完全二叉堆(不知道的自行了解哈)來實現隊列元素排序,我們在向DelayQueue隊列中添加元素時,會給元素一個Delay(延遲時間)作為排序條件,隊列中最小的元素會優先放在隊首。隊列中的元素只有到了Delay時間才允許從隊列中取出。隊列中可以放基本數據類型或自定義實體類,在存放基本數據類型時,優先隊列中元素默認升序排列,自定義實體類就需要我們根據類屬性值比較計算了。

先簡單實現一下看看效果,添加三個order入隊DelayQueue,分別設置訂單在當前時間的5秒、10秒、15秒后取消。

 

要實現DelayQueue延時隊列,隊中元素要implements Delayed 接口,這哥接口里只有一個getDelay方法,用于設置延期時間。Order類中compareTo方法負責對隊列中的元素進行排序。

  1. public class Order implements Delayed { 
  2.     /** 
  3.      * 延遲時間 
  4.      */ 
  5.     @JsonFormat(locale = "zh", timezone = "GMT+8", pattern = "yyyy-MM-dd HH:mm:ss"
  6.     private long time
  7.     String name
  8.      
  9.     public Order(String name, long time, TimeUnit unit) { 
  10.         this.name = name
  11.         this.time = System.currentTimeMillis() + (time > 0 ? unit.toMillis(time) : 0); 
  12.     } 
  13.      
  14.     @Override 
  15.     public long getDelay(TimeUnit unit) { 
  16.         return time - System.currentTimeMillis(); 
  17.     } 
  18.     @Override 
  19.     public int compareTo(Delayed o) { 
  20.         Order Order = (Order) o; 
  21.         long diff = this.time - Order.time
  22.         if (diff <= 0) { 
  23.             return -1; 
  24.         } else { 
  25.             return 1; 
  26.         } 
  27.     } 

DelayQueue的put方法是線程安全的,因為put方法內部使用了ReentrantLock鎖進行線程同步。DelayQueue還提供了兩種出隊的方法 poll() 和 take() , poll() 為非阻塞獲取,沒有到期的元素直接返回null;take() 阻塞方式獲取,沒有到期的元素線程將會等待。

  1. public class DelayQueueDemo { 
  2.  
  3.     public static void main(String[] args) throws InterruptedException { 
  4.         Order Order1 = new Order("Order1", 5, TimeUnit.SECONDS); 
  5.         Order Order2 = new Order("Order2", 10, TimeUnit.SECONDS); 
  6.         Order Order3 = new Order("Order3", 15, TimeUnit.SECONDS); 
  7.         DelayQueue<Order> delayQueue = new DelayQueue<>(); 
  8.         delayQueue.put(Order1); 
  9.         delayQueue.put(Order2); 
  10.         delayQueue.put(Order3); 
  11.  
  12.         System.out.println("訂單延遲隊列開始時間:" + LocalDateTime.now().format(DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss"))); 
  13.         while (delayQueue.size() != 0) { 
  14.             /** 
  15.              * 取隊列頭部元素是否過期 
  16.              */ 
  17.             Order task = delayQueue.poll(); 
  18.             if (task != null) { 
  19.                 System.out.format("訂單:{%s}被取消, 取消時間:{%s}\n", task.name, LocalDateTime.now().format(DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss"))); 
  20.             } 
  21.             Thread.sleep(1000); 
  22.         } 
  23.     } 

上邊只是簡單的實現入隊與出隊的操作,實際開發中會有專門的線程,負責消息的入隊與消費。

執行后看到結果如下,Order1、Order2、Order3 分別在 5秒、10秒、15秒后被執行,至此就用DelayQueue實現了延時隊列。

  1. 訂單延遲隊列開始時間:2020-05-06 14:59:09 
  2. 訂單:{Order1}被取消, 取消時間:{2020-05-06 14:59:14} 
  3. 訂單:{Order2}被取消, 取消時間:{2020-05-06 14:59:19} 
  4. 訂單:{Order3}被取消, 取消時間:{2020-05-06 14:59:24} 

2、Quartz 定時任務

Quartz一款非常經典任務調度框架,在Redis、RabbitMQ還未廣泛應用時,超時未支付取消訂單功能都是由定時任務實現的。定時任務它有一定的周期性,可能很多單子已經超時,但還沒到達觸發執行的時間點,那么就會造成訂單處理的不夠及時。

引入quartz框架依賴包

  1. <dependency> 
  2.      <groupId>org.springframework.boot</groupId> 
  3.      <artifactId>spring-boot-starter-quartz</artifactId> 
  4. </dependency> 

在啟動類中使用@EnableScheduling注解開啟定時任務功能。

  1. @EnableScheduling 
  2. @SpringBootApplication 
  3. public class DelayqueueApplication { 
  4.  public static void main(String[] args) { 
  5.   SpringApplication.run(DelayqueueApplication.class, args); 
  6.  } 

編寫一個定時任務,每個5秒執行一次。

  1. @Component 
  2. public class QuartzDemo { 
  3.  
  4.     //每隔五秒 
  5.     @Scheduled(cron = "0/5 * * * * ? "
  6.     public void process(){ 
  7.         System.out.println("我是定時任務!"); 
  8.     } 

Redis的數據結構Zset,同樣可以實現延遲隊列的效果,主要利用它的score屬性,redis通過score來為集合中的成員進行從小到大的排序。

通過zadd命令向隊列delayqueue 中添加元素,并設置score值表示元素過期的時間;向delayqueue 添加三個order1、order2、order3,分別是10秒、20秒、30秒后過期。

  1. zadd delayqueue 3 order3 

消費端輪詢隊列delayqueue, 將元素排序后取最小時間與當前時間比對,如小于當前時間代表已經過期移除key。

  1. /** 
  2.      * 消費消息 
  3.      */ 
  4.     public void pollOrderQueue() { 
  5.  
  6.         while (true) { 
  7.             Set<Tuple> set = jedis.zrangeWithScores(DELAY_QUEUE, 0, 0); 
  8.  
  9.             String value = ((Tuple) set.toArray()[0]).getElement(); 
  10.             int score = (int) ((Tuple) set.toArray()[0]).getScore(); 
  11.              
  12.             Calendar cal = Calendar.getInstance(); 
  13.             int nowSecond = (int) (cal.getTimeInMillis() / 1000); 
  14.             if (nowSecond >= score) { 
  15.                 jedis.zrem(DELAY_QUEUE, value); 
  16.                 System.out.println(sdf.format(new Date()) + " removed key:" + value); 
  17.             } 
  18.  
  19.             if (jedis.zcard(DELAY_QUEUE) <= 0) { 
  20.                 System.out.println(sdf.format(new Date()) + " zset empty "); 
  21.                 return
  22.             } 
  23.             Thread.sleep(1000); 
  24.         } 
  25.     } 

我們看到執行結果符合預期

  1. 2020-05-07 13:24:09 add finished. 
  2. 2020-05-07 13:24:19 removed key:order1 
  3. 2020-05-07 13:24:29 removed key:order2 
  4. 2020-05-07 13:24:39 removed key:order3 
  5. 2020-05-07 13:24:39 zset empty  

4、Redis 過期回調

Redis 的key過期回調事件,也能達到延遲隊列的效果,簡單來說我們開啟監聽key是否過期的事件,一旦key過期會觸發一個callback事件。

修改redis.conf文件開啟notify-keyspace-events Ex

  1. notify-keyspace-events Ex 

Redis監聽配置,注入Bean RedisMessageListenerContainer

  1. @Configuration 
  2. public class RedisListenerConfig { 
  3.     @Bean 
  4.     RedisMessageListenerContainer container(RedisConnectionFactory connectionFactory) { 
  5.  
  6.         RedisMessageListenerContainer container = new RedisMessageListenerContainer(); 
  7.         container.setConnectionFactory(connectionFactory); 
  8.         return container; 
  9.     } 

編寫Redis過期回調監聽方法,必須繼承KeyExpirationEventMessageListener ,有點類似于MQ的消息監聽。

  1. @Component 
  2. public class RedisKeyExpirationListener extends KeyExpirationEventMessageListener { 
  3.   
  4.     public RedisKeyExpirationListener(RedisMessageListenerContainer listenerContainer) { 
  5.         super(listenerContainer); 
  6.     } 
  7.     @Override 
  8.     public void onMessage(Message message, byte[] pattern) { 
  9.         String expiredKey = message.toString(); 
  10.         System.out.println("監聽到key:" + expiredKey + "已過期"); 
  11.     } 

到這代碼就編寫完成,非常的簡單,接下來測試一下效果,在redis-cli客戶端添加一個key 并給定3s的過期時間。

  1. set xiaofu 123 ex 3 

在控制臺成功監聽到了這個過期的key。

  1. 監聽到過期的key為:xiaofu 

5、RabbitMQ 延時隊列

利用 RabbitMQ 做延時隊列是比較常見的一種方式,而實際上RabbitMQ 自身并沒有直接支持提供延遲隊列功能,而是通過 RabbitMQ 消息隊列的 TTL和 DXL這兩個屬性間接實現的。

先來認識一下 TTL和 DXL兩個概念:

Time To Live(TTL) :

TTL 顧名思義:指的是消息的存活時間,RabbitMQ可以通過x-message-tt參數來設置指定Queue(隊列)和 Message(消息)上消息的存活時間,它的值是一個非負整數,單位為微秒。

RabbitMQ 可以從兩種維度設置消息過期時間,分別是隊列和消息本身

  • 設置隊列過期時間,那么隊列中所有消息都具有相同的過期時間。
  • 設置消息過期時間,對隊列中的某一條消息設置過期時間,每條消息TTL都可以不同。

如果同時設置隊列和隊列中消息的TTL,則TTL值以兩者中較小的值為準。而隊列中的消息存在隊列中的時間,一旦超過TTL過期時間則成為Dead Letter(死信)。

Dead Letter Exchanges(DLX)

DLX即死信交換機,綁定在死信交換機上的即死信隊列。RabbitMQ的 Queue(隊列)可以配置兩個參數x-dead-letter-exchange 和 x-dead-letter-routing-key(可選),一旦隊列內出現了Dead Letter(死信),則按照這兩個參數可以將消息重新路由到另一個Exchange(交換機),讓消息重新被消費。

x-dead-letter-exchange:隊列中出現Dead Letter后將Dead Letter重新路由轉發到指定 exchange(交換機)。

x-dead-letter-routing-key:指定routing-key發送,一般為要指定轉發的隊列。

隊列出現Dead Letter的情況有:

  • 消息或者隊列的TTL過期
  • 隊列達到最大長度
  • 消息被消費端拒絕(basic.reject or basic.nack)

下邊結合一張圖看看如何實現超30分鐘未支付關單功能,我們將訂單消息A0001發送到延遲隊列order.delay.queue,并設置x-message-tt消息存活時間為30分鐘,當到達30分鐘后訂單消息A0001成為了Dead Letter(死信),延遲隊列檢測到有死信,通過配置x-dead-letter-exchange,將死信重新轉發到能正常消費的關單隊列,直接監聽關單隊列處理關單邏輯即可。

 

發送消息時指定消息延遲的時間

  1. public void send(String delayTimes) { 
  2.         amqpTemplate.convertAndSend("order.pay.exchange""order.pay.queue","大家好我是延遲數據", message -> { 
  3.             // 設置延遲毫秒值 
  4.             message.getMessageProperties().setExpiration(String.valueOf(delayTimes)); 
  5.             return message; 
  6.         }); 
  7.     } 

設置延遲隊列出現死信后的轉發規則

  1. /** 
  2.      * 延時隊列 
  3.      */ 
  4.     @Bean(name = "order.delay.queue"
  5.     public Queue getMessageQueue() { 
  6.         return QueueBuilder 
  7.                 .durable(RabbitConstant.DEAD_LETTER_QUEUE) 
  8.                 // 配置到期后轉發的交換 
  9.                 .withArgument("x-dead-letter-exchange""order.close.exchange"
  10.                 // 配置到期后轉發的路由鍵 
  11.                 .withArgument("x-dead-letter-routing-key""order.close.queue"
  12.                 .build(); 
  13.     } 

6、時間輪

前邊幾種延時隊列的實現方法相對簡單,比較容易理解,時間輪算法就稍微有點抽象了。kafka、netty都有基于時間輪算法實現延時隊列,下邊主要實踐Netty的延時隊列講一下時間輪是什么原理。

先來看一張時間輪的原理圖,解讀一下時間輪的幾個基本概念

wheel :時間輪,圖中的圓盤可以看作是鐘表的刻度。比如一圈round 長度為24秒,刻度數為 8,那么每一個刻度表示 3秒。那么時間精度就是 3秒。時間長度 / 刻度數值越大,精度越大。

 

當添加一個定時、延時任務A,假如會延遲25秒后才會執行,可時間輪一圈round 的長度才24秒,那么此時會根據時間輪長度和刻度得到一個圈數 round和對應的指針位置 index,也是就任務A會繞一圈指向0格子上,此時時間輪會記錄該任務的round和index信息。當round=0,index=0 ,指針指向0格子 任務A并不會執行,因為 round=0不滿足要求。

所以每一個格子代表的是一些時間,比如1秒和25秒 都會指向0格子上,而任務則放在每個格子對應的鏈表中,這點和HashMap的數據有些類似。

Netty構建延時隊列主要用HashedWheelTimer,HashedWheelTimer底層數據結構依然是使用DelayedQueue,只是采用時間輪的算法來實現。

下面我們用Netty 簡單實現延時隊列,HashedWheelTimer構造函數比較多,解釋一下各參數的含義。

ThreadFactory :表示用于生成工作線程,一般采用線程池;

tickDuration和unit:每格的時間間隔,默認100ms;

ticksPerWheel:一圈下來有幾格,默認512,而如果傳入數值的不是2的N次方,則會調整為大于等于該參數的一個2的N次方數值,有利于優化hash值的計算。

  1. public HashedWheelTimer(ThreadFactory threadFactory, long tickDuration, TimeUnit unit, int ticksPerWheel) { 
  2.         this(threadFactory, tickDuration, unit, ticksPerWheel, true); 
  3.     } 
  • TimerTask:一個定時任務的實現接口,其中run方法包裝了定時任務的邏輯。
  • Timeout:一個定時任務提交到Timer之后返回的句柄,通過這個句柄外部可以取消這個定時任務,并對定時任務的狀態進行一些基本的判斷。
  • Timer:是HashedWheelTimer實現的父接口,僅定義了如何提交定時任務和如何停止整個定時機制。
  1. public class NettyDelayQueue { 
  2.  
  3.     public static void main(String[] args) { 
  4.  
  5.         final Timer timer = new HashedWheelTimer(Executors.defaultThreadFactory(), 5, TimeUnit.SECONDS, 2); 
  6.  
  7.         //定時任務 
  8.         TimerTask task1 = new TimerTask() { 
  9.             public void run(Timeout timeout) throws Exception { 
  10.                 System.out.println("order1  5s 后執行 "); 
  11.                 timer.newTimeout(this, 5, TimeUnit.SECONDS);//結束時候再次注冊 
  12.             } 
  13.         }; 
  14.         timer.newTimeout(task1, 5, TimeUnit.SECONDS); 
  15.         TimerTask task2 = new TimerTask() { 
  16.             public void run(Timeout timeout) throws Exception { 
  17.                 System.out.println("order2  10s 后執行"); 
  18.                 timer.newTimeout(this, 10, TimeUnit.SECONDS);//結束時候再注冊 
  19.             } 
  20.         }; 
  21.  
  22.         timer.newTimeout(task2, 10, TimeUnit.SECONDS); 
  23.  
  24.         //延遲任務 
  25.         timer.newTimeout(new TimerTask() { 
  26.             public void run(Timeout timeout) throws Exception { 
  27.                 System.out.println("order3  15s 后執行一次"); 
  28.             } 
  29.         }, 15, TimeUnit.SECONDS); 
  30.  
  31.     } 

從執行的結果看,order3、order3延時任務只執行了一次,而order2、order1為定時任務,按照不同的周期重復執行。

  1. order1  5s 后執行  
  2. order2  10s 后執行 
  3. order3  15s 后執行一次 
  4. order1  5s 后執行  
  5. order2  10s 后執行 

總結

為了讓大家更容易理解,上邊的代碼寫的都比較簡單粗糙,幾種實現方式的demo已經都提交到github 地址:https://github.com/chengxy-nds/delayqueue,感興趣的小伙伴可以下載跑一跑。

這篇文章肝了挺長時間,寫作一點也不比上班干活輕松,查證資料反復驗證demo的可行性,搭建各種RabbitMQ、Redis環境,只想說我太難了!

可能寫的有不夠完善的地方,如哪里有錯誤或者不明了的,歡迎大家踴躍指正!!!

責任編輯:武曉燕 來源: 程序員內點事
相關推薦

2020-04-16 12:42:42

附近的人共享單車App

2020-08-12 09:55:07

附近的人數據庫MySQL

2022-05-24 11:50:46

延時消息分布式

2020-04-14 13:32:56

@Transacti失效場景

2021-12-06 08:30:49

SpringSpring Bean面試題

2020-03-31 08:12:25

Kafka架構數據庫

2020-07-31 10:15:32

分布式ID數據庫MySQL

2020-11-04 14:20:58

分布式數據庫MySQL

2020-09-24 09:08:04

分布式系統架構

2020-07-08 07:45:44

OAuth2.0授權

2020-12-21 06:07:35

Mybatis設計模式

2020-07-10 07:44:26

Session方式Web

2021-06-08 22:43:07

IPC方式Qt

2025-05-14 01:55:00

FCMCPAI

2021-03-29 12:22:25

微信iOS蘋果

2020-06-04 07:45:07

過濾器和攔截器

2023-12-18 23:09:25

開源優化引擎

2020-10-22 12:30:33

MySQL

2024-03-26 09:42:27

分片算法應用

2025-11-11 08:47:00

點贊
收藏

51CTO技術棧公眾號

亚洲成人av片在线观看| 亚洲狼人国产精品| 国产精品免费视频xxxx| 999精品久久久| 视频二区欧美| 色狠狠一区二区三区香蕉| 亚洲第一导航| 蜜臀av在线观看| 日韩国产欧美三级| 欧美激情第1页| 欧美一区二区三区成人精品| 日日夜夜综合| 欧美日韩在线视频一区| 中文字幕乱码一区二区三区| 色欲av永久无码精品无码蜜桃| 日日嗨av一区二区三区四区| 九九视频直播综合网| 黄瓜视频污在线观看| 99综合久久| 日韩欧美在线网址| 老司机激情视频| 高清美女视频一区| 成人性色生活片| 国产精品久久在线观看| 韩国av免费观看| 女人天堂亚洲aⅴ在线观看| 亚洲男女自偷自拍图片另类| 黑人巨大猛交丰满少妇| 国产极品嫩模在线观看91精品| 亚洲va天堂va国产va久| 波多野结衣激情| 看电影就来5566av视频在线播放| 国产精品一级片| 国产精品人成电影| 五月天激情国产综合婷婷婷| 午夜日韩电影| 波霸ol色综合久久| 鲁丝一区二区三区| 丝袜美腿综合| 亚洲成人黄色网| 日本在线视频播放| 日本超碰一区二区| 欧美精品三级日韩久久| 不卡av免费在线| 手机在线观看av网站| 一区二区三区成人在线视频| 国产a级片免费看| 精品乱码一区二区三四区视频 | 91av中文字幕| 国产黄色片视频| 欧美二区不卡| 精品视频9999| 少妇久久久久久被弄高潮| 国产精品88久久久久久| 色伦专区97中文字幕| 日本乱子伦xxxx| 禁果av一区二区三区| 亚洲人成网站999久久久综合| 男人网站在线观看| 精品国产一区二区三区成人影院| 欧美成人vps| www.四虎精品| 精品淫伦v久久水蜜桃| 精品久久久久久久人人人人传媒| jjzzjjzz欧美69巨大| 97se亚洲| 亚洲电影天堂av | 久久精品人人做| 欧美日韩一区在线视频| 成人性爱视频在线观看| 国产精品日产欧美久久久久| 曰韩不卡视频| 天堂av在线电影| 亚洲成人综合在线| 日日碰狠狠躁久久躁婷婷| 午夜日韩成人影院| 欧美挠脚心视频网站| 国产大片一区二区三区| 99久久香蕉| 日韩电影大片中文字幕| 中文字幕网站在线观看| 久久一区二区三区喷水| 欧美成人午夜免费视在线看片| 久久久.www| 99视频精品免费观看| 国产99久久精品一区二区永久免费| 日韩乱码一区二区三区| 国产一区二区不卡老阿姨| 国产一区二区免费在线观看| 国产三级在线| 亚洲精品成人a在线观看| 欧美 国产 综合| 国产伊人久久| 欧美va亚洲va在线观看蝴蝶网| 中文字幕xxx| 久久久久久久久丰满| 久久人人爽人人爽人人片av高清| 日本少妇毛茸茸高潮| 日韩福利视频网| 99在线视频播放| 免费人成黄页在线观看忧物| 亚洲欧美一区二区久久| 男人天堂999| 91精品一区| 亚洲九九九在线观看| 欧洲美女女同性互添| 国产精品资源| 51蜜桃传媒精品一区二区| 色久视频在线播放| 亚洲欧美电影院| 三年中国国语在线播放免费| 成人福利一区| 久久久精品一区二区三区| 久久久久久久久久久影院| 国产美女娇喘av呻吟久久| 欧美一级片免费观看| 黄网av在线| 欧美精品乱人伦久久久久久| 无码熟妇人妻av| 激情自拍一区| 91牛牛免费视频| 国产对白叫床清晰在线播放| 亚洲一区二区av电影| 一本岛在线视频| 亚洲人成伊人成综合图片| 欧美大片免费看| 一级黄色大片网站| 国产丝袜美腿一区二区三区| 在线观看av的网址| 日韩精品第二页| 亚洲另类图片色| 国产又色又爽又黄的| 国产成人福利片| 一区一区视频| 黑人一区二区三区| 亚洲精品视频免费在线观看| 国产一区二区三区影院| 福利91精品一区二区三区| 一级二级三级欧美| 黄色精品视频| 在线观看不卡av| 免费视频久久久| 91在线观看污| 日本精品免费在线观看| 美女福利一区| 午夜精品久久17c| 老熟妇高潮一区二区高清视频| 亚洲欧美日韩小说| 成人性生交视频免费观看| 手机在线一区二区三区| 国产欧美韩国高清| 欧美日韩视频在线播放| 欧美午夜精品一区二区三区| 欧洲美熟女乱又伦| 日本系列欧美系列| 亚洲一卡二卡三卡| av国产精品| 久久国产视频网站| 亚洲高清精品视频| 亚洲成人1区2区| 无遮挡aaaaa大片免费看| 免费在线日韩av| 欧美黑人3p| 欧美一级做a| 久久精品电影一区二区| av老司机久久| 亚洲综合精品自拍| av黄色一级片| 首页国产欧美日韩丝袜| 亚洲v国产v在线观看| 四虎影视国产精品| 欧美交受高潮1| 外国精品视频在线观看| 在线看不卡av| 欧美精品久久久久久久久46p| 国产激情偷乱视频一区二区三区| 福利视频免费在线观看| 欧美三级午夜理伦三级在线观看| 国产91亚洲精品| 麻豆网站在线免费观看| 精品国产免费人成电影在线观看四季| 日本特黄特色aaa大片免费| 久久综合九色综合欧美98| 国产嫩草在线观看| 欧美久久综合| 久久资源av| 亚洲精品三区| 91国语精品自产拍在线观看性色| 国产在线观看黄| 日韩三级.com| 69亚洲精品久久久蜜桃小说| 最新高清无码专区| 可以直接看的无码av| 青青青伊人色综合久久| 菠萝蜜视频在线观看入口| 免费一区二区三区视频导航| 成人国产精品久久久久久亚洲| 丰满诱人av在线播放| 一区二区三区国产在线观看| 国产高清免费av| 欧美视频免费在线观看| 国精品无码一区二区三区| 91看片淫黄大片一级在线观看| 日本激情综合网| 在线亚洲一区| 视色,视色影院,视色影库,视色网| 美女一区2区| 成人免费在线视频网址| 天堂中文最新版在线中文| 久久久91精品国产| 你懂的在线观看| 日韩美女在线视频| 亚洲性生活大片| 黑人巨大精品欧美一区二区| 丝袜美腿小色网| 国产欧美日韩不卡| 深爱五月激情网| av资源站一区| 亚洲国产欧美日韩在线| 免费av成人在线| 欧美aⅴ在线观看| 伊人蜜桃色噜噜激情综合| 午夜探花在线观看| 欧美日韩高清| 欧美性天天影院| 噜噜噜天天躁狠狠躁夜夜精品| 亚洲影院色无极综合| 国产成人精品一区二区三区免费| 97超碰蝌蚪网人人做人人爽| 密臀av在线| 欧美精品午夜视频| 国产激情在线| 久久精品电影网站| 黄色网页网址在线免费| 中文字幕亚洲欧美| 九色在线播放| 亚洲欧洲xxxx| 男人天堂资源在线| 日韩国产一区三区| 天堂√在线中文官网在线| 亚洲第一区第一页| 亚洲欧美综合一区二区| 亚洲精品久久久久久久久久久| 超碰在线播放97| 日韩精品中午字幕| 成人高潮片免费视频| 日韩一区二区电影在线| jlzzjlzzjlzz亚洲人| 欧美一级片在线看| 97精品人妻一区二区三区在线| 欧美日韩免费一区二区三区视频| 午夜视频网站在线观看| 欧美日韩在线综合| 一级片一区二区三区| 欧美日韩高清在线播放| 国产又粗又大又爽| 欧美一区二区三区四区视频| 国产三级自拍视频| 日韩欧美在线影院| 日韩专区第一页| 精品无人区太爽高潮在线播放| 天天影院图片亚洲| 一区二区三区视频在线| 天天影视久久综合| 欧美成人免费在线视频| 91超碰国产在线| 欧美最顶级的aⅴ艳星| 在线日本欧美| 国产免费观看久久黄| 日本一区二区三区电影免费观看| 成人av免费电影| 亚洲区小说区| 影音先锋亚洲视频| 黄色成人在线网址| 国产高清精品在线观看| 免费在线观看成人| 日本少妇一区二区三区| 91美女蜜桃在线| 91大神福利视频| 午夜影院久久久| www.av88| 欧美zozozo| 国产美女性感在线观看懂色av| 日韩亚洲欧美成人| 国产探花视频在线观看| 日韩美女av在线免费观看| 日本在线一区二区| 国产精品v欧美精品v日韩| 蜜桃精品wwwmitaows| 精品少妇人妻av一区二区| 亚洲茄子视频| 日韩av卡一卡二| va亚洲va日韩不卡在线观看| 欧美极品jizzhd欧美18| 亚洲国产综合色| 中文字幕免费播放| 亚洲精品xxx| 黄色av免费在线| 欧美亚洲午夜视频在线观看| 中文字幕综合| 奇米精品在线| 欧美久久一级| 中文字幕有码av| www.欧美精品一二区| 91ts人妖另类精品系列| 精品久久久久久久久久久久久| 亚洲天堂视频网| 亚洲乱码一区二区| 欧美78videosex性欧美| 国产精品久久久久久久久久ktv| 伊人久久亚洲| 正在播放亚洲| 久久在线91| 亚洲激情 欧美| 亚洲精品国产一区二区精华液 | 99福利在线| 国产精品久久久久久亚洲调教| 久久悠悠精品综合网| 大桥未久一区二区三区| 日日夜夜精品视频天天综合网| 亚洲av成人精品一区二区三区| 亚洲视频香蕉人妖| 在线免费观看av片| 亚洲午夜激情免费视频| 九色porny丨入口在线| 国产福利久久精品| 欧美精品91| 黄色a级三级三级三级| 国产精品热久久久久夜色精品三区| 欧美特黄aaaaaa| 亚洲精品久久久久久久久久久久久 | 99re8这里只有精品| 麻豆精品一区二区三区| 国产一区二区三区四区五区六区| 欧美日韩国产综合视频在线观看中文| 超碰免费在线97| 欧美丰满少妇xxxx| 亚洲天堂av资源在线观看| 影音先锋男人的网站| 国产在线不卡一卡二卡三卡四卡| 成人性视频免费看| 欧美日韩在线播放三区四区| а天堂8中文最新版在线官网| 国产成人精品一区二区| 久久99蜜桃| 日本成人中文字幕在线| 国产视频一区不卡| 亚洲精品国产欧美在线观看| 国产一区二区三区丝袜| 澳门av一区二区三区| 日本不卡一区| 日韩av高清在线观看| 国产精品久久久视频| 欧美午夜精品一区| 蜜桃av在线免费观看| 91日韩在线视频| 欧美精品成人| 精品国产一区在线| 色综合天天综合网国产成人综合天| 人人九九精品| 国产精品久久久久久久久久小说| 不卡中文一二三区| 三日本三级少妇三级99| 亚洲精品久久7777| 五月天久久久久久| 国产成人一区二区三区| 91精品综合久久久久久久久久久 | 少妇激情综合网| 国产一区二区三区免费观看在线 | 久久精品国产精品国产精品污 | 国产精品黄色av| 国产精品88久久久久久| 无码国产精品久久一区免费| 亚洲成av人在线观看| 国产粉嫩一区二区三区在线观看| 成人黄色午夜影院| 精品69视频一区二区三区Q| 亚洲精品成人无码熟妇在线| 欧美色图片你懂的| 污污影院在线观看| 久久婷婷开心| 久久国产精品免费| 国产真实乱偷精品视频| 亚洲欧美综合精品久久成人| 国模大尺度视频一区二区| 日韩欧美国产综合在线| 国产欧美日韩精品a在线观看| 国产黄a三级三级看三级| 91精品国产免费久久久久久 | 色偷偷9999www| 国产精品国产| 污网站免费在线| 亚洲国产精品久久久男人的天堂| 欧洲毛片在线| 99中文视频在线| 麻豆久久精品| 亚洲av鲁丝一区二区三区| 亚洲日本成人网| eeuss国产一区二区三区四区| 网站一区二区三区|