国产精品电影_久久视频免费_欧美日韩国产激情_成年人视频免费在线播放_日本久久亚洲电影_久久都是精品_66av99_九色精品美女在线_蜜臀a∨国产成人精品_冲田杏梨av在线_欧美精品在线一区二区三区_麻豆mv在线看

老大吩咐的可重入分布式鎖,終于實現了~

開發 后端 分布式
最近在做一個項目,將一個其他公司的實現系統(下文稱作舊系統),完整的整合到自己公司的系統(下文稱作新系統)中,這其中需要將對方實現的功能完整在自己系統也實現一遍。

[[330080]]

本文轉載自微信公眾號「程序通事」,作者樓下小黑哥  。轉載本文請聯系程序通事公眾號。

重做永遠比改造簡單

最近在做一個項目,將一個其他公司的實現系統(下文稱作舊系統),完整的整合到自己公司的系統(下文稱作新系統)中,這其中需要將對方實現的功能完整在自己系統也實現一遍。

舊系統還有一批存量商戶,為了不影響存量商戶的體驗,新系統提供的對外接口,還必須得跟以前一致。最后系統完整切換之后,功能只運行在新系統中,這就要求舊系統的數據還需要完整的遷移到新系統中。

當然這些在做這個項目之前就有預期,想過這個過程很難,但是沒想到有那么難。原本感覺排期大半年,時間還是挺寬裕,現在感覺就是大坑,還不得不在坑里一點點去填。

哎,說多都是淚,不吐槽了,等到下次做完再給大家復盤下真正心得體會。

回到正文,上篇文章Redis 分布式鎖,咱們基于 Redis 實現一個分布式鎖。這個分布式鎖基本功能沒什么問題,但是缺少可重入的特性,所以這篇文章小黑哥就帶大家來實現一下可重入的分布式鎖。

本篇文章將會涉及以下內容:

  • 可重入
  • 基于 ThreadLocal 實現方案
  • 基于 Redis Hash 實現方案

可重入

說到可重入鎖,首先我們來看看一段來自 wiki 上可重入的解釋:

“若一個程序或子程序可以“在任意時刻被中斷然后操作系統調度執行另外一段代碼,這段代碼又調用了該子程序不會出錯”,則稱其為可重入(reentrant或re-entrant)的。即當該子程序正在運行時,執行線程可以再次進入并執行它,仍然獲得符合設計時預期的結果。與多線程并發執行的線程安全不同,可重入強調對單個線程執行時重新進入同一個子程序仍然是安全的。

當一個線程執行一段代碼成功獲取鎖之后,繼續執行時,又遇到加鎖的代碼,可重入性就就保證線程能繼續執行,而不可重入就是需要等待鎖釋放之后,再次獲取鎖成功,才能繼續往下執行。

