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

張開濤:應用級緩存之緩存使用模式實踐—

開發 開發工具
前面已經介紹了Java緩存的使用,本文主要以Guava Cache和Ehcache3.x作為實踐框架來講解。

前面已經介紹了Java緩存的使用。對于我們來說如果有人總結一些緩存使用模式/模板的話,我們在使用時直接照著模式寫即可。而實際確實已經有總結好的模式,主要分兩大類:Cache-Aside和Cache-As-SoR(Read-through、Write-through、Write-behind)。

[[190403]]

首先,同步兩個名詞。

  • SoR(system-of-record):記錄系統,或者可以叫做數據源,即實際存儲原始數據的系統。
  • Cache:緩存,是SoR的快照數據,Cache的訪問速度比SoR要快,放入Cache的目的是提升訪問速度,減少回源到SoR的次數。
  • 回源:即回到數據源頭獲取數據,Cache沒有***時,需要從SoR讀取數據,這叫做回源。

本文主要以Guava Cache和Ehcache3.x作為實踐框架來講解。

一、Cache-Aside

Cache-Aside即業務代碼圍繞著Cache寫,是由業務代碼直接維護緩存,示例代碼如下所示。

讀場景,先從緩存獲取數據,如果沒有***,則回源到SoR并將源數據放入緩存供下次讀取使用。

  1. //1、先從緩存中獲取數據 
  2. value = myCache.getIfPresent(key); 
  3. if(value == null) { 
  4.     //2.1、如果緩存沒有***,則回源到SoR獲取源數據 
  5.     value = loadFromSoR(key); 
  6.     //2.2、將數據放入緩存,下次即可從緩存中獲取數據 
  7.     myCache.put(key, value); 

寫場景,先將數據寫入SoR,寫入成功后立即將數據同步寫入緩存。

  1. //1、先將數據寫入SoR 
  2. writeToSoR(key,value); 
  3. //2、執行成功后立即同步寫入緩存 
  4. myCache.put(key, value); 

或者先將數據寫入SoR,寫入成功后將緩存數據過期,下次讀取時再加載緩存。

  1. //1、先將數據寫入SoR 
  2. writeToSoR(key,value); 
  3. //2、失效緩存,然后下次讀時再加載緩存 
  4. myCache.invalidate(key); 

Cache-Aside適合使用AOP模式去實現,可以參考筆者的博客《Spring Cache抽象詳解》去實現。

對于Cache-Aside可能存在并發更新情況,即如果多個應用實例同時更新,那么緩存怎么辦?

● 如果是用戶維度的數據(如訂單數據、用戶數據),則出現這種幾率非常小,因為并發的情況很少,可以不考慮這個問題,加上過期時間來解決即可。

● 對于如商品這種基礎數據,可以考慮使用canal訂閱binlog進行增量更新分布式緩存,這樣不會存在緩存數據不一致的情況,但是,緩存更新會存在延遲。而本地緩存根據不一致容忍度設置合理的過期時間。

● 讀服務場景,可以考慮使用一致性哈希,將相同的操作負載均衡到同一個實例,從而減少并發幾率。或者設置比較短的過期時間,可參考“第17章 京東商品詳情頁服務閉環實踐”。

二、Cache-As-SoR

Cache-As-SoR即把Cache看作為SoR,所有操作都是對Cache進行,然后Cache再委托給SoR進行真實的讀/寫。即業務代碼中只看到Cache的操作,看不到關于SoR相關的代碼。有三種實現:read-through、write-through、write-behind。

1. Read-Through

Read-Through,業務代碼首先調用Cache,如果Cache不***由Cache回源到SoR,而不是業務代碼(即由Cache讀SoR)。使用Read-Through模式,需要配置一個CacheLoader組件用來回源到SoR加載源數據。Guava Cache和Ehcache 3.x都支持該模式。

Guava Cache實現

  1. LoadingCache<Integer,Result<Category>> getCache = 
  2.        CacheBuilder.newBuilder() 
  3.                .softValues() 
  4.                .maximumSize(5000).expireAfterWrite(2, TimeUnit.MINUTES) 
  5.                 .build(new CacheLoader<Integer,Result<Category>>() { 
  6.                    @Override 
  7.                    public Result<Category> load(final Integer sortId) throwsException { 
  8.                         return categoryService.get(sortId); 
  9.                    } 
  10.                }); 

在build Cache時,傳入一個CacheLoader用來加載緩存,操作流程如下。

  • 應用業務代碼直接調用getCache.get(sortId)。
  • 首先查詢Cache,如果緩存中有,則直接返回緩存數據。
  • 如果緩存沒有***,則委托給CacheLoader,CacheLoader會回源到SoR查詢源數據(返回值必須不為null,可以包裝為Null對象),然后寫入緩存。

使用CacheLoader后有幾個好處。

● 應用業務代碼更簡潔了,不需要像Cache-Aside模式那樣緩存查詢代碼和SoR代碼交織在一起。如果緩存使用邏輯散落在多處,則使用這種方式很簡單的消除了重復代碼。

● 解決Dog-pile effect,即當某個緩存失效時,又有大量相同的請求沒***緩存,從而同時請求到后端,導致后端壓力太大,此時限制一個請求去拿即可。

  1. if (firstCreateNewEntry) {//***個請求加載緩存的線程去SoR加載源數據 
  2.   try { 
  3.     synchronized (e) { 
  4.       returnloadSync(key, hash, loadingValueReference, loader); 
  5.     } 
  6.   } finally{ 
  7.     statsCounter.recordMisses(1); 
  8.   } 
  9. } else {//其他并發線程等待“***個線程”加載的數據 
  10.   return waitForLoadingValue(e, key,valueReference); 
  11.  
  12. Guava Cache還支持get(K key, Callable<? extends V> valueLoader)方法,傳入一個Callable實例,當緩存沒***時,會調用Callable#call來查詢SoR加載源數據。 
  13.  
  14. Ehcache 3.x實現 
  15. CacheManager cacheManager = CacheManagerBuilder. newCacheManagerBuilder(). build(true); 
  16. org.ehcache.Cache<String, String> myCache =cacheManager. createCache ("myCache", 
  17.        CacheConfigurationBuilder.newCacheConfigurationBuilder(String.class,String.class, 
  18.                ResourcePoolsBuilder.newResourcePoolsBuilder().heap(100,MemoryUnit.MB)) 
  19.                .withDispatcherConcurrency(4) 
  20.                .withExpiry(Expirations.timeToLiveExpiration(Duration.of(10,TimeUnit.SECONDS))) 
  21.                 .withLoaderWriter(newDefaultCacheLoaderWriter<String, String> () { 
  22.                    @Override 
  23.                    public String load(String key) throws Exception { 
  24.                         return readDB(key); 
  25.                    } 
  26.                     @Override 
  27.                    public Map<String, String> loadAll(Iterable<? extendsString> keys) throws BulkCacheLoadingException, Exception { 
  28.                         return null; 
  29.                    } 
  30.                })); 

Ehcache 3.x使用CacheLoaderWriter來實現,通過load(K key)和loadAll(Iterable keys)分別來加載單個KEY和批量KEY。Ehcache 3.1沒有自己去解決Dog-pile effect。

2. Write-Through

Write-Through,稱之為穿透寫模式/直寫模式,業務代碼首先調用Cache寫(新增/修改)數據,然后由Cache負責寫緩存和寫SoR,而不是業務代碼。使用Write-Through模式需要配置一個CacheWriter組件用來回寫SoR。Guava Cache沒有提供支持。Ehcache 3.x支持該模式。Ehcache需要配置一個CacheLoaderWriter,CacheLoaderWriter知道如何去寫SoR。當Cache需要寫(新增/修改)數據時,首先調用CacheLoaderWriter來同步(立即)到SoR,成功后會更新緩存。

  1. CacheManager cacheManager = CacheManagerBuilder.newCacheManagerBuilder().build(true); 
  2. org.ehcache.Cache<String, String> myCache =cacheManager.createCache ("myCache", 
  3.        CacheConfigurationBuilder.newCacheConfigurationBuilder(String.class,String.class, 
  4.                ResourcePoolsBuilder.newResourcePoolsBuilder().heap(100,MemoryUnit.MB)) 
  5.                .withDispatcherConcurrency(4) 
  6.                .withExpiry(Expirations.timeToLiveExpiration(Duration.of(10,TimeUnit.SECONDS))) 
  7.                 .withLoaderWriter(newDefaultCacheLoaderWriter<String, String> () { 
  8.                    @Override 
  9.                    public void write(String key, String value) throws Exception{ 
  10.                         //write 
  11.                    } 
  12.                    @Override 
  13.                    public void writeAll(Iterable<? extends Map.Entry<? extendsString, ? extends String>> entries) throws BulkCacheWritingException,Exception { 
  14.                         for(Object entry: entries) { 
  15.                             //batch write 
  16.                         } 
  17.                    } 
  18.                    @Override 
  19.                     public void delete(Stringkey) throws Exception { 
  20.                         //delete 
  21.                    } 
  22.                    @Override 
  23.                    public void deleteAll(Iterable<? extends String>keys) throws BulkCacheWritingException, Exception { 
  24.                         for(Object key :keys) { 
  25.                             //batch delete 
  26.                         } 
  27.                    } 
  28.                }).build()); 

Ehcache 3.x還是使用CacheLoaderWriter來實現,通過write(String key, String value)、writeAll(Iterable> entries)和delete(String key)、deleteAll(Iterable keys)分別來支持單個寫、批量寫和單個刪除、批量刪除操作。

操作流程如下。

  • 當我們調用myCache.put("e","123")或者myCache.putAll(map)時,寫緩存。
  • 首先,Cache會將寫操作立即委托給CacheLoaderWriter#write和#writeAll,然后由CacheLoaderWriter負責立即去寫SoR。
  • 當寫SoR成功后,再寫入Cache。

3. Write-Behind

Write-Behind,也叫Write-Back,稱之為回寫模式,不同于Write-Through是同步寫SoR和Cache,Write-Behind是異步寫。異步之后可以實現批量寫、合并寫、延時和限流。

(1) 異步寫

  1. CacheManager cacheManager = CacheManagerBuilder. newCacheManagerBuilder() 
  2.        .using(PooledExecutionServiceConfigurationBuilder 
  3.                .newPooledExecutionServiceConfigurationBuilder() 
  4.                .pool("writeBehindPool", 1, 5) 
  5.                .build()) 
  6.        .build(true); 
  7. org.ehcache.Cache<String, String> myCache =cacheManager. createCache ("myCache", 
  8.        CacheConfigurationBuilder.newCacheConfigurationBuilder(String.class,String.class, 
  9.                ResourcePoolsBuilder.newResourcePoolsBuilder().heap(100,MemoryUnit.MB)) 
  10.                .withDispatcherConcurrency(4) 
  11.                .withExpiry(Expirations.timeToLiveExpiration(Duration.of(10,TimeUnit.SECONDS))) 
  12.                 .withLoaderWriter(new DefaultCacheLoaderWriter<String,String >() { 
  13.                    @Override 
  14.                    public void write(String key, String value) throws Exception{ 
  15.                         //write 
  16.                     } 
  17.   
  18.                    @Override 
  19.                    public void delete(String key) throws Exception { 
  20.                         //delete 
  21.                     } 
  22.                }) 
  23.                .add(WriteBehindConfigurationBuilder 
  24.                         .newUnBatchedWriteBehindConfiguration() 
  25.                         .queueSize(5) 
  26.                         .concurrencyLevel(2 
  27.                         .useThreadPool("writeBehindPool") 
  28.                         .build())); 

幾個重要配置如下。

  • hreadPool:使用PooledExecutionServiceConfigurationBuilder配置線程池;然后WriteBehindConfigurationBuilder通過useThreadPool配置使用哪一個線程池;
  • WriteBehindConfigurationBuilder:配置WriteBehind策略;
  • CacheLoaderWriter:配置WriteBehind如何操作SoR。

WriteBehindConfigurationBuilder會進行如下幾個配置。

  • newUnBatchedWriteBehindConfiguration:表示不進行批量處理,那么所有批量操作都將會轉換成單個操作,即CacheLoaderWriter只需要實現write和delete即可。
  • queueSize(int size):因為操作是異步回寫SoR,需要將操作先放入寫操作等待隊列,因此,使用queue size定義寫操作等待隊列***大小,即線程池隊列大小。內部使用NonBatchingLocalHeapWriteBehindQueue。
  • concurrencyLevel(int concurrency):配置使用多少個并發線程和隊列進行WriteBehind。因為我們只傳入一個線程池,這是如何實現該模式的呢?首先看如下代碼片段。
    1. for (int i = 0; i < writeBehindConcurrency; i++) { 
    2.   if (config.getBatchingConfiguration()== null) { 
    3.     this.stripes.add(newNonBatchingLocalHeapWriteBehindQueue<K, V>(executionService,defaultThreadPool, config, cacheLoaderWriter)); 
    4.   } else { 
    5.     this.stripes.add(newBatchingLocalHeapWriteBehindQueue<K, V>(executionService, defaultThreadPool,config, cacheLoaderWriter)); 
    6.   } 

可以看到會創建concurrencyLevel個隊列NonBatchingLocalHeapWriteBehindQueue,其又通過如下代碼片段創建線程池和線程池隊列。

  1. this.executorQueue = new LinkedBlockingQueue<Runnable>(config.getMaxQueueSize()); 
  2. if (config.getThreadPoolAlias() == null) { 
  3.   this.executorexecutionService.getOrderedExecutor(defaultThreadPool, executorQueue); 
  4. } else { 
  5.   this.executorexecutionService.getOrderedExecutor(config. getThreadPoolAlias(), executorQueue); 
  6. ●     CacheLoaderWriter:此處我們只配置了write和delete,而writeAll和deleteAll將會把批量操作委托給write和delete。 
  7. PooledExecutionService#getOrderedExecutor方法會創建PartitionedOrderedExecutor實例。 
  8. PartitionedOrderedExecutor(BlockingQueue<Runnable> queue,ExecutorService executor) { 
  9.   this.delegatenew PartitionedUnorderedExecutor(queue, executor, 1); 

其使用maxWorkers=1創建了PartitionedUnorderedExecutor,然后Partitioned UnorderedExecutor通過this.runnerPermit = newSemaphore(maxWorkers)來控制并發,即maxWorkers=1就實現了一個并發。

因此,Ehcache實際能寫的***隊列大小為concurrency level *queue size。

因為內部使用線程池去寫,因此就實現了異步寫,又因為使用了隊列,因此控制了總的吞吐量(此處有注意根據實際場景給線程池配置Rejected Policy),接下來看下如何實現批量寫。

(2) 批量寫

  1. .withLoaderWriter(new DefaultCacheLoaderWriter<String,String>() { 
  2.    @Override 
  3.     publicvoid writeAll(Iterable<? extends Map.Entry<? extends String,? extends String>> entries) throws BulkCacheWritingException,Exception { 
  4.         for(Objectentry : entries) { 
  5.             //batchwrite 
  6.         } 
  7.     } 
  8.    @Override 
  9.     publicvoid deleteAll(Iterable<? extends String> keys) throws BulkCacheWritingException,Exception { 
  10.         for(Objectkey : keys) { 
  11.             //batchdelete 
  12.         } 
  13.     } 
  14. }) 
  15. .add(WriteBehindConfigurationBuilder 
  16.         .newBatchedWriteBehindConfiguration(3,TimeUnit.SECONDS, 2) 
  17.        .queueSize(5) 
  18.        .concurrencyLevel(1) 
  19.        .enableCoalescing() 
  20.        .useThreadPool("writeBehindPool") 
  21.        .build())); 

和上一個示例不同的地方是使用了newBatchedWriteBehindConfiguration進行批量配置。

● newBatchedWriteBehindConfiguration(longmaxDelay, TimeUnit maxDelayUnit, int batchSize):設置批處理大小和***延遲。batchSize用于定義批處理大小,當寫操作數量等于批處理大小時,將把這一批數據發給CacheLoaderWriter進行處理。Ehcache使用BatchingLocalHeapWriteBehindQueue實現批量隊列,其中操作批量的代碼如下。

  1. if (openBatch.add(operation)) {//往batch里添加操作,添加的數量等于批處理大小時 
  2.   submit(openBatch);//異步提交批處理操作 
  3.   openBatchnull

因此,Ehcache實際能寫的***隊列大小為concurrency level * queue size * batch size。

maxDelay用于配置未完成的批處理***延遲,比如,我們設置批處理大小為3,而我們實際只寫入了兩個數據,當寫第3個數據時,會觸發提交批處理操作。但是,如果我們不寫第3個,那么將造成這2個數據一直等待,我們可以設置maxDelay,當超時時也會將這兩個數據提交批處理。

● enableCoalescing:是否需要合并寫,即對于相同的Key只記錄***一次數據。

● CacheLoaderWriter:write和delete會轉換為writeAll和deleteAll,即批處理。

三、Copy Pattern

有兩種Copy Pattern,Copy-On-Read(在讀時復制)和Copy-On-Write(在寫時復制),對于Guava Cache和Ehcache中堆緩存都是基于引用的,這樣如果有人拿到緩存數據并修改了它,則可能發生不可預測的問題,筆者就見過因為這種情況造成數據錯誤。Guava Cache沒有提供支持,Ehcache 3.x提供了支持。

  1. public interface Copier<T> { 
  2.   TcopyForRead(T obj); //Copy-On-Read,比如myCache.get() 
  3.   TcopyForWrite(T obj); //Copy-On-Write,比如myCache.put() 

通過如下方法來配置Key和Value的Copier。

  1. CacheConfigurationBuilder.withKeyCopier() 
  2. CacheConfigurationBuilder.withValueCopier() 

【本文是51CTO專欄作者張開濤的原創文章,作者微信公眾號:開濤的博客( kaitao-1234567)】

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

責任編輯:趙寧寧 來源: 51CTO專欄
相關推薦

2017-05-01 17:03:01

Java緩存分布式

2017-04-21 08:51:42

API緩存分布式

2017-05-10 11:40:29

緩存Nginx HTTP

2017-04-18 14:49:38

應用層API代碼

2017-05-18 16:07:23

回滾數據庫代碼

2019-11-05 14:24:31

緩存雪崩框架

2013-03-28 13:08:15

Web緩存

2009-06-18 15:23:49

緩存控制器模式Ajax模式

2017-06-16 15:16:15

2017-07-02 16:50:21

2017-06-04 16:24:27

線程線程池中斷

2010-10-19 08:59:40

PHP緩存技術

2022-06-13 10:23:34

Helios緩存服務端

2018-12-10 15:13:06

緩存系統性能數據

2025-09-01 08:28:41

2012-12-13 17:38:48

2012年度IT博客大IT博客大賽博客

2009-09-23 09:37:07

Hibernate緩存

2009-08-13 17:50:49

Hibernate 3

2009-08-03 18:47:12

ASP.NET數據緩存

2016-08-23 10:50:50

WebJavascript緩存
點贊
收藏

51CTO技術棧公眾號

99麻豆久久久国产精品免费 | 欧美视频在线观看一区| 欧洲精品亚洲精品| 一区二区日韩在线观看| 欧美精品videosex| 成人精品视频.| 色777狠狠综合秋免鲁丝| 日韩精品视频网址| 色网在线免费观看| 亚洲欧洲另类国产综合| 国产精品av在线播放| 国产一二三四区| 欧美变态网站| 欧美一级片在线观看| 国产综合免费视频| 羞羞的视频在线观看| 26uuu国产日韩综合| 欧美日韩国产成人| 人人妻人人藻人人爽欧美一区| 日本大片在线播放| 国产日韩视频一区二区三区| 国产69精品久久久久久| 国产又粗又长又黄的视频| 噜噜噜狠狠夜夜躁精品仙踪林| 欧美日韩精品一区二区| 免费无码不卡视频在线观看| 麻豆传媒在线观看| 91蜜桃婷婷狠狠久久综合9色| 91久久精品国产91久久| 国产情侣免费视频| 午夜亚洲福利在线老司机| 欧美刺激性大交免费视频| 久久久久无码精品国产sm果冻 | 日本а中文在线天堂| 亚洲欧洲美洲综合色网| 视频一区三区| 日韩a在线观看| 粉嫩蜜臀av国产精品网站| 成人春色激情网| 国产成人麻豆免费观看| 国产精品人人爽人人做我的可爱| 欧美成年人网站| 国产日产精品一区二区三区的介绍| 国产精品欧美在线观看| 亚洲精品国产综合久久| av免费观看不卡| 8x国产一区二区三区精品推荐| 欧美美女bb生活片| 亚洲精品成人在线播放| 日本免费成人| 欧美日韩亚洲不卡| 在线观看成人av| av中文字幕免费| 久久国产欧美日韩精品| 欧美日本在线视频中文字字幕| 永久免费毛片在线观看| 欧美日韩中文一区二区| 一区二区日韩精品| 亚洲v国产v欧美v久久久久久| 亚洲v天堂v手机在线| 亚洲福利视频网站| 中文字幕 亚洲一区| 日本欧美不卡| 色狠狠色狠狠综合| 杨幂毛片午夜性生毛片| 久久久久黄色| 日韩欧美在线观看一区二区三区| 在线一区二区不卡| 亚洲午夜免费| 亚洲国产一区二区三区在线观看| 久久久久亚洲AV成人无码国产| 精品日本视频| 欧美日韩国产一级二级| 狠狠干狠狠操视频| 色妞ww精品视频7777| 亚洲国产欧美在线成人app| 男人的天堂影院| 国产成人精品免费视| 尤物精品国产第一福利三区| 91香蕉国产视频| 你懂的国产精品| 97国产真实伦对白精彩视频8| 久久夜色精品国产噜噜亚洲av| 日韩精品1区2区3区| 国产精品一区二区久久国产| 国产三区在线播放| a级精品国产片在线观看| 日韩成人av电影在线| 欧美视频久久久| 99精品偷自拍| 亚洲乱码一区二区三区| 女囚岛在线观看| 91国产免费观看| 香蕉在线观看视频| 校花撩起jk露出白色内裤国产精品| 亚洲香蕉在线观看| 在线免费日韩av| 丝袜诱惑亚洲看片| 亚洲一区亚洲二区亚洲三区| 深夜福利在线视频| 亚洲欧洲另类国产综合| 北条麻妃在线视频观看| 欧美视频第一| 日韩成人在线免费观看| 欧美美女性生活视频| 亚洲精品乱码| 成人免费在线视频网站| 婷婷婷国产在线视频| 亚洲欧美电影院| 国产男女在线观看| 美国十次综合久久| 亚洲日本aⅴ片在线观看香蕉| 99视频只有精品| 日本中文字幕一区二区有限公司| 成人黄色在线免费观看| 大地资源中文在线观看免费版| 亚洲综合色自拍一区| 中文字幕永久有效| 久久99国产精品视频| 日韩久久午夜影院| 在线观看美女av| 日韩二区在线观看| 久久久久网址| 爱福利在线视频| 欧美日韩视频免费播放| 久久综合桃花网| 成人中文视频| 国产成人精品一区二区| 六月丁香色婷婷| 亚洲品质自拍视频| 中文字幕 91| 国产一区二区三区四区| 欧洲中文字幕国产精品| 亚洲中文字幕无码爆乳av| 成人一区二区三区| 日日噜噜夜夜狠狠久久丁香五月 | 91亚洲精品久久久久久久久久久久 | 久久久国产亚洲精品| 欧洲美女免费图片一区| 丰满少妇被猛烈进入| 亚洲视频狠狠干| 亚洲a级黄色片| 久久中文亚洲字幕| 国产精品九九九| 国产鲁鲁视频在线观看免费| 日韩欧美高清在线视频| 一起草在线视频| 国产欧美日韩综合一区在线播放 | 国产性生活一级片| 欧美丰满日韩| 成人精品久久av网站| 一级毛片视频在线观看| 欧美日韩精品欧美日韩精品| 日韩av网站在线播放| 久久99精品久久久久久动态图| 色之综合天天综合色天天棕色 | 国产日韩综合| 久久精品五月婷婷| 欧美日韩精品免费观看视欧美高清免费大片| 日韩精品小视频| 91黑人精品一区二区三区| 国产丝袜在线精品| 国产精品一区二区小说| 久久久久国产精品| 99精品国产高清在线观看| 人交獸av完整版在线观看| 精品成人佐山爱一区二区| 日本一区二区三区四区五区| 99r国产精品| 91av俱乐部| 五月开心六月丁香综合色啪| 99高清视频有精品视频| 国产偷倩在线播放| 亚洲精品成人av| 午夜精品久久久久久久蜜桃| 国产精品黄色在线观看| gogo亚洲国模私拍人体| 在线亚洲自拍| 亚洲成人一区二区三区| 韩国精品视频在线观看 | 欧洲精品久久| 24小时成人在线视频| 韩国三级电影久久久久久| 国产无套粉嫩白浆在线2022年| 欧美日韩一卡二卡| 久久久国产精华液| 久久久不卡网国产精品一区| 欧美国产日韩另类 | 国内精品小视频在线观看| 欧美欧美欧美| 91精品久久久久久久久99蜜臂| 国产一线在线观看| 久久午夜av| 欧美大片免费播放| 午夜精品影视国产一区在线麻豆| 国产免费一区视频观看免费| sqte在线播放| 日韩一区二区在线观看视频| 西西44rtwww国产精品| 成人在线一区二区三区| 日本999视频| 黑丝一区二区三区| 亚洲午夜高清视频| 日本久久成人网| 91久久久久久久久| 电影一区二区三| 欧美国产精品日韩| a√在线中文网新版址在线| 精品国产髙清在线看国产毛片| jizz国产在线观看| 亚洲国产日韩综合久久精品| 台湾佬美性中文| 日本欧美一区二区三区| 欧美国产亚洲一区| 国产中文一区| 中文字幕日韩一区二区三区不卡 | 草莓视频成人appios| 欧美精品久久久久久久免费观看| 在线a人片免费观看视频| 亚洲精品国精品久久99热一| 精品久久久久久亚洲综合网站| 91电影在线观看| 日韩在线视频免费播放| 亚洲成人激情自拍| 日韩视频中文字幕在线观看| 国产欧美中文在线| 久久精品国产亚洲av麻豆| 成人一区在线看| 国产不卡的av| 狠狠色丁香婷综合久久| 中文字幕日韩一区二区三区不卡| 九九精品久久| 久久免费一区| 亚洲传媒在线| 久久国产精品高清| 嫩草国产精品入口| 99精品99久久久久久宅男| 国产午夜精品一区在线观看| 国产欧美日韩精品在线观看| 成人黄色在线| 国产精品午夜视频| a屁视频一区二区三区四区| 国产成人精品电影| 久久91导航| 国产精品v片在线观看不卡| 三上悠亚激情av一区二区三区| 欧美一级电影在线| 在线能看的av网址| 国产91露脸中文字幕在线| 亚洲承认视频| 国产精品久久久久久久久久新婚| 电影天堂国产精品| 国产精品日韩在线一区| 欧美美女被草| 91亚洲午夜在线| 日本精品国产| 国产精品国产三级欧美二区| 国产精品极品在线观看| 久久精品日产第一区二区三区精品版 | 欧美日韩高清不卡| 国产男男gay体育生白袜| 欧美一级二级三级乱码| 精品人妻一区二区三区三区四区| 日韩美女在线视频| 天天干,天天操,天天射| 亚洲欧美中文字幕| 午夜精品一区| 色综合久久88色综合天天看泰| av不卡高清| 国产成人免费av| 亚洲成人高清| 国产99在线播放| 亚洲免费成人av在线| 午夜精品美女久久久久av福利| 久久影院100000精品| 国产在线拍揄自揄拍无码| 亚洲高清激情| 一区二区成人网| 国产成人aaa| 国产交换配乱淫视频免费| 国产精品美女久久久久aⅴ国产馆| 青花影视在线观看免费高清| 亚洲国产裸拍裸体视频在线观看乱了| 国产精品美女久久久久av爽| 欧美亚一区二区| 亚洲欧美激情在线观看| 国产一区二区三区久久精品| 最近中文字幕免费mv2018在线| 91精品国产91久久久久久最新 | 欧美精品乱码久久久久久 | 成人深夜在线观看| 少妇av片在线观看| 亚洲一二三四区| 免费看av在线| 亚洲а∨天堂久久精品9966| 最新97超碰在线| 91精品国产色综合久久不卡98口 | 欧美男男激情videos| 国产精品网站入口| 国产精品网站在线看| 性欧美videosex高清少妇| 尹人成人综合网| 999在线观看| 久久久久久久综合狠狠综合| 18岁成人毛片| 欧美男女性生活在线直播观看| 无码国产精品一区二区免费16| 日韩亚洲在线观看| 在线精品亚洲欧美日韩国产| 超碰97网站| 97人人精品| 九色porny91| a亚洲天堂av| 丁香花五月激情| 欧美日本在线看| 激情小说 在线视频| 久久久久久久一| 国产高清亚洲| 一区二区三区在线视频111| 美女91精品| 一本加勒比波多野结衣| 一区二区在线观看视频| 在线免费看av的网站| 亚洲视频在线观看| 欧美激情护士| 精品人伦一区二区三区 | 亚洲成人a**址| 久久精品在线| 男男做爰猛烈叫床爽爽小说| 亚洲激情在线播放| 国产精品毛片一区视频播 | 亚洲国产精品久久人人爱| 在线观看中文字幕码| 揄拍成人国产精品视频| 中文字幕在线免费观看视频| 精品无码久久久久国产| 在线精品一区| 亚洲日本久久久| 夜夜嗨av一区二区三区中文字幕 | 中文字幕视频一区二区在线有码| 成人性生交大片免费看午夜| 日韩av快播网址| 日韩欧美影院| 中文字幕无码精品亚洲35| 99久久免费精品| 久草国产精品视频| 日韩国产欧美精品一区二区三区| 白浆在线视频| 久久综合给合久久狠狠色| 亚洲一区欧美二区| 国产精品扒开腿做爽爽| 91久久免费观看| a天堂中文在线| 成人国产精品免费视频| 影音先锋日韩在线| 成年人看片网站| 天天免费综合色| 日本国产在线| 国产精品久久久久秋霞鲁丝| 9999国产精品| 9191在线视频| 五月天一区二区三区| 亚洲欧洲成人在线| 国产不卡精品视男人的天堂| 久久一区91| 白丝校花扒腿让我c| 精品久久久久久中文字幕一区奶水 | 国产一区二区三区18 | 国产午夜精品一区| 成人激情开心网| 玖玖爱视频在线| 一区二区三区.www| 天天爽夜夜爽夜夜爽| 日本久久久久久久久久久| 91亚洲国产| 欧美做受高潮中文字幕| 欧美性xxxx极品hd满灌| av影片免费在线观看| 亚洲最大的成人网| 国产欧美自拍一区| 国内外成人免费激情视频| 国产精品素人视频| 性一交一乱一色一视频麻豆| 亚洲最新av在线| 韩国一区二区三区视频| 国产 日韩 欧美在线| 久久精品人人做人人爽人人| 国产精品自产拍| 91chinesevideo永久地址| 日韩一区欧美| 日本精品一二三区| 欧美羞羞免费网站| 激情av在线| 亚洲永久一区二区三区在线| 成人免费看的视频| 影音先锋黄色网址| 97在线看福利| 亚洲成人一区| 91狠狠综合久久久久久| 亚洲高清久久网|