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

基于Consul的分布式信號量實現(xiàn)

開發(fā) 開發(fā)工具 分布式
本文將繼續(xù)討論基于Consul的分布式鎖實現(xiàn)。信號量是我們在實現(xiàn)并發(fā)控制時會經(jīng)常使用的手段,主要用來限制同時并發(fā)線程或進程的數(shù)量。

本文將繼續(xù)討論基于Consul的分布式鎖實現(xiàn)。信號量是我們在實現(xiàn)并發(fā)控制時會經(jīng)常使用的手段,主要用來限制同時并發(fā)線程或進程的數(shù)量,比如:Zuul默認情況下就使用信號量來限制每個路由的并發(fā)數(shù),以實現(xiàn)不同路由間的資源隔離。

[[190924]]

信號量(Semaphore),有時被稱為信號燈,是在多線程環(huán)境下使用的一種設(shè)施,是可以用來保證兩個或多個關(guān)鍵代碼段不被并發(fā)調(diào)用。在進入一個關(guān)鍵代碼段之前,線程必須獲取一個信號量;一旦該關(guān)鍵代碼段完成了,那么該線程必須釋放信號量。其他想進入該關(guān)鍵代碼段的線程必須等待直到***個線程釋放信號量。為了完成這個過程,需要創(chuàng)建一個信號量VI,然后將Acquire Semaphore VI以及Release Semaphore VI分別放置在每個關(guān)鍵代碼段的首末端,確認這些信號量VI引用的是初始創(chuàng)建的信號量。如在這個停車場系統(tǒng)中,車位是公共資源,每輛車好比一個線程,看門人起的就是信號量的作用。

實現(xiàn)思路

  1. 信號量存儲:semaphore/key
  2. acquired操作:
  3. 創(chuàng)建session
  4. 鎖定key競爭者:semaphore/key/session
  5. 查詢信號量:semaphore/key/.lock,可以獲得如下內(nèi)容(如果是***次創(chuàng)建信號量,將獲取不到,這個時候就直接創(chuàng)建)
    1.     "limit": 3, 
    2.     "holders": [ 
    3.         "90c0772a-4bd3-3a3c-8215-3b8937e36027", 
    4.         "93e5611d-5365-a374-8190-f80c4a7280ab" 
    5.     ] 
  6. 如果持有者已達上限,返回false,如果阻塞模式,就繼續(xù)嘗試acquired操作
  7. 如果持有者未達上限,更新semaphore/key/.lock的內(nèi)容,將當(dāng)前線程的sessionId加入到holders中。注意:更新的時候需要設(shè)置cas,它的值是“查詢信號量”步驟獲得的“ModifyIndex”值,該值用于保證更新操作的基礎(chǔ)沒有被其他競爭者更新。如果更新成功,就開始執(zhí)行具體邏輯。如果沒有更新成功,說明有其他競爭者搶占了資源,返回false,阻塞模式下繼續(xù)嘗試acquired操作
  8. release操作:
  • 從semaphore/key/.lock的holders中移除當(dāng)前sessionId
  • 刪除semaphore/key/session
  • 刪除當(dāng)前的session

流程圖