用一段 Java 代碼解釋可重入:

  1. public synchronized void a() { 
  2.     b(); 
  3.  
  4. public synchronized void b() { 
  5.     // pass 

假設 X 線程在 a 方法獲取鎖之后,繼續執行 b 方法,如果此時不可重入,線程就必須等待鎖釋放,再次爭搶鎖。

鎖明明是被 X 線程擁有,卻還需要等待自己釋放鎖,然后再去搶鎖,這看起來就很奇怪,我釋放我自己~

可重入性就可以解決這個尷尬的問題,當線程擁有鎖之后,往后再遇到加鎖方法,直接將加鎖次數加 1,然后再執行方法邏輯。退出加鎖方法之后,加鎖次數再減 1,當加鎖次數為 0 時,鎖才被真正的釋放。

可以看到可重入鎖最大特性就是計數,計算加鎖的次數。所以當可重入鎖需要在分布式環境實現時,我們也就需要統計加鎖次數。

分布式可重入鎖實現方式有兩種:

  • 基于 ThreadLocal 實現方案
  • 基于 Redis Hash 實現方案

首先我們看下基于 ThreadLocal 實現方案。

基于 ThreadLocal 實現方案

實現方式

Java 中 ThreadLocal可以使每個線程擁有自己的實例副本,我們可以利用這個特性對線程重入次數進行計數。

下面我們定義一個ThreadLocal的全局變量 LOCKS,內存存儲 Map 實例變量。

  1. private static ThreadLocal<Map<String, Integer>> LOCKS = ThreadLocal.withInitial(HashMap::new); 

每個線程都可以通過 ThreadLocal獲取自己的 Map實例,Map 中 key 存儲鎖的名稱,而 value存儲鎖的重入次數。

加鎖的代碼如下:

  1. /** 
  2.  * 可重入鎖 
  3.  * 
  4.  * @param lockName  鎖名字,代表需要爭臨界資源 
  5.  * @param request   唯一標識,可以使用 uuid,根據該值判斷是否可以重入 
  6.  * @param leaseTime 鎖釋放時間 
  7.  * @param unit      鎖釋放時間單位 
  8.  * @return 
  9.  */ 
  10. public Boolean tryLock(String lockName, String request, long leaseTime, TimeUnit unit) { 
  11.     Map<String, Integer> counts = LOCKS.get(); 
  12.     if (counts.containsKey(lockName)) { 
  13.         counts.put(lockName, counts.get(lockName) + 1); 
  14.         return true
  15.     } else { 
  16.         if (redisLock.tryLock(lockName, request, leaseTime, unit)) { 
  17.             counts.put(lockName, 1); 
  18.             return true
  19.         } 
  20.     } 
  21.     return false

“ps: redisLock#tryLock 為上一篇文章實現的分布鎖。由于公號外鏈無法直接跳轉,關注『程序通事』,回復分布式鎖獲取源代碼。

加鎖方法首先判斷當前線程是否已經已經擁有該鎖,若已經擁有,直接對鎖的重入次數加 1。

若還沒擁有該鎖,則嘗試去 Redis 加鎖,加鎖成功之后,再對重入次數加 1 。

釋放鎖的代碼如下:

  1. /** 
  2.  * 解鎖需要判斷不同線程池 
  3.  * 
  4.  * @param lockName 
  5.  * @param request 
  6.  */ 
  7. public void unlock(String lockName, String request) { 
  8.     Map<String, Integer> counts = LOCKS.get(); 
  9.     if (counts.getOrDefault(lockName, 0) <= 1) { 
  10.         counts.remove(lockName); 
  11.         Boolean result = redisLock.unlock(lockName, request); 
  12.         if (!result) { 
  13.             throw new IllegalMonitorStateException("attempt to unlock lock, not locked by lockName:+" + lockName + " with request: " 
  14.                     + request); 
  15.         } 
  16.  
  17.     } else { 
  18.         counts.put(lockName, counts.get(lockName) - 1); 
  19.     } 

釋放鎖的時首先判斷重入次數,若大于 1,則代表該鎖是被該線程擁有,所以直接將鎖重入次數減 1 即可。

若當前可重入次數小于等于 1,首先移除 Map中鎖對應的 key,然后再到 Redis 釋放鎖。

這里需要注意的是,當鎖未被該線程擁有,直接解鎖,可重入次數也是小于等于 1 ,這次可能無法直接解鎖成功。

“ThreadLocal 使用過程要記得及時清理內部存儲實例變量,防止發生內存泄漏,上下文數據串用等問題。下次咱來聊聊最近使用 ThreadLocal 寫的 Bug。

相關問題

使用 ThreadLocal 這種本地記錄重入次數,雖然真的簡單高效,但是也存在一些問題。

過期時間問題

上述加鎖的代碼可以看到,重入加鎖時,僅僅對本地計數加 1 而已。這樣可能就會導致一種情況,由于業務執行過長,Redis 已經過期釋放鎖。

而再次重入加鎖時,由于本地還存在數據,認為鎖還在被持有,這就不符合實際情況。

如果要在本地增加過期時間,還需要考慮本地與 Redis 過期時間一致性的,代碼就會變得很復雜。

不同線程/進程可重入問題

狹義上可重入性應該只是對于同一線程的可重入,但是實際業務可能需要不同的應用線程之間可以重入同把鎖。

而 ThreadLocal的方案僅僅只能滿足同一線程重入,無法解決不同線程/進程之間重入問題。

不同線程/進程重入問題就需要使用下述方案 Redis Hash 方案解決。

基于 Redis Hash 可重入鎖

實現方式

ThreadLocal 的方案中我們使用了 Map 記載鎖的可重入次數,而 Redis 也同樣提供了 Hash (哈希表)這種可以存儲鍵值對數據結構。所以我們可以使用 Redis Hash 存儲的鎖的重入次數,然后利用 lua 腳本判斷邏輯。

加鎖的 lua 腳本如下:

  1. ---- 1 代表 true 
  2. ---- 0 代表 false 
  3.  
  4. if (redis.call('exists', KEYS[1]) == 0) then 
  5.     redis.call('hincrby', KEYS[1], ARGV[2], 1); 
  6.     redis.call('pexpire', KEYS[1], ARGV[1]); 
  7.     return 1; 
  8. end ; 
  9. if (redis.call('hexists', KEYS[1], ARGV[2]) == 1) then 
  10.     redis.call('hincrby', KEYS[1], ARGV[2], 1); 
  11.     redis.call('pexpire', KEYS[1], ARGV[1]); 
  12.     return 1; 
  13. end ; 
  14. return 0; 

“如果 KEYS:[lock],ARGV[1000,uuid]

不熟悉 lua 語言同學也不要怕,上述邏輯還是比較簡單的。

加鎖代碼首先使用 Redis exists 命令判斷當前 lock 這個鎖是否存在。

如果鎖不存在的話,直接使用 hincrby創建一個鍵為 lock hash 表,并且為 Hash 表中鍵為 uuid 初始化為 0,然后再次加 1,最后再設置過期時間。

如果當前鎖存在,則使用 hexists判斷當前 lock 對應的 hash 表中是否存在 uuid 這個鍵,如果存在,再次使用 hincrby 加 1,最后再次設置過期時間。

最后如果上述兩個邏輯都不符合,直接返回。

加鎖代碼如下:

  1. // 初始化代碼 
  2.  
  3. String lockLuaScript = IOUtils.toString(ResourceUtils.getURL("classpath:lock.lua").openStream(), Charsets.UTF_8); 
  4. lockScript = new DefaultRedisScript<>(lockLuaScript, Boolean.class); 
  5.  
  6. /** 
  7.  * 可重入鎖 
  8.  * 
  9.  * @param lockName  鎖名字,代表需要爭臨界資源 
  10.  * @param request   唯一標識,可以使用 uuid,根據該值判斷是否可以重入 
  11.  * @param leaseTime 鎖釋放時間 
  12.  * @param unit      鎖釋放時間單位 
  13.  * @return 
  14.  */ 
  15. public Boolean tryLock(String lockName, String request, long leaseTime, TimeUnit unit) { 
  16.     long internalLockLeaseTime = unit.toMillis(leaseTime); 
  17.     return stringRedisTemplate.execute(lockScript, Lists.newArrayList(lockName), String.valueOf(internalLockLeaseTime), request); 

“Spring-Boot 2.2.7.RELEASE

只要搞懂 Lua 腳本加鎖邏輯,Java 代碼實現還是挺簡單的,直接使用 SpringBoot 提供的 StringRedisTemplate 即可。

解鎖的 Lua 腳本如下:

  1. -- 判斷 hash set 可重入 key 的值是否等于 0 
  2. -- 如果為 0 代表 該可重入 key 不存在 
  3. if (redis.call('hexists', KEYS[1], ARGV[1]) == 0) then 
  4.     return nil; 
  5. end ; 
  6. -- 計算當前可重入次數 
  7. local counter = redis.call('hincrby', KEYS[1], ARGV[1], -1); 
  8. -- 小于等于 0 代表可以解鎖 
  9. if (counter > 0) then 
  10.     return 0; 
  11. else 
  12.     redis.call('del', KEYS[1]); 
  13.     return 1; 
  14. end ; 
  15. return nil; 

首先使用 hexists 判斷 Redis Hash 表是否存給定的域。

如果 lock 對應 Hash 表不存在,或者 Hash 表不存在 uuid 這個 key,直接返回 nil。

若存在的情況下,代表當前鎖被其持有,首先使用 hincrby使可重入次數減 1 ,然后判斷計算之后可重入次數,若小于等于 0,則使用 del 刪除這把鎖。

解鎖的 Java 代碼如下:

  1. // 初始化代碼: 
  2.  
  3.  
  4. String unlockLuaScript = IOUtils.toString(ResourceUtils.getURL("classpath:unlock.lua").openStream(), Charsets.UTF_8); 
  5. unlockScript = new DefaultRedisScript<>(unlockLuaScript, Long.class); 
  6.  
  7. /** 
  8.  * 解鎖 
  9.  * 若可重入 key 次數大于 1,將可重入 key 次數減 1 <br> 
  10.  * 解鎖 lua 腳本返回含義:<br> 
  11.  * 1:代表解鎖成功 <br> 
  12.  * 0:代表鎖未釋放,可重入次數減 1 <br> 
  13.  * nil:代表其他線程嘗試解鎖 <br> 
  14.  * <p> 
  15.  * 如果使用 DefaultRedisScript<Boolean>,由于 Spring-data-redis eval 類型轉化,<br> 
  16.  * 當 Redis 返回  Nil bulk, 默認將會轉化為 false,將會影響解鎖語義,所以下述使用:<br> 
  17.  * DefaultRedisScript<Long> 
  18.  * <p> 
  19.  * 具體轉化代碼請查看:<br> 
  20.  * JedisScriptReturnConverter<br> 
  21.  * 
  22.  * @param lockName 鎖名稱 
  23.  * @param request  唯一標識,可以使用 uuid 
  24.  * @throws IllegalMonitorStateException 解鎖之前,請先加鎖。若為加鎖,解鎖將會拋出該錯誤 
  25.  */ 
  26. public void unlock(String lockName, String request) { 
  27.     Long result = stringRedisTemplate.execute(unlockScript, Lists.newArrayList(lockName), request); 
  28.     // 如果未返回值,代表其他線程嘗試解鎖 
  29.     if (result == null) { 
  30.         throw new IllegalMonitorStateException("attempt to unlock lock, not locked by lockName:+" + lockName + " with request: " 
  31.                 + request); 
  32.     } 

解鎖代碼執行方式與加鎖類似,只不過解鎖的執行結果返回類型使用 Long。這里之所以沒有跟加鎖一樣使用 Boolean ,這是因為解鎖 lua 腳本中,三個返回值含義如下:

  • 1 代表解鎖成功,鎖被釋放
  • 0 代表可重入次數被減 1
  • null 代表其他線程嘗試解鎖,解鎖失敗

如果返回值使用 Boolean,Spring-data-redis 進行類型轉換時將會把 null 轉為 false,這就會影響我們邏輯判斷,所以返回類型只好使用 Long。

以下代碼來自 JedisScriptReturnConverter:

相關問題

spring-data-redis 低版本問題

如果 Spring-Boot 使用 Jedis 作為連接客戶端,并且使用Redis Cluster 集群模式,需要使用 2.1.9 以上版本的spring-boot-starter-data-redis,不然執行過程中將會拋出:

  1. org.springframework.dao.InvalidDataAccessApiUsageException: EvalSha is not supported in cluster environment. 

如果當前應用無法升級 spring-data-redis也沒關系,可以使用如下方式,直接使用原生 Jedis 連接執行 lua 腳本。

以加鎖代碼為例:

  1. public boolean tryLock(String lockName, String reentrantKey, long leaseTime, TimeUnit unit) { 
  2.     long internalLockLeaseTime = unit.toMillis(leaseTime); 
  3.     Boolean result = stringRedisTemplate.execute((RedisCallback<Boolean>) connection -> { 
  4.         Object innerResult = eval(connection.getNativeConnection(), lockScript, Lists.newArrayList(lockName), Lists.newArrayList(String.valueOf(internalLockLeaseTime), reentrantKey)); 
  5.         return convert(innerResult); 
  6.     }); 
  7.     return result; 
  8.  
  9. private Object eval(Object nativeConnection, RedisScript redisScript, final List<String> keys, final List<String> args) { 
  10.  
  11.     Object innerResult = null
  12.     // 集群模式和單點模式雖然執行腳本的方法一樣,但是沒有共同的接口,所以只能分開執行 
  13.     // 集群 
  14.     if (nativeConnection instanceof JedisCluster) { 
  15.         innerResult = evalByCluster((JedisCluster) nativeConnection, redisScript, keys, args); 
  16.     } 
  17.     // 單點 
  18.     else if (nativeConnection instanceof Jedis) { 
  19.         innerResult = evalBySingle((Jedis) nativeConnection, redisScript, keys, args); 
  20.     } 
  21.     return innerResult; 

數據類型轉化問題

如果使用 Jedis 原生連接執行 Lua 腳本,那么可能又會碰到數據類型的轉換坑。

可以看到 Jedis#eval返回 Object,我們需要具體根據 Lua 腳本的返回值的,再進行相關轉化。這其中就涉及到 Lua 數據類型轉化為 Redis 數據類型。

下面主要我們來講下 Lua 數據轉化 Redis 的規則中幾條比較容易踩坑:

1、Lua number 與 Redis 數據類型轉換

Lua 中 number 類型是一個雙精度的浮點數,但是 Redis 只支持整數類型,所以這個轉化過程將會丟棄小數位。

2、Lua boolean 與 Redis 類型轉換

這個轉化比較容易踩坑,Redis 中是不存在 boolean 類型,所以當Lua 中 true 將會轉為 Redis 整數 1。而 Lua 中 false 并不是轉化整數,而是轉化 null 返回給客戶端。

3、Lua nil 與 Redis 類型轉換

Lua nil 可以當做是一個空值,可以等同于 Java 中的 null。在 Lua 中如果 nil 出現在條件表達式,將會當做 false 處理。

所以 Lua nil 也將會 null 返回給客戶端。

其他轉化規則比較簡單,詳情參考:

http://doc.redisfans.com/script/eval.html

總結

可重入分布式鎖關鍵在于對于鎖重入的計數,這篇文章主要給出兩種解決方案,一種基于 ThreadLocal 實現方案,這種方案實現簡單,運行也比較高效。但是若要處理鎖過期的問題,代碼實現就比較復雜。

另外一種采用 Redis Hash 數據結構實現方案,解決了 ThreadLocal 的缺陷,但是代碼實現難度稍大,需要熟悉 Lua 腳本,以及Redis 一些命令。另外使用 spring-data-redis 等操作 Redis 時不經意間就會遇到各種問題。

幫助

https://www.sofastack.tech/blog/sofa-jraft-rheakv-distributedlock/

https://tech.meituan.com/2016/09/29/distributed-system-mutually-exclusive-idempotence-cerberus-gtis.html

 

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

2021-07-08 09:21:17

ZooKeeper分布式鎖 Curator

2021-06-27 21:24:55

RedissonJava數據

2024-01-30 08:41:33

線程執行Redis分布式鎖

2021-07-10 10:02:30

ZooKeeperCurator并發

2024-02-04 09:29:07

Redis數據庫

2021-07-09 06:48:31

ZooKeeperCurator源碼

2024-11-28 15:11:28

2019-06-19 15:40:06

分布式鎖RedisJava

2021-02-28 07:49:28

Zookeeper分布式

2017-01-16 14:13:37

分布式數據庫

2017-04-13 10:51:09

Consul分布式

2018-04-03 16:24:34

分布式方式

2022-04-08 08:27:08

分布式鎖系統

2019-02-26 09:51:52

分布式鎖RedisZookeeper

2023-08-21 19:10:34

Redis分布式

2021-10-25 10:21:59

ZK分布式鎖ZooKeeper

2022-01-06 10:58:07

Redis數據分布式鎖

2022-12-31 09:42:14

超時功能

2020-04-10 08:03:04

分布式鎖Redlock算法流行算法

2018-11-23 09:25:00

TCC分布式事務
點贊
收藏

51CTO技術棧公眾號

狠狠躁18三区二区一区| 亚洲欧美日韩专区| 欧美日韩另类国产亚洲欧美一级| 91免费网站视频| 精品中文字幕一区二区三区| 日精品一区二区| 久久久精品999| 丝袜在线观看| 偷拍一区二区三区| 欧美午夜小视频| 国产精品原创视频| 制服丝袜在线91| 日本午夜视频| 本田岬高潮一区二区三区| 欧美福利一区二区三区| 日韩欧美三级| 国产精品久久av| 国产精品欧美一区二区三区不卡 | 肉大捧一出免费观看网站在线播放| 欧美三级午夜理伦三级小说| 欧美精品在线免费| 日韩av首页| 久久国产精品久久久久久久久久| 3344国产永久在线观看视频| 99久久精品情趣| 春日野结衣av| 中文字幕一区二区三区色视频| 九热视频在线观看| 国产精品毛片大码女人| 成人xxxx| 一区二区三区免费看视频| 99热在线看| 天天操天天干天天综合网| 青青国产在线| 精品成a人在线观看| 国产精品一二三产区| 亚洲深夜福利视频| 久久综合伊人77777蜜臀| 爱啪啪综合导航| 久久国产精品影视| 国产一区二区香蕉| 免费在线观看视频| 欧美三级在线视频| 黑森林国产精品av| 久久久亚洲成人| 亚洲小说区图片区| 免费毛片网站在线观看| 亚洲国产视频一区二区| 国产福利在线免费观看| 久久久久久久久中文字幕| 国内激情久久| 免费黄色特级片| 亚洲国产高清福利视频| 欧美色网在线| 成人国产精品一区| 成人免费黄色在线| 黄色av免费在线观看| 色久欧美在线视频观看| 亚洲精品一二三区区别| 另类亚洲自拍| 免费看污污网站| 日韩欧美一级二级三级| 亚洲人成亚洲精品| 老太脱裤让老头玩ⅹxxxx| 欧美系列一区二区| 国产精品天天看天天狠| 日本黄色a视频| 精品欧美一区二区三区| av在线日韩| 伊人久久av导航| 欧美日韩小视频| 欧美日韩一区二区三区四区不卡| 国产精品视频一| 午夜影院欧美| 国产熟女高潮视频| 日韩一区二区麻豆国产| 欧美日韩精品| 韩国三级在线观看久| 国产极品jizzhd欧美| 国产99精品在线观看| 宅男在线观看免费高清网站| 亚洲伊人久久大香线蕉av| 偷窥少妇高潮呻吟av久久免费| 精品久久精品| 日韩av一级大片| 欧美在线你懂的| 在线观看国产精品入口| 少妇激情av一区二区三区| 久久在精品线影院精品国产| 99国产精品久久久久久久久久| 97超碰国产精品| 99久精品国产| 国产精品毛片无码| 久久精品国产sm调教网站演员| 亚洲国产小视频| 老司机午夜精品99久久| gogo高清在线播放免费| 老司机午夜免费福利视频| 深夜成人在线观看| 国产欧美精品在线观看| 老牛精品亚洲成av人片| 九色porny在线观看| 国产一区二区黄色| 亚洲美女在线视频| 91论坛在线播放| 日韩av不卡一区| 你懂的在线观看视频网站| 亚洲不卡1区| 91国内产香蕉| 欧美一区二区三区在线看| 99久久伊人久久99| 日韩av成人| 日韩亚洲一区在线播放| 超碰精品一区二区三区乱码| 丁香五六月婷婷久久激情| 国产精品18久久久久久久网站| 日韩av有码| 岛国av免费在线观看| 在线理论视频| 亚州精品一二三区| 国产大尺度在线观看| 成人av在线网址| 久久综合88中文色鬼| 欧美精品一区二区三区一线天视频| 亚洲精品国产无天堂网2021| 国产酒店精品激情| 日韩国产欧美一区二区三区| 日韩在线第七页| 亚洲人体在线| 成人黄色动漫| 午夜免费福利在线观看| 香蕉影院在线| 午夜影院免费播放| 97福利电影| 调教视频vk| 日本xxxxxxx免费视频| 先锋影音男人资源| 亚洲二区自拍| 一区二区三区三区在线| 国产伦精品一区二区三区在线| 国产精自产拍久久久久久| 久久久久久久97| 欧美黑人xxx| 日本a级片电影一区二区| 午夜精品视频在线| 欧美精品久久久久| 欧美在线免费视频| 国产日韩在线观看av| 日韩av手机在线看| 91成人免费在线观看| 欧美一区二区在线视频观看| 中文字幕一区二区三区四区五区六区| 日韩偷拍一区二区| 日韩激情视频在线观看| 女生裸体视频网站免费观看| 老太脱裤让老头玩ⅹxxxx| 欧美激情成人网| 香港日本韩国三级| 色鬼7777久久| 四虎在线免费看| 伊人手机在线| 好吊妞国产欧美日韩免费观看网站 | 久久精品视频导航| 欧美成人午夜激情视频| 欧美一区二区.| 2019中文字幕全在线观看| 国产精品久久91| 97视频热人人精品| 日韩欧美国产二区| 九色porn| 国产免费视频传媒| 欧美hdsex| 黄色在线观看网站| 高潮按摩久久久久久av免费| 久久99青青| 久久一区激情| 亚洲综合一区二区三区| 亚洲大胆美女视频| 俄罗斯精品一区二区三区| 久久久精品三级| 视频一区二区三区在线看免费看| 亚欧精品一区| 91网在线播放| av软件在线观看| 日本精品网站| 美女在线视频一区| 亚洲欧美另类久久久精品| 1区2区3区精品视频| 日韩免费高清视频| 国产精品国产精品| 国产三级在线免费观看| 国产欧美日韩影院| 成人av手机在线观看| 国产精品国产三级国产aⅴ无密码 国产精品国产三级国产aⅴ原创 | 日韩成人精品在线| 高清av一区| 欧美韩国日本在线观看| 成人精品福利视频| 久草免费福利在线| 在线视频专区|