代碼實現(xiàn)

  1. public class Semaphore { 
  2.   
  3.     private Logger logger = Logger.getLogger(getClass()); 
  4.   
  5.     private static final String prefix = "semaphore/";  // 信號量參數(shù)前綴 
  6.   
  7.     private ConsulClient consulClient; 
  8.     private int limit; 
  9.     private String keyPath; 
  10.     private String sessionId = null
  11.     private boolean acquired = false
  12.   
  13.     /** 
  14.      * 
  15.      * @param consulClient consul客戶端實例 
  16.      * @param limit 信號量上限值 
  17.      * @param keyPath 信號量在consul中存儲的參數(shù)路徑 
  18.      */ 
  19.     public Semaphore(ConsulClient consulClient, int limit, String keyPath) { 
  20.         this.consulClient = consulClient; 
  21.         this.limit = limit; 
  22.         this.keyPath = prefix + keyPath; 
  23.     } 
  24.   
  25.     /** 
  26.      * acquired信號量 
  27.      * 
  28.      * @param block 是否阻塞。如果為true,那么一直嘗試,直到獲取到該資源為止。 
  29.      * @return 
  30.      * @throws IOException 
  31.      */ 
  32.     public Boolean acquired(boolean block) throws IOException { 
  33.   
  34.         if(acquired) { 
  35.             logger.error(sessionId + " - Already acquired"); 
  36.             throw new RuntimeException(sessionId + " - Already acquired"); 
  37.         } 
  38.   
  39.         // create session 
  40.         clearSession(); 
  41.         this.sessionId = createSessionId("semaphore"); 
  42.         logger.debug("Create session : " + sessionId); 
  43.   
  44.         // add contender entry 
  45.         String contenderKey = keyPath + "/" + sessionId; 
  46.         logger.debug("contenderKey : " + contenderKey); 
  47.         PutParams putParams = new PutParams(); 
  48.         putParams.setAcquireSession(sessionId); 
  49.         Boolean b = consulClient.setKVValue(contenderKey, "", putParams).getValue(); 
  50.         if(!b) { 
  51.             logger.error("Failed to add contender entry : " + contenderKey + ", " + sessionId); 
  52.             throw new RuntimeException("Failed to add contender entry : " + contenderKey + ", " + sessionId); 
  53.         } 
  54.   
  55.         while(true) { 
  56.             // try to take the semaphore 
  57.             String lockKey = keyPath + "/.lock"; 
  58.             String lockKeyValue; 
  59.   
  60.             GetValue lockKeyContent = consulClient.getKVValue(lockKey).getValue(); 
  61.   
  62.             if (lockKeyContent != null) { 
  63.                 // lock值轉(zhuǎn)換 
  64.                 lockKeyValue = lockKeyContent.getValue(); 
  65.                 BASE64Decoder decoder = new BASE64Decoder(); 
  66.                 byte[] v = decoder.decodeBuffer(lockKeyValue); 
  67.                 String lockKeyValueDecode = new String(v); 
  68.                 logger.debug("lockKey=" + lockKey + "lockKeyValueDecode=" + lockKeyValueDecode); 
  69.   
  70.                 Gson gson = new Gson(); 
  71.                 ContenderValue contenderValue = gson.fromJson(lockKeyValueDecode, ContenderValue.class); 
  72.                 // 當(dāng)前信號量已滿 
  73.                 if(contenderValue.getLimit() == contenderValue.getHolders().size()) { 
  74.                     logger.debug("Semaphore limited " + contenderValue.getLimit() + ", waiting..."); 
  75.                     if(block) { 
  76.                         // 如果是阻塞模式,再嘗試 
  77.                         try { 
  78.                             Thread.sleep(100L); 
  79.                         } catch (InterruptedException e) { 
  80.                         } 
  81.                         continue; 
  82.                     } 
  83.                     // 非阻塞模式,直接返回沒有獲取到信號量 
  84.                     return false; 
  85.                 } 
  86.                 // 信號量增加 
  87.                 contenderValue.getHolders().add(sessionId); 
  88.                 putParams = new PutParams(); 
  89.                 putParams.setCas(lockKeyContent.getModifyIndex()); 
  90.                 boolean c = consulClient.setKVValue(lockKey, contenderValue.toString(), putParams).getValue(); 
  91.                 if(c) { 
  92.                     acquired = true
  93.                     return true; 
  94.                 } 
  95.                 else 
  96.                     continue; 
  97.             } else { 
  98.                 // 當(dāng)前信號量還沒有,所以創(chuàng)建一個,并馬上搶占一個資源 
  99.                 ContenderValue contenderValue = new ContenderValue(); 
  100.                 contenderValue.setLimit(limit); 
  101.                 contenderValue.getHolders().add(sessionId); 
  102.   
  103.                 putParams = new PutParams(); 
  104.                 putParams.setCas(0L); 
  105.                 boolean c = consulClient.setKVValue(lockKey, contenderValue.toString(), putParams).getValue(); 
  106.                 if (c) { 
  107.                     acquired = true
  108.                     return true; 
  109.                 } 
  110.                 continue; 
  111.             } 
  112.         } 
  113.     } 
  114.   
  115.     /** 
  116.      * 創(chuàng)建sessionId 
  117.      * @param sessionName 
  118.      * @return 
  119.      */ 
  120.     public String createSessionId(String sessionName) { 
  121.         NewSession newnewSession = new NewSession(); 
  122.         newSession.setName(sessionName); 
  123.         return consulClient.sessionCreate(newSession, null).getValue(); 
  124.     } 
  125.   
  126.     /** 
  127.      * 釋放session、并從lock中移除當(dāng)前的sessionId 
  128.      * @throws IOException 
  129.      */ 
  130.     public void release() throws IOException { 
  131.         if(this.acquired) { 
  132.             // remove session from lock 
  133.             while(true) { 
  134.                 String contenderKey = keyPath + "/" + sessionId; 
  135.                 String lockKey = keyPath + "/.lock"; 
  136.                 String lockKeyValue; 
  137.   
  138.                 GetValue lockKeyContent = consulClient.getKVValue(lockKey).getValue(); 
  139.                 if (lockKeyContent != null) { 
  140.                     // lock值轉(zhuǎn)換 
  141.                     lockKeyValue = lockKeyContent.getValue(); 
  142.                     BASE64Decoder decoder = new BASE64Decoder(); 
  143.                     byte[] v = decoder.decodeBuffer(lockKeyValue); 
  144.                     String lockKeyValueDecode = new String(v); 
  145.                     Gson gson = new Gson(); 
  146.                     ContenderValue contenderValue = gson.fromJson(lockKeyValueDecode, ContenderValue.class); 
  147.                     contenderValue.getHolders().remove(sessionId); 
  148.                     PutParams putParams = new PutParams(); 
  149.                     putParams.setCas(lockKeyContent.getModifyIndex()); 
  150.                     consulClient.deleteKVValue(contenderKey); 
  151.                     boolean c = consulClient.setKVValue(lockKey, contenderValue.toString(), putParams).getValue(); 
  152.                     if(c) { 
  153.                         break; 
  154.                     } 
  155.                 } 
  156.             } 
  157.             // remove session key 
  158.   
  159.         } 
  160.         this.acquired = false
  161.         clearSession(); 
  162.     } 
  163.   
  164.     public void clearSession() { 
  165.         if(sessionId != null) { 
  166.             consulClient.sessionDestroy(sessionId, null); 
  167.             sessionId = null
  168.         } 
  169.     } 
  170.   
  171.     class ContenderValue implements Serializable { 
  172.   
  173.         private Integer limit; 
  174.         private List<String> holders = new ArrayList<>(); 
  175.   
  176.         public Integer getLimit() { 
  177.             return limit; 
  178.         } 
  179.   
  180.         public void setLimit(Integer limit) { 
  181.             this.limit = limit; 
  182.         } 
  183.   
  184.         public List<String> getHolders() { 
  185.             return holders; 
  186.         } 
  187.   
  188.         public void setHolders(List<String> holders) { 
  189.             this.holders = holders; 
  190.         } 
  191.   
  192.         @Override 
  193.         public String toString() { 
  194.             return new Gson().toJson(this); 
  195.         } 
  196.   
  197.     } 
  198.   

單元測試 

下面單元測試的邏輯:通過線程的方式來模擬不同的分布式服務(wù)來獲取信號量執(zhí)行業(yè)務(wù)邏輯。由于信號量與簡單的分布式互斥鎖有所不同,它不是只限定一個線程可以操作,而是可以控制多個線程的并發(fā),所以通過下面的單元測試,我們設(shè)置信號量為3,然后同時啟動15個線程來競爭的情況,來觀察分布式信號量實現(xiàn)的結(jié)果如何。

  1. INFO  [Thread-6] SemaphoreRunner - Thread 7 start! 
  2. INFO  [Thread-2] SemaphoreRunner - Thread 3 start! 
  3. INFO  [Thread-7] SemaphoreRunner - Thread 8 start! 
  4. INFO  [Thread-2] SemaphoreRunner - Thread 3 end! 
  5. INFO  [Thread-5] SemaphoreRunner - Thread 6 start! 
  6. INFO  [Thread-6] SemaphoreRunner - Thread 7 end! 
  7. INFO  [Thread-9] SemaphoreRunner - Thread 10 start! 
  8. INFO  [Thread-5] SemaphoreRunner - Thread 6 end! 
  9. INFO  [Thread-1] SemaphoreRunner - Thread 2 start! 
  10. INFO  [Thread-7] SemaphoreRunner - Thread 8 end! 
  11. INFO  [Thread-10] SemaphoreRunner - Thread 11 start! 
  12. INFO  [Thread-10] SemaphoreRunner - Thread 11 end! 
  13. INFO  [Thread-12] SemaphoreRunner - Thread 13 start! 
  14. INFO  [Thread-1] SemaphoreRunner - Thread 2 end! 
  15. INFO  [Thread-3] SemaphoreRunner - Thread 4 start! 
  16. INFO  [Thread-9] SemaphoreRunner - Thread 10 end! 
  17. INFO  [Thread-0] SemaphoreRunner - Thread 1 start! 
  18. INFO  [Thread-3] SemaphoreRunner - Thread 4 end! 
  19. INFO  [Thread-14] SemaphoreRunner - Thread 15 start! 
  20. INFO  [Thread-12] SemaphoreRunner - Thread 13 end! 
  21. INFO  [Thread-0] SemaphoreRunner - Thread 1 end! 
  22. INFO  [Thread-13] SemaphoreRunner - Thread 14 start! 
  23. INFO  [Thread-11] SemaphoreRunner - Thread 12 start! 
  24. INFO  [Thread-13] SemaphoreRunner - Thread 14 end! 
  25. INFO  [Thread-4] SemaphoreRunner - Thread 5 start! 
  26. INFO  [Thread-4] SemaphoreRunner - Thread 5 end! 
  27. INFO  [Thread-8] SemaphoreRunner - Thread 9 start! 
  28. INFO  [Thread-11] SemaphoreRunner - Thread 12 end! 
  29. INFO  [Thread-14] SemaphoreRunner - Thread 15 end! 
  30. INFO  [Thread-8] SemaphoreRunner - Thread 9 end! 
  1. public class TestLock { 
  2.   
  3.     private Logger logger = Logger.getLogger(getClass()); 
  4.   
  5.     @Test 
  6.     public void testSemaphore() throws Exception { 
  7.         new Thread(new SemaphoreRunner(1)).start(); 
  8.         new Thread(new SemaphoreRunner(2)).start(); 
  9.         new Thread(new SemaphoreRunner(3)).start(); 
  10.         new Thread(new SemaphoreRunner(4)).start(); 
  11.         new Thread(new SemaphoreRunner(5)).start(); 
  12.         new Thread(new SemaphoreRunner(6)).start(); 
  13.         new Thread(new SemaphoreRunner(7)).start(); 
  14.         new Thread(new SemaphoreRunner(8)).start(); 
  15.         new Thread(new SemaphoreRunner(9)).start(); 
  16.         new Thread(new SemaphoreRunner(10)).start(); 
  17.         Thread.sleep(1000000L); 
  18.     }  
  19.    
  20. public class SemaphoreRunner implements Runnable { 
  21.   
  22.     private Logger logger = Logger.getLogger(getClass()); 
  23.   
  24.     private int flag; 
  25.   
  26.     public SemaphoreRunner(int flag) { 
  27.         this.flag = flag; 
  28.     } 
  29.   
  30.     @Override 
  31.     public void run() { 
  32.         Semaphore semaphore = new Semaphore(new ConsulClient(), 3, "mg-init"); 
  33.         try { 
  34.             if (semaphore.acquired(true)) { 
  35.                 // 獲取到信號量,執(zhí)行業(yè)務(wù)邏輯 
  36.                 logger.info("Thread " + flag + " start!"); 
  37.                 Thread.sleep(new Random().nextInt(10000)); 
  38.                 logger.info("Thread " + flag + " end!"); 
  39.             } 
  40.         } catch (Exception e) { 
  41.             e.printStackTrace(); 
  42.         } finally { 
  43.             try { 
  44.                 // 信號量釋放、Session鎖釋放、Session刪除 
  45.                 semaphore.release(); 
  46.             } catch (IOException e) { 
  47.                 e.printStackTrace(); 
  48.             } 
  49.         } 
  50.     } 

從測試結(jié)果,我們可以發(fā)現(xiàn)當(dāng)信號量持有者數(shù)量達到信號量上限3的時候,其他競爭者就開始進行等待了,只有當(dāng)某個持有者釋放信號量之后,才會有新的線程變成持有者,從而開始執(zhí)行自己的業(yè)務(wù)邏輯。所以,分布式信號量可以幫助我們有效的控制同時操作某個共享資源的并發(fā)數(shù)。

優(yōu)化建議

同前文一樣,這里只是做了簡單的實現(xiàn)。線上應(yīng)用還必須加入TTL的session清理以及對.lock資源中的無效holder進行清理的機制。

參考文檔:https://www.consul.io/docs/guides/semaphore.html

實現(xiàn)代碼

【本文為51CTO專欄作者“翟永超”的原創(chuàng)稿件,轉(zhuǎn)載請通過51CTO聯(lián)系作者獲取授權(quán)】

戳這里,看該作者更多好文

責(zé)任編輯:趙寧寧 來源: 51CTO專欄
相關(guān)推薦

2017-04-13 10:51:09

Consul分布式

2010-04-21 16:25:13

Unix信號量

2010-04-21 16:42:48

Unix信號量

2025-05-16 08:58:47

Mongodb分布式存儲

2022-10-27 10:44:14

分布式Zookeeper

2021-04-13 09:20:15

鴻蒙HarmonyOS應(yīng)用開發(fā)

2020-11-05 09:59:24

Linux內(nèi)核信號量

2010-04-21 16:50:31

Unix信號量

2009-06-19 14:23:41

RMIJava分布式計算

2023-01-06 09:19:12

Seata分布式事務(wù)

2015-04-21 09:39:03

javajava分布式爬蟲

2017-10-24 11:28:23

Zookeeper分布式鎖架構(gòu)

2021-09-07 07:53:42

Semaphore 信號量源碼

2020-09-25 07:34:40

Linux系統(tǒng)編程信號量

2010-04-21 15:37:38

Unix信號量

2022-11-06 19:28:02

分布式鎖etcd云原生

2009-12-08 12:14:43

2013-08-21 14:06:05

iOS隊列信號

2022-03-08 15:24:23

BitMapRedis數(shù)據(jù)

2025-04-01 00:44:04

點贊
收藏

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

日本精品免费一区二区三区| 欧美高清激情brazzers| 精品乱色一区二区中文字幕| 在线观看国产亚洲| 最新国产精品视频| 欧美日韩精品一区二区三区蜜桃 | 欧美激情久久久久久久久久久| 午夜精品一区二区三区电影天堂| 欧美重口乱码一区二区| 一本色道久久综合精品婷婷| 狠狠干成人综合网| 国产一区二区日韩| 日韩大尺度视频| 日本综合久久| 欧美成人综合在线| 欧美日韩卡一| 婷婷亚洲久悠悠色悠在线播放| 日韩av一区二区三区在线观看| 国产精品久久久久久久久毛片| 亚洲午夜av| 色yeye香蕉凹凸一区二区av| 日韩av手机在线播放| 亚州精品国产| 欧美色播在线播放| 国产制服91一区二区三区制服| 国产香蕉在线| av在线不卡免费看| 欧美色窝79yyyycom| 日本精品免费视频| 黄色在线免费观看大全| 国产1区2区3区精品美女| 国产精品极品美女在线观看免费 | 亚洲乱码精品久久久久..| 日日夜夜精品视频免费| 久久免费精品日本久久中文字幕| 又色又爽的视频| 无码日韩精品一区二区免费| 欧美不卡123| 亚洲欧美日本一区二区| 美女网站视频一区| 欧美天天综合色影久久精品| 日韩精品综合在线| 99视频免费在线观看| 国产精品久久久久一区二区三区共 | 香蕉视频在线网址| 阿v免费在线观看| 久久综合网色—综合色88| 国产精品国产一区二区| www.四虎在线观看| 国产精品影视在线| 91免费国产网站| 国产影视一区二区| 久久99精品久久久久久动态图 | 国产亚洲福利一区| 日本高清www| 亚洲电影一级片| 精品福利在线观看| 欧美性潮喷xxxxx免费视频看| 欧美96在线| 国产精品家庭影院| 一区二区在线观| 午夜影院日韩| 欧美成人r级一区二区三区| 亚洲免费成人在线视频| 欧美一级网址| 7777精品伊人久久久大香线蕉 | 午夜激情视频在线播放| 国产一区二区三区电影在线观看 | 国产美女永久免费无遮挡| 九一亚洲精品| 在线电影av不卡网址| 99久久99久久精品免费看小说.| 精品国产乱码久久久| 中文字幕av一区二区| 蜜桃av.com| 亚洲精品电影| 久久人人爽人人| 青青国产在线观看| 日本91福利区| 亚洲va国产va天堂va久久| 超碰在线人人干| 成人免费视频视频| 蜜桃av噜噜一区二区三区| 免费一级在线观看| 中文字幕av一区二区三区高| 最新av在线免费观看| 亚洲男同gay网站| 精品日本美女福利在线观看| 日韩有码免费视频| 国产精品1区| 日韩高清免费观看| www.涩涩爱| 激情综合中文娱乐网| 日韩av免费看| 国产成人精品一区二区无码呦| 成人爱爱电影网址| 日本一区二区视频| 羞羞视频在线免费国产| 黑人巨大精品欧美一区二区一视频| 成人免费毛片网| www.成人在线.com| 日韩成人免费视频| 色老板免费视频| 国产日韩欧美| 91网站免费观看| 日中文字幕在线| 亚洲乱码中文字幕| 国产真实乱子伦| 欧美视频精品全部免费观看| 国产视频精品免费播放| a级片在线观看免费| 噜噜噜久久亚洲精品国产品小说| 91久久国产精品91久久性色| 香蕉视频黄色片| 亚洲猫色日本管| 熟女人妇 成熟妇女系列视频| 老司机亚洲精品一区二区| 亚洲午夜久久久影院| 久久精品国产亚洲av香蕉| 美腿丝袜一区二区三区| 久久久水蜜桃| 综合久久2o19| 欧美日韩mp4| 爱爱免费小视频| 亚洲国产免费| 51蜜桃传媒精品一区二区| av中文在线| 日韩欧美精品免费在线| 国产人妖在线观看| 99精品美女| 国产精品免费一区二区三区都可以| 黄色一级a毛片| 亚洲乱码国产乱码精品精的特点| 一区二区三区免费播放| 免费一区二区三区视频导航| 久久久噜噜噜久久| www.狠狠干| 亚洲同性同志一二三专区| 亚洲黄色a v| 精品在线观看入口| 69av视频在线播放| 日韩一级片免费看| 亚洲五月六月丁香激情| 手机看片国产精品| 中文视频一区| 亚洲在线免费视频| av在线app| 欧美一区二区免费| 中文字幕电影av| 国产在线精品一区二区不卡了| 亚洲午夜精品久久久久久浪潮| 写真福利精品福利在线观看| 日韩精品视频在线播放| 日韩精品在线观看免费| 91色视频在线| 国产网站免费在线观看| 亚洲人成亚洲精品| 日本精品视频在线| porn视频在线观看| 欧美日韩高清影院| 9999热视频| 国产盗摄一区二区三区| 欧美交换配乱吟粗大25p| 精品欧美视频| 久久理论片午夜琪琪电影网| 五月天婷婷在线播放| 欧美日韩亚洲激情| 九九九视频在线观看| 极品美女销魂一区二区三区 | **亚洲第一综合导航网站| 成年人黄视频在线观看| 精品欧美久久久| 日本亚洲色大成网站www久久| 99久久伊人网影院| 午夜免费精品视频| 色天天久久综合婷婷女18| 91在线免费网站| av资源在线看片| 亚洲人成网站777色婷婷| 国产精品sm调教免费专区| 综合久久综合久久| 亚洲一区和二区| 男人天堂欧美日韩| 致1999电视剧免费观看策驰影院| 国产亚洲久久| 91国内揄拍国内精品对白| 国产视频网址在线| 欧美一区午夜视频在线观看| 久久国产精品系列| 国产精品久久久爽爽爽麻豆色哟哟| 少妇性l交大片7724com| 美女久久网站| 精品国产无码在线| 天堂俺去俺来也www久久婷婷| 国产精品美女久久久免费| 深夜国产在线播放| 亚洲欧美日韩中文在线| 国产同性人妖ts口直男| 日韩欧美国产视频| 男人的天堂久久久| 国产亚洲欧美在线| 日本少妇一级片| 男女视频一区二区| 日韩精品xxxx| 午夜久久久久| 日韩视频专区| 精品视频自拍| 亚洲在线一区二区| 欧美大胆性生话| 久久久久久一区二区三区| 国产大学生校花援交在线播放| 日韩你懂的电影在线观看| 波多野结衣在线观看视频| 亚洲国产精品影院| 国产精品成人69xxx免费视频| 成人免费看视频| 91小视频在线播放| 丝袜脚交一区二区| 亚洲熟妇无码一区二区三区| 91亚洲国产| 日韩欧美99| 日韩精品福利一区二区三区| av一区二区三区四区电影| 国产伊人久久| 日本精品一区二区三区在线播放视频 | 国产精品成人3p一区二区三区| 日韩av第一页| 不卡一二三区| 97超级碰碰人国产在线观看| 亚洲精品白浆| 九九热最新视频//这里只有精品| 91伦理视频在线观看| 亚洲桃花岛网站| 天堂а在线中文在线无限看推荐| 欧美本精品男人aⅴ天堂| 一级黄色录像大片| 欧美欧美午夜aⅴ在线观看| 日本熟女毛茸茸| 欧美性猛交xxxx免费看| 国产精品美女久久久久av爽| 亚洲成人综合在线| 日本天堂在线视频| 婷婷成人综合网| 午夜影院在线看| 亚洲.国产.中文慕字在线| 精品少妇久久久| 亚洲成av人片在线| 日本少妇bbwbbw精品| 亚洲国产日韩a在线播放性色| 少妇影院在线观看| 亚洲免费av观看| 久草视频免费在线播放| 亚洲精品成a人| 久久国产在线视频| 亚洲电影中文字幕在线观看| 久久久久久久极品内射| 午夜欧美大尺度福利影院在线看| 国产中文字幕免费| 精品国产户外野外| 亚洲成人第一网站| 欧美午夜精品一区| 一本一道精品欧美中文字幕| 欧美一区二区三区视频| 国产欧美综合视频| 日韩精品专区在线| 日韩一区二区三区不卡| 精品视频在线播放| 国产福利片在线| 欧美另类在线观看| sm在线播放| 欧美一区二区三区图| 另类中文字幕国产精品| 成人免费视频网址| silk一区二区三区精品视频| 精品亚洲欧美日韩| 日韩精品一区二区三区免费观影| 午夜在线视频免费观看| 国内自拍一区| 国产av无码专区亚洲精品| 免费xxxx性欧美18vr| 成人一区二区三区仙踪林| 91丨九色丨蝌蚪丨老版| 中文字幕精品亚洲| 亚洲综合激情小说| 男人天堂av在线播放| 欧美精品在线观看播放| 亚洲黄色一级大片| 亚洲性69xxxbbb| 97超碰在线公开在线看免费| 午夜精品一区二区三区在线| 日韩不卡视频在线观看| 97欧洲一区二区精品免费| 蜜乳av综合| 超碰10000| 丝袜诱惑制服诱惑色一区在线观看 | 久久精品毛片| 久久精品无码一区二区三区毛片| 972aa.com艺术欧美| 久艹在线观看视频| 精品久久久久久久大神国产| 在线观看国产一区二区三区| 欧美成人激情免费网| 午夜不卡视频| 5252色成人免费视频| 国语精品视频| 欧美一区二区高清在线观看| 欧美福利影院| 蜜臀av免费观看| 94色蜜桃网一区二区三区| 91免费公开视频| 在线区一区二视频| 熟妇高潮一区二区高潮| 精品久久久av| 欧美大电影免费观看| 国产欧美日韩在线播放| 久久一区二区中文字幕| 久久久噜噜噜www成人网| 国产白丝精品91爽爽久久| 丁香六月激情综合| 一本色道a无线码一区v| 少妇人妻精品一区二区三区| 久久中文字幕国产| 欧美日韩视频免费看| 欧美日韩免费精品| 国产精品久久久久久模特| xxxx视频在线观看| 亚洲日韩欧美一区二区在线| 中文字幕欧美色图| 亚洲人成绝费网站色www| 激情黄产视频在线免费观看| caoporen国产精品| 亚洲综合中文| 在线视频观看一区二区| 国产精品每日更新| 波多野结衣影片| 亚洲日本成人女熟在线观看| 狠狠躁少妇一区二区三区| 国产精品国产三级国产专区53 | 蜜臀久久99精品久久久久久宅男 | 国产精品三区www17con| 欧美一区精品| 亚洲天堂伊人网| 国产精品国产三级国产普通话99| 成人午夜精品视频| 一个人看的www久久| 日本综合视频| 日韩色妇久久av| 久久成人免费网| 天堂а√在线中文在线鲁大师| 欧美三级乱人伦电影| 天堂аⅴ在线地址8| 国产免费久久av| 国产高清一区| 永久看看免费大片| 亚洲影院在线观看| 欧美亚洲精品在线观看| 欧美极品少妇xxxxⅹ免费视频| caoporn成人| 日本在线xxx| 久久久亚洲精品石原莉奈| 男人天堂2024| 中文字幕在线观看亚洲| 国产精品亚洲成在人线| 在线观看成人免费| 国产精品996| 国产一区二区三区影院| 亚洲精品视频久久| 成人黄页网站视频| 26uuu成人| 成人免费三级在线| 国产婷婷色一区二区在线观看| 国产亚洲人成a一在线v站| 久久久久久久性潮| 妞干网这里只有精品| 国产成人av一区二区| 免费观看成人毛片| 中文字幕日韩综合av| 精品国产乱码一区二区三区| youjizz.com在线观看| 久久午夜老司机| 97人妻人人澡人人爽人人精品| 欧美成人性色生活仑片| 五月天亚洲色图| 日韩成人av免费| 天天综合天天做天天综合| 在线看的av网站| 国产一区二区三区av在线| 葵司免费一区二区三区四区五区| 男人的午夜天堂| 日韩成人激情在线| 91丨精品丨国产| 丁香花在线影院观看在线播放| 久久精子c满五个校花| av网站免费大全| 日韩美女在线看| 中文字幕一区二区av | 免费观看成人在线视频| 亚洲欧美日韩中文播放| 青青草视频免费在线观看| 亚洲aⅴ男人的天堂在线观看|