国产无遮挡裸体免费直播视频,久久精品国产蜜臀av,动漫在线视频一区二区,欧亚日韩一区二区三区,久艹在线 免费视频,国产精品美女网站免费,正在播放 97超级视频在线观看,斗破苍穹年番在线观看免费,51最新乱码中文字幕

JetCache?緩存框架的使用及源碼解析(推薦)

 更新時間:2022年01月12日 10:02:24   作者:月圓吖  
JetCache是一個基于Java的緩存系統(tǒng)封裝,提供統(tǒng)一的API和注解來簡化緩存的使用。本文重點給大家介紹JetCache?緩存框架的使用及源碼分析,感興趣的朋友一起看看吧

一、簡介

JetCache是一個基于Java的緩存系統(tǒng)封裝,提供統(tǒng)一的API和注解來簡化緩存的使用。 JetCache提供了比SpringCache更加強(qiáng)大的注解,可以原生的支持TTL、兩級緩存、分布式自動刷新,還提供了Cache接口用于手工緩存操作。 當(dāng)前有四個實現(xiàn):RedisCacheRedisLettuceCacheCaffeineCache、LinkedHashMapCache

特性:

  • 通過統(tǒng)一的API訪問Cache系統(tǒng)
  • 通過注解實現(xiàn)聲明式的方法緩存,支持TTL和兩級緩存
  • 通過注解創(chuàng)建并配置Cache實例
  • 針對所有Cache實例和方法緩存的自動統(tǒng)計
  • Key的生成策略和Value的序列化策略支持自定義配置
  • 分布式緩存自動刷新,分布式鎖
  • 異步Cache API (使用Redis的Lettuce客戶端時)

緩存類型:

  • 本地

LinkedHashMap:使用LinkedHashMap做LUR方式淘汰
Caffeine:基于Java8開發(fā)的提供了近乎最佳命中率的高性能的緩存庫

  • 遠(yuǎn)程(訪問Redis的客戶端)

Redis:使用Jedis客戶端,Redis官方首選的Java客戶端
RedisSpringData:使用SpringData訪問Redis(官網(wǎng)未作介紹)
RedisLettuce:使用Lettuce客戶端,一個高性能基于Java的Redis驅(qū)動框架,支持線程安全的同步、異步操作,底層集成了Project Reactor,提供反應(yīng)式編程,參考:關(guān)于SpringBoot整合redis使用Lettuce客戶端超時問題

為什么使用緩存?

在高并發(fā)、大流量等場景下,降低系統(tǒng)延遲,緩解數(shù)據(jù)庫壓力,提高系統(tǒng)整體的性能,讓用戶有更好的體驗。

使用場景

讀多寫少、不追求強(qiáng)一致性、請求入?yún)⒉灰鬃兓?/p>

使用規(guī)范

選擇了遠(yuǎn)程緩存請設(shè)置keyPrefix,保證存放至Redis的緩存key規(guī)范化,避免與其他系統(tǒng)出現(xiàn)沖突,例如這樣設(shè)計:系統(tǒng)簡稱:所屬名字:,這樣存儲到Redis的緩存key為:系統(tǒng)簡稱:所屬名字:緩存key

選擇了本地緩存請設(shè)置limit,全局默認(rèn)設(shè)置了100,本地緩存的數(shù)據(jù)存放于內(nèi)存,減輕內(nèi)存的損耗,如果使用了Caffeine,緩存的key過多可能導(dǎo)致內(nèi)存溢出

請勿濫用緩存注解,對于非必要添加緩存的方法我們盡量不使用緩存

二、如何使用

說明:以下使用方式是基于SpringBoot引入JetCache緩存框架的,如果不是SpringBoot工程,請參考JetCache官網(wǎng)使用

引入maven依賴

<dependencies>
    <!-- 使用 jedis 客戶端添加以下依賴 -->
    <dependency>
        <groupId>com.alicp.jetcache</groupId>
        <artifactId>jetcache-starter-redis</artifactId>
        <version>${version}</version>
    </dependency>
    <!-- 使用 lettuce 客戶端添加以下依賴 -->
    <dependency>
        <groupId>com.alicp.jetcache</groupId>
        <artifactId>jetcache-starter-redis-lettuce</artifactId>
        <version>${version}</version>
    </dependency>
</dependencies>

添加配置

jetcache:
  statIntervalMinutes: 60
  areaInCacheName: false
  penetrationProtect: false
  enableMethodCache: true
  hiddenPackages: com.xxx.xxx,com.xxx.xxx
  local:
    default:
      type: caffeine # 支持的類型:linkedhashmap、caffeine
      limit: 100
      keyConvertor: fastjson # 支持的類型:fastjson,可自定義轉(zhuǎn)換器函數(shù)
      expireAfterWriteInMillis: 600000
      expireAfterAccessInMillis: 300000 
  remote:
    default:
      type: redis.lettuce # 支持的類型:redis、redis.lettuce
      keyPrefix: '系統(tǒng)簡稱:所屬名字:'
      keyConvertor: fastjson
      valueEncoder: java # 支持的類型:kryo、java,可自定義編碼器
      valueDecoder: java # 支持的類型:kryo、java,可自定義解碼器
      expireAfterWriteInMillis: 3600000
      #readFrom: slavePreferred # 優(yōu)先從Slave節(jié)點中讀取
      uri: redis-sentinel://host1:26379,host2:26379,host3:26379/?sentinelMasterId=mymaster # 哨兵模式
      #uri: redis://127.0.0.1:6379/ # 單節(jié)點模式
      #mode: masterslave # 設(shè)置為主從模式
      #uri: # 集群模式
      #- redis://127.0.0.1:7000
      #- redis://127.0.0.1:7001
      #- redis://127.0.0.1:7002
    example:
      keyPrefix: '系統(tǒng)簡稱:所屬名字:'
      type: redis
      keyConvertor: fastjson
      valueEncoder: java
      valueDecoder: java
      expireAfterWriteInMillis: 3600000
      poolConfig:
        minIdle: 10
        maxIdle: 20
        maxTotal: 50
      #password: xxx # 連接密碼
      #timeout: 2000 # 連接的超時時間,讀取數(shù)據(jù)的超時時間
      #database: 0 # 連接的數(shù)據(jù)庫
      #clientName: null # 客戶端名稱
      #ssl: 是否使用SSL
      host: ${redis.host}
      port: ${redis.port}
      #sentinel: host1:26379,host2:26379,host3:26379 # 哨兵模式
      #masterName: mymaster

配置說明

jetcache的全局配置

屬性默認(rèn)值說明
jetcache.statIntervalMinutes0用于統(tǒng)計緩存調(diào)用相關(guān)信息的統(tǒng)計間隔(分鐘),0表示不統(tǒng)計。
jetcache.areaInCacheNametrue緩存實例名稱cacheName會作為緩存key的前綴,2.4.3以前的版本總是把a(bǔ)reaName加在cacheName中,因此areaName也出現(xiàn)在key前綴中。我們一般設(shè)置為false。
jetcache.penetrationProtectfalse當(dāng)緩存訪問未命中的情況下,對并發(fā)進(jìn)行的加載行為進(jìn)行保護(hù)。 當(dāng)前版本實現(xiàn)的是單JVM內(nèi)的保護(hù),即同一個JVM中同一個key只有一個線程去加載,其它線程等待結(jié)果。這是全局配置,如果緩存實例沒有指定則使用全局配置。
jetcache.enableMethodCachetrue是否使用jetcache緩存。
jetcache.hiddenPackages自動生成緩存實例名稱時,為了不讓名稱太長,hiddenPackages指定的包名前綴會被截掉,多個包名使用逗號分隔。我們一般會指定每個緩存實例的名稱。

本地緩存的全局配置

屬性默認(rèn)值說明
jetcache.local.${area}.type本地緩存類型,支持 linkedhashmap、caffeine。
jetcache.local.${area}.limit100每個緩存實例存儲的緩存數(shù)量的全局配置,僅本地緩存需要配置,如果緩存實例沒有指定則使用全局配置,請結(jié)合實例的業(yè)務(wù)場景進(jìn)行配置該參數(shù)。
jetcache.local.${area}.keyConvertor緩存key轉(zhuǎn)換器的全局配置,支持的類型:fastjson。僅當(dāng)使用@CreateCache且緩存類型為LOCAL時可以指定為none,此時通過equals方法來識別key。方法緩存必須指定keyConvertor。支持自定義轉(zhuǎn)換器函數(shù),可設(shè)置為:bean:beanName,然后會從spring容器中獲取該bean。
jetcache.local.${area}.expireAfterWriteInMillis無窮大本地緩存超時時間的全局配置(毫秒)。
jetcache.local.${area}.expireAfterAccessInMillis0多長時間沒訪問就讓緩存失效的全局配置(毫秒),僅支持本地緩存。0表示不使用這個功能。

遠(yuǎn)程緩存的全局配置

屬性默認(rèn)值說明
jetcache.remote.${area}.type連接Redis的客戶端類型,支持 redis、redis.lettuce、redis.springdata。
jetcache.remote.${area}.keyPrefix保存至遠(yuǎn)程緩存key的前綴,請規(guī)范使用。
jetcache.remote.${area}.keyConvertor參考上述說明。
jetcache.remote.${area}.valueEncoderjava保存至遠(yuǎn)程緩存value的編碼函數(shù),支持:java、kryo。支持自定義編碼函數(shù),可設(shè)置為:bean:beanName,然后會從spring容器中獲取該bean。
jetcache.remote.${area}.valueDecoderjava保存至遠(yuǎn)程緩存value的解碼函數(shù),支持:java、kryo。支持自定義解碼函數(shù),可設(shè)置為:bean:beanName,然后會從spring容器中獲取該bean。
jetcache.remote.${area}.expireAfterWriteInMillis無窮大遠(yuǎn)程緩存超時時間的全局配置(毫秒)。
jetcache.remote.${area}.uriredis節(jié)點信息。

上表中${area}對應(yīng)@Cached和@CreateCache的area屬性,如果注解上沒有指定area,默認(rèn)值是"default"。

關(guān)于緩存的超時時間:

  1. put等方法上指定了超時時間,則以此時間為準(zhǔn);
  2. put等方法上未指定超時時間,使用Cache實例的默認(rèn)超時時間;
  3. Cache實例的默認(rèn)超時時間,通過在@CreateCache和@Cached上的expire屬性指定,如果沒有指定,使用yml中定義的全局配置,例如@Cached(cacheType=local)使用jetcache.local.default.expireAfterWriteInMillis,如果仍未指定則是無窮大。

注解說明

如果需要使用jetcache緩存,啟動類添加兩個注解:@EnableCreateCacheAnnotation@EnableMethodCache

@EnableCreateCacheAnnotation

開啟可通過@CreateCache注解創(chuàng)建Cache實例功能。

@EnableMethodCache

開啟可通過@Cached注解創(chuàng)建Cache實例功能,初始化spring aop,注解說明:

屬性默認(rèn)值說明
basePackagesjetcache需要攔截的包名,只有這些包名下的Cache實例才會生效
orderOrdered.LOWEST_PRECEDENCE指定AOP切面執(zhí)行過程的順序,默認(rèn)最低優(yōu)先級
modeAdviceMode.PROXYSpring AOP的模式,目前就提供默認(rèn)值讓你修改
proxyTargetClassfalse

@Cached

為一個方法添加緩存,創(chuàng)建對應(yīng)的緩存實例,注解可以添加在接口或者類的方法上面,該類必須是spring bean,注解說明:

屬性默認(rèn)值說明
area"default"如果在配置中配置了多個緩存area,在這里指定使用哪個area。
name未定義指定緩存實例名稱,如果沒有指定,會根據(jù)類名+方法名自動生成。name會被用于遠(yuǎn)程緩存的key前綴。另外在統(tǒng)計中,一個簡短有意義的名字會提高可讀性。
enabledtrue是否激活緩存。
timeUnitTimeUnit.SECONDS指定expire的單位。
expire未定義超時時間。如果注解上沒有定義,會使用全局配置,如果此時全局配置也沒有定義,則為無窮大。
localExpire未定義僅當(dāng)cacheType為BOTH時適用,為本地緩存指定一個不一樣的超時時間,通常應(yīng)該小于expire。如果沒有設(shè)置localExpire且cacheType為BOTH,那么本地緩存的超時時間和遠(yuǎn)程緩存保持一致。
cacheTypeCacheType.REMOTE緩存的類型,支持:REMOTE、LOCAL、BOTH,如果定義為BOTH,會使用LOCAL和REMOTE組合成兩級緩存。
localLimit未定義如果cacheType為LOCAL或BOTH,這個參數(shù)指定本地緩存的最大元素數(shù)量,以控制內(nèi)存占用。如果注解上沒有定義,會使用全局配置,如果此時你沒有定義全局配置,則使用默認(rèn)的全局配置100。請結(jié)合實際業(yè)務(wù)場景進(jìn)行設(shè)置該值。
serialPolicy未定義指定遠(yuǎn)程緩存VALUE的序列化方式,支持SerialPolicy.JAVA、SerialPolicy.KRYO。如果注解上沒有定義,會使用全局配置,如果你沒有定義全局配置,則使用默認(rèn)的全局配置SerialPolicy.JAVA。
keyConvertor未定義指定KEY的轉(zhuǎn)換方式,用于將復(fù)雜的KEY類型轉(zhuǎn)換為緩存實現(xiàn)可以接受的類型,支持:KeyConvertor.FASTJSON、KeyConvertor.NONE。NONE表示不轉(zhuǎn)換,F(xiàn)ASTJSON可以將復(fù)雜對象KEY轉(zhuǎn)換成String。如果注解上沒有定義,會使用全局配置。
key未定義使用SpEL指定緩存key,如果沒有指定會根據(jù)入?yún)⒆詣由伞?/td>
cacheNullValuefalse當(dāng)方法返回值為null的時候是否要緩存。
condition未定義使用SpEL指定條件,如果表達(dá)式返回true的時候才去緩存中查詢。
postCondition未定義使用SpEL指定條件,如果表達(dá)式返回true的時候才更新緩存,該評估在方法執(zhí)行后進(jìn)行,因此可以訪問到#result。

@CacheInvalidate

用于移除緩存,配置說明:

配置默認(rèn)值說明
area"default"如果在配置中配置了多個緩存area,在這里指定使用哪個area。
name指定緩存的唯一名稱,一般指向?qū)?yīng)的@Cached定義的name。
key未定義使用SpEL指定key,如果沒有指定會根據(jù)入?yún)⒆詣由伞?/td>
condition未定義使用SpEL指定條件,如果表達(dá)式返回true才執(zhí)行刪除,可訪問方法結(jié)果#result。刪除緩存實例中key的元素。
multifalse如果根據(jù)SpEL指定的key是一個集合,是否從緩存實例中刪除對應(yīng)的每個緩存。如果設(shè)置為true,但是key不是集合,則不會刪除緩存。

@CacheUpdate

用于更新緩存,配置說明:

配置默認(rèn)值說明
area"default"如果在配置中配置了多個緩存area,在這里指定使用哪個area。
name指定緩存的唯一名稱,一般指向?qū)?yīng)的@Cached定義的name。
key未定義使用SpEL指定key,如果沒有指定會根據(jù)入?yún)⒆詣由伞?/td>
value使用SpEL指定value。
condition未定義使用SpEL指定條件,如果表達(dá)式返回true才執(zhí)行更新,可訪問方法結(jié)果#result。更新緩存實例中key的元素。
multifalse如果根據(jù)SpEL指定key和value都是集合并且元素的個數(shù)相同,則是否更新緩存實例中的對應(yīng)的每個元素。如果設(shè)置為true,但是key不是集合或者value不是集合或者它們的元素的個數(shù)不相同,也不會更新緩存。

@CacheRefresh

用于自定刷新緩存,配置說明:

配置默認(rèn)值說明
refresh刷新間隔
stopRefreshAfterLastAccess未定義指定該key多長時間沒有訪問就停止刷新,如果不指定會一直刷新。
refreshLockTimeout60秒類型為BOTH/REMOTE的緩存刷新時,同時只會有一臺服務(wù)器在刷新,這臺服務(wù)器會在遠(yuǎn)程緩存放置一個分布式鎖,此配置指定該鎖的超時時間。
timeUnitTimeUnit.SECONDS指定refresh時間單位。

@CachePenetrationProtect

當(dāng)緩存訪問未命中的情況下,對并發(fā)進(jìn)行的加載行為進(jìn)行保護(hù)。 當(dāng)前版本實現(xiàn)的是單JVM內(nèi)的保護(hù),即同一個JVM中同一個key只有一個線程去加載,其它線程等待結(jié)果,配置說明:

配置默認(rèn)值說明
valuetrue是否開啟保護(hù)模式。
timeout未定義其他線程的等待超時時間,如果超時則自己執(zhí)行方法直接返回結(jié)果。
timeUnitTimeUnit.SECONDS指定timeout時間單位。

@CreateCache

在Spring Bean中使用該注解可創(chuàng)建一個Cache實例,配置說明:

配置默認(rèn)值說明
area"default"如果在配置中配置了多個緩存area,在這里指定使用哪個area。
name未定義指定緩存實例名稱,如果沒有指定,會根據(jù)類名+方法名自動生成。name會被用于遠(yuǎn)程緩存的key前綴。另外在統(tǒng)計中,一個簡短有意義的名字會提高可讀性。
timeUnitTimeUnit.SECONDS指定expire的單位。
expire未定義超時時間。如果注解上沒有定義,會使用全局配置,如果此時全局配置也沒有定義,則為無窮大。
localExpire未定義僅當(dāng)cacheType為BOTH時適用,為本地緩存指定一個不一樣的超時時間,通常應(yīng)該小于expire。如果沒有設(shè)置localExpire且cacheType為BOTH,那么本地緩存的超時時間和遠(yuǎn)程緩存保持一致。
cacheTypeCacheType.REMOTE緩存的類型,支持:REMOTELOCAL、BOTH,如果定義為BOTH,會使用LOCAL和REMOTE組合成兩級緩存。
localLimit未定義如果cacheType為LOCAL或BOTH,這個參數(shù)指定本地緩存的最大元素數(shù)量,以控制內(nèi)存占用。如果注解上沒有定義,會使用全局配置,如果此時你沒有定義全局配置,則使用默認(rèn)的全局配置100。請結(jié)合實際業(yè)務(wù)場景進(jìn)行設(shè)置該值。
serialPolicy未定義指定遠(yuǎn)程緩存VALUE的序列化方式,支持SerialPolicy.JAVA、SerialPolicy.KRYO。如果注解上沒有定義,會使用全局配置,如果你沒有定義全局配置,則使用默認(rèn)的全局配置SerialPolicy.JAVA。
keyConvertor未定義指定KEY的轉(zhuǎn)換方式,用于將復(fù)雜的KEY類型轉(zhuǎn)換為緩存實現(xiàn)可以接受的類型,支持:KeyConvertor.FASTJSONKeyConvertor.NONE。NONE表示不轉(zhuǎn)換,F(xiàn)ASTJSON可以將復(fù)雜對象KEY轉(zhuǎn)換成String。如果注解上沒有定義,會使用全局配置。

使用示例

/**
 * 啟動類
 */
@SpringBootApplication
@EnableCreateCacheAnnotation
@EnableMethodCache(basePackages = "com.xxx.xxx")
public class Application {
    public static void main(String[] args){
        SpringApplication.run(Application.class, args);
    }
}
/**
 * 接口
 */
public interface JetCacheExampleService {
    
    User getValue(long userId);
    
    void updateValue(User  user);
    
    void deleteValue(User  user);
}
/**
 * 實現(xiàn)類
 */
@Service
public class JetCacheExampleServiceImpl implements JetCacheExampleService {
    
    @CreateCache(name = "JetCacheExampleServiceImpl.exampleCache" , localLimit = 50 ,cacheType = CacheType.LOCAL)
    @CachePenetrationProtect
    private Cache<Long, User> exampleCache;
    
    @Override
    @Cached(name = "JetCacheExampleService.getValue", expire = 3600 * 6, localLimit = 50, cacheType = CacheType.BOTH)
    @CacheRefresh(refresh = 3600, stopRefreshAfterLastAccess = 3600 * 2)
    @CachePenetrationProtect
    public User getValue(long userId){
        String result = new User();
        // ... 處理邏輯
        return result;
    }
    
    @Override
    @CacheUpdate(name = "JetCacheExampleService.getValue", key="#user.userId", value="#user")
    public void updateValue(User user){
        // 處理邏輯
    }
    
    @Override
    @CacheInvalidate(name = "JetCacheExampleService.getValue", key="#user.userId")
    public void deleteValue(User user){
        // 處理邏輯
    }
    
}

如上述所示

getValue方法會創(chuàng)建一個緩存實例,通過@Cached注解可以看到緩存實例名稱cacheName為'JetCacheExampleService.getValue',緩存的有效時長為6小時,本地緩存的數(shù)量最多為50,緩存類型為BOTH(優(yōu)先從本地緩存獲?。煌ㄟ^@CacheRefresh注解可以看到會為該緩存實例設(shè)置一個刷新策略,刷新間隔為1小時,2個小時沒訪問后不再刷新,需要刷新的緩存實例會為其每一個緩存數(shù)據(jù)創(chuàng)建一個RefreshTask周期性任務(wù);@CachePenetrationProtect注解表示該緩存實例開啟保護(hù)模式,當(dāng)緩存未命中,同一個JVM中同一個key只有一個線程去加載數(shù)據(jù),其它線程等待結(jié)果。

updateValue方法可以更新緩存,通過@CacheUpdate注解可以看到會更新緩存實例'JetCacheExampleService.getValue'中緩存key為#user.userId的緩存value為#user。

deleteValue方法可以刪除緩存,通過@CacheInvalidate注解可以看到會刪除緩存實例'JetCacheExampleService.getValue'中緩存key為#user.userId緩存數(shù)據(jù)。

exampleCache字段會作為一個緩存實例對象,通過@CreateCache注解可以看到,會將該字段作為cacheName為'JetCacheExampleService.getValue'緩存實例對象,本地緩存的數(shù)量最多為50,緩存類型為LOCAL,@CachePenetrationProtect注解表示該緩存實例開啟保護(hù)模式。

我的業(yè)務(wù)場景是使用上述的getValue方法創(chuàng)建緩存實例即可。

注意:

  • @Cached注解不能和@CacheUpdate或者@CacheInvalidate同時使用
  • @CacheInvalidate可以多個同時使用

另外通過@CreateCache注解創(chuàng)建緩存實例也可以這樣初始化:

@Service
public class JetCacheExampleServiceImpl implements JetCacheExampleService {
    
	@CreateCache(name = "JetCacheExampleServiceImpl.exampleCache" , localLimit = 50 ,cacheType = CacheType.LOCAL)
	private Cache<Long, User> exampleCache;
	@PostConstruct
	public exampleCacheInit(){
    	RefreshPolicy policy = RefreshPolicy.newPolicy(60, TimeUnit.MINUTES)
                	.stopRefreshAfterLastAccess(120, TimeUnit.MINUTES);
        exampleCache.config().setLoader(this::loadFromDatabase);
        exampleCache.config().setRefreshPolicy(policy);
	}
}

更加詳細(xì)的使用方法請參考JetCache官方地址。

三、源碼解析

參考本人Git倉庫中的JetCache項目,已做詳細(xì)的注釋。

簡單概括:利用Spring AOP功能,在調(diào)用需要緩存的方法前,通過解析注解獲取緩存配置,根據(jù)這些配置創(chuàng)建不同的實例對象,進(jìn)行緩存等操作。

JetCache分為兩部分,一部分是Cache API以及實現(xiàn),另一部分是注解支持。

項目的各個子模塊

  • jetcache-anno-api:定義JetCache注解和常量。
  • jetcache-core:核心API,Cache接口的實現(xiàn),提供各種緩存實例的操作,不依賴于Spring。
  • jetcache-autoconfigure:完成初始化,解析application.yml配置文件中的相關(guān)配置,以提供不同緩存實例的CacheBuilder構(gòu)造器
  • jetcache-anno:基于Spring提供@Cached@CreateCache注解支持,初始化Spring AOP以及JetCache注解等配置。
  • jetcache-redis:使用Jedis提供Redis支持。
  • jetcache-redis-lettuce:使用Lettuce提供Redis支持,實現(xiàn)了JetCache異步訪問緩存的的接口。
  • jetcache-redis-springdata:使用Spring Data提供Redis支持。
  • jetcache-starter-redis:提供pom文件,Spring Boot方式的Starter,基于Jedis。
  • jetcache-starter-redis-lettuce:提供pom文件,Spring Boot方式的Starter,基于Lettuce。
  • jetcache-starter-redis-springdata:提供pom文件,Spring Boot方式的Starter,基于Spring Data。
  • jetcache-test:提供相關(guān)測試。

常用注解與變量

在jetcache-anno-api模塊中定義了需要用的緩存注解與常量,在上述已經(jīng)詳細(xì)的講述過,其中@CacheInvalidateContainer注解定義value為@CacheInvalidate數(shù)組,然后通過jdk8新增的@Repeatable注解,在@CacheInvalidate注解上面添加@Repeatable(CacheInvalidateContainer.class),即可支持同一個地方可以使用多個@CacheInvalidate注解。

緩存API

主要查看jetcache-core子模塊,提供各種Cache緩存,以支持不同的緩存類型

Cache接口的子關(guān)系,結(jié)構(gòu)如下圖:

主要對象描述:

  • Cache:緩存接口,定義基本方法
  • AbstractCache:抽象類,緩存接口的繼承者,提供基本實現(xiàn),具體實現(xiàn)交由不同的子類
  • LinkedHashMapCache:基于LinkedHashMap設(shè)計的簡易內(nèi)存緩存
  • CaffeineCache:基于Caffeine工具設(shè)計的內(nèi)存緩存
  • RedisCache:Redis實現(xiàn),使用Jedis客戶端
  • RedisLettuceCache:Redis實現(xiàn),使用Lettuce客戶端
  • MultiLevelCache:兩級緩存,用于封裝EmbeddedCache(本地緩存)和ExternalCache(遠(yuǎn)程緩存)
  • RefreshCache:基于裝飾器模式Decorator,提供自動刷新功能
  • LazyInitCache:用于@CreateCache注解創(chuàng)建的緩存實例,依賴于Spring

Cache接口

com.alicp.jetcache.Cache接口,定義了緩存實例的操作方法(部分有默認(rèn)實現(xiàn)),以及獲取分布式鎖(非嚴(yán)格,用于刷新遠(yuǎn)程緩存)的實現(xiàn),因為繼承了java.io.Closeable接口,所以也提供了close方法的默認(rèn)實現(xiàn),空方法,交由不同緩存實例的實現(xiàn)去實現(xiàn)該方法用于釋放資源,在com.alicp.jetcache.anno.support.ConfigProvider.doShutdown()方法中會調(diào)用每個緩存實例對象的close方法進(jìn)行資源釋放。主要代碼如下:

public interface Cache<K, V> extends Closeable {
    Logger logger = LoggerFactory.getLogger(Cache.class);
    //-----------------------------JSR 107 style API------------------------------------------------
    default V get(K key) throws CacheInvokeException {
        CacheGetResult<V> result = GET(key);
        if (result.isSuccess()) {
            return result.getValue();
        } else {
            return null;
        }
    }
    default Map<K, V> getAll(Set<? extends K> keys) throws CacheInvokeException {
        MultiGetResult<K, V> cacheGetResults = GET_ALL(keys);
        return cacheGetResults.unwrapValues();
    }
    default void put(K key, V value) {
        PUT(key, value);
    }
    default void putAll(Map<? extends K, ? extends V> map) {
        PUT_ALL(map);
    }
    default boolean putIfAbsent(K key, V value) { // 多級緩存MultiLevelCache不支持此方法
        CacheResult result = PUT_IF_ABSENT(key, value, config().getExpireAfterWriteInMillis(), TimeUnit.MILLISECONDS);
        return result.getResultCode() == CacheResultCode.SUCCESS;
    }
    default boolean remove(K key) {
        return REMOVE(key).isSuccess();
    }
    default void removeAll(Set<? extends K> keys) {
        REMOVE_ALL(keys);
    }
    <T> T unwrap(Class<T> clazz);
    @Override
    default void close() {
    }
    //--------------------------JetCache API---------------------------------------------
    CacheConfig<K, V> config();
    default AutoReleaseLock tryLock(K key, long expire, TimeUnit timeUnit) {
        if (key == null) {
            return null;
        }
        // 隨機(jī)生成一個值
        final String uuid = UUID.randomUUID().toString();
        // 過期時間
        final long expireTimestamp = System.currentTimeMillis() + timeUnit.toMillis(expire);
        final CacheConfig config = config();
        AutoReleaseLock lock = () -> { // 創(chuàng)建一把會自動釋放資源的鎖,實現(xiàn)其 close() 方法
            int unlockCount = 0;
            while (unlockCount++ < config.getTryLockUnlockCount()) {
                if(System.currentTimeMillis() < expireTimestamp) { // 這把鎖還沒有過期,則刪除
                    // 刪除對應(yīng)的 Key 值
                    // 出現(xiàn)的結(jié)果:成功,失敗,Key 不存在
                    CacheResult unlockResult = REMOVE(key);
                    if (unlockResult.getResultCode() == CacheResultCode.FAIL
                            || unlockResult.getResultCode() == CacheResultCode.PART_SUCCESS) {
                        // 刪除對應(yīng)的 Key 值過程中出現(xiàn)了異常,則重試
                        logger.info("[tryLock] [{} of {}] [{}] unlock failed. Key={}, msg = {}",
                                unlockCount, config.getTryLockUnlockCount(), uuid, key, unlockResult.getMessage());
                        // retry
                    } else if (unlockResult.isSuccess()) { // 釋放成功
                        logger.debug("[tryLock] [{} of {}] [{}] successfully release the lock. Key={}",
                                unlockCount, config.getTryLockUnlockCount(), uuid, key);
                        return;
                    } else { // 鎖已經(jīng)被釋放了
                        logger.warn("[tryLock] [{} of {}] [{}] unexpected unlock result: Key={}, result={}",
                                unlockCount, config.getTryLockUnlockCount(), uuid, key, unlockResult.getResultCode());
                        return;
                    }
                } else { // 該鎖已失效
                    logger.info("[tryLock] [{} of {}] [{}] lock already expired: Key={}",
                            unlockCount, config.getTryLockUnlockCount(), uuid, key);
                    return;
                }
            }
        };
        int lockCount = 0;
        Cache cache = this;
        while (lockCount++ < config.getTryLockLockCount()) {
            // 往 Redis(或者本地) 中存放 Key 值(_#RL#結(jié)尾的Key)
            // 返回的結(jié)果:成功、已存在、失敗
            CacheResult lockResult = cache.PUT_IF_ABSENT(key, uuid, expire, timeUnit);
            if (lockResult.isSuccess()) { // 成功獲取到鎖
                logger.debug("[tryLock] [{} of {}] [{}] successfully get a lock. Key={}",
                        lockCount, config.getTryLockLockCount(), uuid, key);
                return lock;
            } else if (lockResult.getResultCode() == CacheResultCode.FAIL || lockResult.getResultCode() == CacheResultCode.PART_SUCCESS) {
                logger.info("[tryLock] [{} of {}] [{}] cache access failed during get lock, will inquiry {} times. Key={}, msg={}",
                        lockCount, config.getTryLockLockCount(), uuid,
                        config.getTryLockInquiryCount(), key, lockResult.getMessage());
                // 嘗試獲取鎖的過程中失敗了,也就是往 Redis 中存放 Key 值出現(xiàn)異常
                // 這個時候可能 Key 值已經(jīng)存儲了,但是由于其他原因?qū)е路祷氐慕Y(jié)果表示執(zhí)行失敗
                int inquiryCount = 0;
                while (inquiryCount++ < config.getTryLockInquiryCount()) {
                    CacheGetResult inquiryResult = cache.GET(key);
                    if (inquiryResult.isSuccess()) {
                        if (uuid.equals(inquiryResult.getValue())) {
                            logger.debug("[tryLock] [{} of {}] [{}] successfully get a lock after inquiry. Key={}",
                                    inquiryCount, config.getTryLockInquiryCount(), uuid, key);
                            return lock;
                        } else {
                            logger.debug("[tryLock] [{} of {}] [{}] not the owner of the lock, return null. Key={}",
                                    inquiryCount, config.getTryLockInquiryCount(), uuid, key);
                            return null;
                        }
                    } else {
                        logger.info("[tryLock] [{} of {}] [{}] inquiry failed. Key={}, msg={}",
                                inquiryCount, config.getTryLockInquiryCount(), uuid, key, inquiryResult.getMessage());
                        // retry inquiry
                    }
                }
            } else { // 已存在表示該鎖被其他人占有
                // others holds the lock
                logger.debug("[tryLock] [{} of {}] [{}] others holds the lock, return null. Key={}",
                        lockCount, config.getTryLockLockCount(), uuid, key);
                return null;
            }
        }
        logger.debug("[tryLock] [{}] return null after {} attempts. Key={}", uuid, config.getTryLockLockCount(), key);
        return null;
    }
    default boolean tryLockAndRun(K key, long expire, TimeUnit timeUnit, Runnable action){
        // Release the lock use Java 7 try-with-resources.
        try (AutoReleaseLock lock = tryLock(key, expire, timeUnit)) { // 嘗試獲取鎖
            if (lock != null) { // 獲取到鎖則執(zhí)行下面的任務(wù)
                action.run();
                return true;
            } else {
                return false;
            }
            // 執(zhí)行完鎖的操作后會進(jìn)行資源釋放,調(diào)用 AutoCloseable 的 close() 方法
        }
    }
    CacheGetResult<V> GET(K key);
    MultiGetResult<K, V> GET_ALL(Set<? extends K> keys);
    default V computeIfAbsent(K key, Function<K, V> loader) {
        return computeIfAbsent(key, loader, config().isCacheNullValue());
    }
    V computeIfAbsent(K key, Function<K, V> loader, boolean cacheNullWhenLoaderReturnNull);
    V computeIfAbsent(K key, Function<K, V> loader, boolean cacheNullWhenLoaderReturnNull, long expireAfterWrite, TimeUnit timeUnit);
    default void put(K key, V value, long expireAfterWrite, TimeUnit timeUnit) {
        PUT(key, value, expireAfterWrite, timeUnit);
    }
    default CacheResult PUT(K key, V value) {
        if (key == null) {
            return CacheResult.FAIL_ILLEGAL_ARGUMENT;
        }
        return PUT(key, value, config().getExpireAfterWriteInMillis(), TimeUnit.MILLISECONDS);
    }
    CacheResult PUT(K key, V value, long expireAfterWrite, TimeUnit timeUnit);
    default void putAll(Map<? extends K, ? extends V> map, long expireAfterWrite, TimeUnit timeUnit) {
        PUT_ALL(map, expireAfterWrite, timeUnit);
    }
    default CacheResult PUT_ALL(Map<? extends K, ? extends V> map) {
        if (map == null) {
            return CacheResult.FAIL_ILLEGAL_ARGUMENT;
        }
        return PUT_ALL(map, config().getExpireAfterWriteInMillis(), TimeUnit.MILLISECONDS);
    }
    CacheResult PUT_ALL(Map<? extends K, ? extends V> map, long expireAfterWrite, TimeUnit timeUnit);
    CacheResult REMOVE(K key);
    CacheResult REMOVE_ALL(Set<? extends K> keys);
    CacheResult PUT_IF_ABSENT(K key, V value, long expireAfterWrite, TimeUnit timeUnit);
}

com.alicp.jetcache.Cache定義的方法大都是關(guān)于緩存的獲取、刪除和存放操作

其中大寫的方法返回JetCache自定義的CacheResult(完整的返回值,可以清晰的知道執(zhí)行結(jié)果,例如get返回null的時候,無法斷定是對應(yīng)的key不存在,還是訪問緩存發(fā)生了異常)

小寫的方法默認(rèn)實現(xiàn)就是調(diào)用大寫的方法

computeIfAbsent方法最為核心,交由子類去實現(xiàn)

tryLockAndRun方法會非堵塞的嘗試獲取一把AutoReleaseLock分布式鎖(非嚴(yán)格),獲取過程:

  • 嘗試往Redis中設(shè)置(已存在無法設(shè)置)一個鍵值對,key為緩存key_#RL#,value為UUID,并設(shè)置這個鍵值對的過期時間為60秒(默認(rèn))
  • 如果獲取到鎖后進(jìn)行加載任務(wù),也就是重新加載方法并更新遠(yuǎn)程緩存
  • 該鎖實現(xiàn)了java.lang.AutoCloseable接口,使用try-with-resource方式,在執(zhí)行完加載任務(wù)后會自動釋放資源,也就是調(diào)用close方法將獲取鎖過程中設(shè)置的鍵值對從Redis中刪除
  • 在RefreshCache中會調(diào)用該方法,因為如果存在遠(yuǎn)程緩存需要刷新則需要采用分布式鎖的方式

AbstractCache抽象類

com.alicp.jetcache.AbstractCache抽象類,實現(xiàn)了Cache接口,主要代碼如下:

public abstract class AbstractCache<K, V> implements Cache<K, V> {
    /**
     * 當(dāng)緩存未命中時,并發(fā)情況同一個Key是否只允許一個線程去加載,其他線程等待結(jié)果(可以設(shè)置timeout,超時則自己加載并直接返回)
     * 如果是的話則由獲取到Key對應(yīng)的 LoaderLock.signal(采用了 CountDownLatch)的線程進(jìn)行加載
     * loaderMap臨時保存 Key 對應(yīng)的 LoaderLock 對象
     */
    private volatile ConcurrentHashMap<Object, LoaderLock> loaderMap;
    ConcurrentHashMap<Object, LoaderLock> initOrGetLoaderMap() {
        if (loaderMap == null) {
            synchronized (this) {
                if (loaderMap == null) {
                    loaderMap = new ConcurrentHashMap<>();
                }
            }
        }
        return loaderMap;
    }
    @Override
    public final V computeIfAbsent(K key, Function<K, V> loader, boolean cacheNullWhenLoaderReturnNull) {
        return computeIfAbsentImpl(key, loader, cacheNullWhenLoaderReturnNull,
                0, null, this);
    }
    @Override
    public final V computeIfAbsent(K key, Function<K, V> loader, boolean cacheNullWhenLoaderReturnNull,
                                   long expireAfterWrite, TimeUnit timeUnit) {
        return computeIfAbsentImpl(key, loader, cacheNullWhenLoaderReturnNull,
                expireAfterWrite, timeUnit, this);
    }
    private static <K, V> boolean needUpdate(V loadedValue, boolean cacheNullWhenLoaderReturnNull, Function<K, V> loader) {
        if (loadedValue == null && !cacheNullWhenLoaderReturnNull) {
            return false;
        }
        if (loader instanceof CacheLoader && ((CacheLoader<K, V>) loader).vetoCacheUpdate()) {
            return false;
        }
        return true;
    }
    static <K, V> V computeIfAbsentImpl(K key, Function<K, V> loader, boolean cacheNullWhenLoaderReturnNull,
                                               long expireAfterWrite, TimeUnit timeUnit, Cache<K, V> cache) {
    	// 獲取內(nèi)部的 Cache 對象
        AbstractCache<K, V> abstractCache = CacheUtil.getAbstractCache(cache);
        // 封裝 loader 函數(shù)成一個 ProxyLoader 對象,主要在重新加載緩存后發(fā)出一個 CacheLoadEvent 到 CacheMonitor
        CacheLoader<K, V> newLoader = CacheUtil.createProxyLoader(cache, loader, abstractCache::notify);
        CacheGetResult<V> r;
        if (cache instanceof RefreshCache) { // 該緩存實例需要刷新
            RefreshCache<K, V> refreshCache = ((RefreshCache<K, V>) cache);
            /*
             * 從緩存中獲取數(shù)據(jù)
             * 如果是多級緩存(先從本地緩存獲取,獲取不到則從遠(yuǎn)程緩存獲?。?
             * 如果緩存數(shù)據(jù)是從遠(yuǎn)程緩存獲取到的數(shù)據(jù)則會更新至本地緩存,并且如果本地緩存沒有設(shè)置 localExpire 則使用遠(yuǎn)程緩存的到期時間作為自己的到期時間
             * 我一般不設(shè)置 localExpire ,因為可能導(dǎo)致本地緩存的有效時間比遠(yuǎn)程緩存的有效時間更長
             * 如果設(shè)置 localExpire 了記得設(shè)置 expireAfterAccessInMillis
             */
            r = refreshCache.GET(key);
            // 添加/更新當(dāng)前 RefreshCache 的刷新緩存任務(wù),存放于 RefreshCache 的 taskMap 中
            refreshCache.addOrUpdateRefreshTask(key, newLoader);
        } else {
            // 從緩存中獲取數(shù)據(jù)
            r = cache.GET(key);
        }
        if (r.isSuccess()) { // 緩存命中
            return r.getValue();
        } else { // 緩存未命中
            // 創(chuàng)建當(dāng)緩存未命中去更新緩存的函數(shù)
            Consumer<V> cacheUpdater = (loadedValue) -> {
                if(needUpdate(loadedValue, cacheNullWhenLoaderReturnNull, newLoader)) {
                    /*
                     * 未在緩存注解中配置 key 的生成方式則默認(rèn)取入?yún)⒆鳛榫彺?key
                     * 在進(jìn)入當(dāng)前方法時是否可以考慮為 key 創(chuàng)建一個副本????
                     * 因為緩存未命中然后通過 loader 重新加載方法時,如果方法內(nèi)部對入?yún)⑦M(jìn)行了修改,那么生成的緩存 key 也會被修改
                     * 從而導(dǎo)致相同的 key 進(jìn)入該方法時一直與緩存中的 key 不相同,一直出現(xiàn)緩存未命中
                     */
                    if (timeUnit != null) {
                        cache.PUT(key, loadedValue, expireAfterWrite, timeUnit).waitForResult();
                    } else {
                        cache.PUT(key, loadedValue).waitForResult();
                    }
                }
            };
            V loadedValue;
            if (cache.config().isCachePenetrationProtect()) { // 添加了 @CachePenetrationProtect 注解
            	// 一個JVM只允許一個線程執(zhí)行
                loadedValue = synchronizedLoad(cache.config(), abstractCache, key, newLoader, cacheUpdater);
            } else {
            	// 執(zhí)行方法
                loadedValue = newLoader.apply(key);
                // 將新的結(jié)果異步緩存
                cacheUpdater.accept(loadedValue);
            }
            return loadedValue;
        }
    }
    static <K, V> V synchronizedLoad(CacheConfig config, AbstractCache<K,V> abstractCache,
                                     K key, Function<K, V> newLoader, Consumer<V> cacheUpdater) {
        ConcurrentHashMap<Object, LoaderLock> loaderMap = abstractCache.initOrGetLoaderMap();
        Object lockKey = buildLoaderLockKey(abstractCache, key);
        while (true) {
            // 為什么加一個 create[] 數(shù)組 疑問??
            boolean create[] = new boolean[1];
            LoaderLock ll = loaderMap.computeIfAbsent(lockKey, (unusedKey) -> {
                create[0] = true;
                LoaderLock loaderLock = new LoaderLock();
                loaderLock.signal = new CountDownLatch(1);
                loaderLock.loaderThread = Thread.currentThread();
                return loaderLock;
            });
            if (create[0] || ll.loaderThread == Thread.currentThread()) {
                try {
                    // 加載該 Key 實例的方法
                    V loadedValue = newLoader.apply(key);
                    ll.success = true;
                    ll.value = loadedValue;
                    // 將重新加載的數(shù)據(jù)更新至緩存
                    cacheUpdater.accept(loadedValue);
                    return loadedValue;
                } finally {
                    // 標(biāo)記已完成
                    ll.signal.countDown();
                    if (create[0]) {
                        loaderMap.remove(lockKey);
                    }
                }
            } else { // 等待其他線程加載,如果出現(xiàn)異?;蛘叱瑫r則自己加載返回數(shù)據(jù),但是不更新緩存
                try {
                    Duration timeout = config.getPenetrationProtectTimeout();
                    if (timeout == null) {
                        ll.signal.await();
                    } else {
                        boolean ok = ll.signal.await(timeout.toMillis(), TimeUnit.MILLISECONDS);
                        if(!ok) {
                            logger.info("loader wait timeout:" + timeout);
                            return newLoader.apply(key);
                        }
                    }
                } catch (InterruptedException e) {
                    logger.warn("loader wait interrupted");
                    return newLoader.apply(key);
                }
                if (ll.success) {
                    return (V) ll.value;
                } else {
                    continue;
                }
            }
        }
    }
    private static Object buildLoaderLockKey(Cache c, Object key) {
        if (c instanceof AbstractEmbeddedCache) {
            return ((AbstractEmbeddedCache) c).buildKey(key);
        } else if (c instanceof AbstractExternalCache) {
            byte bytes[] = ((AbstractExternalCache) c).buildKey(key);
            return ByteBuffer.wrap(bytes);
        } else if (c instanceof MultiLevelCache) {
            c = ((MultiLevelCache) c).caches()[0];
            return buildLoaderLockKey(c, key);
        } else if(c instanceof ProxyCache) {
            c = ((ProxyCache) c).getTargetCache();
            return buildLoaderLockKey(c, key);
        } else {
            throw new CacheException("impossible");
        }
    }
    /**
     * 重新加載數(shù)據(jù)鎖
     */
    static class LoaderLock {
        /**
         * 柵欄
         */
        CountDownLatch signal;
        /**
         * 持有的線程
         */
        Thread loaderThread;
        /**
         * 是否加載成功
         */
        boolean success;
        /**
         * 加載出來的數(shù)據(jù)
         */,
        Object value;
    }
}

com.alicp.jetcache.AbstractCache實現(xiàn)了Cache接口的大寫方法,內(nèi)部調(diào)用自己定義的抽象方法(以DO_開頭,交由不同的子類實現(xiàn)),操作緩存后發(fā)送相應(yīng)的事件CacheEvent,也就是調(diào)用自己定義的notify方法,遍歷每個CacheMonitor對該事件進(jìn)行后置操作,用于統(tǒng)計信息。

computeIfAbsentImpl方法實現(xiàn)了Cache接口的核心方法,從緩存實例中根據(jù)緩存key獲取緩存value,邏輯如下:

  1. 獲取cache的targetCache,因為我們通過@CreateCache注解創(chuàng)建的緩存實例將生成LazyInitCache對象,需要調(diào)用其getTargetCache方法才會完成緩存實例的初始化
  2. loader函數(shù)是對加載原有方法的封裝,這里再進(jìn)行一層封裝,封裝成ProxyLoader類型,目的是在加載原有方法后將發(fā)送CacheLoadEvent事件
  3. 從緩存實例中獲取對應(yīng)的緩存value,如果緩存實例對象是RefreshCache類型(在com.alicp.jetcache.anno.support.CacheContext.buildCache方法中會將cache包裝成CacheHandlerRefreshCache),則調(diào)用RefreshCache.addOrUpdateRefreshTask方法,判斷是否應(yīng)該為它添加一個定時的刷新任務(wù)
  4. 如果緩存未命中,則執(zhí)行l(wèi)oader函數(shù),如果開啟了保護(hù)模式,則調(diào)用自定義的synchronizedLoad方法,大致邏輯:根據(jù)緩存key從自己的loaderMap(線程安全)遍歷中嘗試獲?。ú淮嬖趧t創(chuàng)建)LoaderLock加載鎖,獲取到這把加載鎖才可以執(zhí)行l(wèi)oader函數(shù),如果已被其他線程占有則進(jìn)行等待(沒有設(shè)置超時時間則一直等待),通過CountDownLatch計數(shù)器實現(xiàn)

AbstractEmbeddedCache本地緩存

com.alicp.jetcache.embedded.AbstractEmbeddedCache抽象類繼承AbstractCache抽象類,定義了本地緩存的存放緩存數(shù)據(jù)的對象為com.alicp.jetcache.embedded.InnerMap接口和一個初始化該接口的createAreaCache抽象方法,基于InnerMap接口實現(xiàn)以DO_開頭的方法,完成緩存實例各種操作的具體實現(xiàn),主要代碼如下:

public abstract class AbstractEmbeddedCache<K, V> extends AbstractCache<K, V> {
    protected EmbeddedCacheConfig<K, V> config;
    /**
     * 本地緩存的 Map
     */
    protected InnerMap innerMap;
    protected abstract InnerMap createAreaCache();
    public AbstractEmbeddedCache(EmbeddedCacheConfig<K, V> config) {
        this.config = config;
        innerMap = createAreaCache();
    }
    @Override
    public CacheConfig<K, V> config() {
        return config;
    }
    public Object buildKey(K key) {
        Object newKey = key;
        Function<K, Object> keyConvertor = config.getKeyConvertor();
        if (keyConvertor != null) {
            newKey = keyConvertor.apply(key);
        }
        return newKey;
    }
    @Override
    protected CacheGetResult<V> do_GET(K key) {
        Object newKey = buildKey(key);
        CacheValueHolder<V> holder = (CacheValueHolder<V>) innerMap.getValue(newKey);
        return parseHolderResult(holder);
    }
    protected CacheGetResult<V> parseHolderResult(CacheValueHolder<V> holder) {
        long now = System.currentTimeMillis();
        if (holder == null) {
            return CacheGetResult.NOT_EXISTS_WITHOUT_MSG;
        } else if (now >= holder.getExpireTime()) {
            return CacheGetResult.EXPIRED_WITHOUT_MSG;
        } else {
            synchronized (holder) {
                long accessTime = holder.getAccessTime();
                if (config.isExpireAfterAccess()) {
                    long expireAfterAccess = config.getExpireAfterAccessInMillis();
                    if (now >= accessTime + expireAfterAccess) {
                        return CacheGetResult.EXPIRED_WITHOUT_MSG;
                    }
                }
                // 設(shè)置該緩存數(shù)據(jù)的最后一次訪問時間
                holder.setAccessTime(now);
            }
            return new CacheGetResult(CacheResultCode.SUCCESS, null, holder);
        }
    }
    @Override
    protected MultiGetResult<K, V> do_GET_ALL(Set<? extends K> keys) {
        ArrayList<K> keyList = new ArrayList<K>(keys.size());
        ArrayList<Object> newKeyList = new ArrayList<Object>(keys.size());
        keys.stream().forEach((k) -> {
            Object newKey = buildKey(k);
            keyList.add(k);
            newKeyList.add(newKey);
        });
        Map<Object, CacheValueHolder<V>> innerResultMap = innerMap.getAllValues(newKeyList);
        Map<K, CacheGetResult<V>> resultMap = new HashMap<>();
        for (int i = 0; i < keyList.size(); i++) {
            K key = keyList.get(i);
            Object newKey = newKeyList.get(i);
            CacheValueHolder<V> holder = innerResultMap.get(newKey);
            resultMap.put(key, parseHolderResult(holder));
        }
        MultiGetResult<K, V> result = new MultiGetResult<>(CacheResultCode.SUCCESS, null, resultMap);
        return result;
    }
    @Override
    protected CacheResult do_PUT(K key, V value, long expireAfterWrite, TimeUnit timeUnit) {
        CacheValueHolder<V> cacheObject = new CacheValueHolder(value ,timeUnit.toMillis(expireAfterWrite));
        innerMap.putValue(buildKey(key), cacheObject);
        return CacheResult.SUCCESS_WITHOUT_MSG;
    }
    @Override
    protected CacheResult do_PUT_ALL(Map<? extends K, ? extends V> map, long expireAfterWrite, TimeUnit timeUnit) {
        HashMap newKeyMap = new HashMap();
        for (Map.Entry<? extends K, ? extends V> en : map.entrySet()) {
            CacheValueHolder<V> cacheObject = new CacheValueHolder(en.getValue(), timeUnit.toMillis(expireAfterWrite));
            newKeyMap.put(buildKey(en.getKey()), cacheObject);
        }
        innerMap.putAllValues(newKeyMap);
        final HashMap resultMap = new HashMap();
        map.keySet().forEach((k) -> resultMap.put(k, CacheResultCode.SUCCESS));
        return CacheResult.SUCCESS_WITHOUT_MSG;
    }
    @Override
    protected CacheResult do_REMOVE(K key) {
        innerMap.removeValue(buildKey(key));
        return CacheResult.SUCCESS_WITHOUT_MSG;
    }
    @Override
    protected CacheResult do_REMOVE_ALL(Set<? extends K> keys) {
        Set newKeys = keys.stream().map((key) -> buildKey(key)).collect(Collectors.toSet());
        innerMap.removeAllValues(newKeys);
        final HashMap resultMap = new HashMap();
        keys.forEach((k) -> resultMap.put(k, CacheResultCode.SUCCESS));
        return CacheResult.SUCCESS_WITHOUT_MSG;
    }
    @Override
    protected CacheResult do_PUT_IF_ABSENT(K key, V value, long expireAfterWrite, TimeUnit timeUnit) {
        CacheValueHolder<V> cacheObject = new CacheValueHolder(value, timeUnit.toMillis(expireAfterWrite));
        if (innerMap.putIfAbsentValue(buildKey(key), cacheObject)) {
            return CacheResult.SUCCESS_WITHOUT_MSG;
        } else {
            return CacheResult.EXISTS_WITHOUT_MSG;
        }
    }
}

com.alicp.jetcache.embedded.AbstractEmbeddedCache抽象類實現(xiàn)了操作本地緩存的相關(guān)方法

  • 定義了緩存實例對象本地緩存的配置信息EmbeddedCacheConfig對象
  • 定義了緩存實例對象本地緩存基于內(nèi)存操作緩存數(shù)據(jù)的InnerMap對象,它的初始化過程交由不同的內(nèi)存緩存實例(LinkedHashMapCache和CaffeineCache)

LinkedHashMapCache

com.alicp.jetcache.embedded.LinkedHashMapCache基于LinkedHashMap完成緩存實例對象本地緩存基于內(nèi)存操作緩存數(shù)據(jù)的InnerMap對象的初始化工作,主要代碼如下:

public class LinkedHashMapCache<K, V> extends AbstractEmbeddedCache<K, V> {
    private static Logger logger = LoggerFactory.getLogger(LinkedHashMapCache.class);
    public LinkedHashMapCache(EmbeddedCacheConfig<K, V> config) {
        super(config);
        // 將緩存實例添加至 Cleaner
        addToCleaner();
    }
    protected void addToCleaner() {
        Cleaner.add(this);
    }
    @Override
    protected InnerMap createAreaCache() {
        return new LRUMap(config.getLimit(), this);
    }
    public void cleanExpiredEntry() {
        ((LRUMap) innerMap).cleanExpiredEntry();
    }
    /**
     * 用于本地緩存類型為 linkedhashmap 緩存實例存儲緩存數(shù)據(jù)
     */
    final class LRUMap extends LinkedHashMap implements InnerMap {
        /**
         * 允許的最大緩存數(shù)量
         */
        private final int max;
        /**
         * 緩存實例鎖
         */
        private Object lock;
        public LRUMap(int max, Object lock) {
            super((int) (max * 1.4f), 0.75f, true);
            this.max = max;
            this.lock = lock;
        }
        /**
         * 當(dāng)元素大于最大值時移除最老的元素
         *
         * @param eldest 最老的元素
         * @return 是否刪除
         */
        @Override
        protected boolean removeEldestEntry(Map.Entry eldest) {
            return size() > max;
        }
        /**
         * 清理過期的元素
         */
        void cleanExpiredEntry() {
            synchronized (lock) { // 占有當(dāng)前緩存實例這把鎖
                for (Iterator it = entrySet().iterator(); it.hasNext();) {
                    Map.Entry en = (Map.Entry) it.next();
                    Object value = en.getValue();
                    if (value != null && value instanceof CacheValueHolder) {
                        CacheValueHolder h = (CacheValueHolder) value;
                        /*
                         * 緩存的數(shù)據(jù)已經(jīng)失效了則刪除
                         * 為什么不對 expireAfterAccess 進(jìn)行判斷,取最小值,疑問????
                         */
                        if (System.currentTimeMillis() >= h.getExpireTime()) {
                            it.remove();
                        }
                    } else {
                        // assert false
                        if (value == null) {
                            logger.error("key " + en.getKey() + " is null");
                        } else {
                            logger.error("value of key " + en.getKey() + " is not a CacheValueHolder. type=" + value.getClass());
                        }
                    }
                }
            }
        }
        @Override
        public Object getValue(Object key) {
            synchronized (lock) {
                return get(key);
            }
        }
        @Override
        public Map getAllValues(Collection keys) {
            Map values = new HashMap();
            synchronized (lock) {
                for (Object key : keys) {
                    Object v = get(key);
                    if (v != null) {
                        values.put(key, v);
                    }
                }
            }
            return values;
        }
        @Override
        public void putValue(Object key, Object value) {
            synchronized (lock) {
                put(key, value);
            }
        }
        @Override
        public void putAllValues(Map map) {
            synchronized (lock) {
                Set<Map.Entry> set = map.entrySet();
                for (Map.Entry en : set) {
                    put(en.getKey(), en.getValue());
                }
            }
        }
        @Override
        public boolean removeValue(Object key) {
            synchronized (lock) {
                return remove(key) != null;
            }
        }
        @Override
        public void removeAllValues(Collection keys) {
            synchronized (lock) {
                for (Object k : keys) {
                    remove(k);
                }
            }
        }
        @Override
        @SuppressWarnings("unchecked")
        public boolean putIfAbsentValue(Object key, Object value) {
            /*
             * 如果緩存 key 不存在,或者對應(yīng)的 value 已經(jīng)失效則放入,否則返回 false
             */
            synchronized (lock) {
                CacheValueHolder h = (CacheValueHolder) get(key);
                if (h == null || parseHolderResult(h).getResultCode() == CacheResultCode.EXPIRED) {
                    put(key, value);
                    return true;
                } else {
                    return false;
                }
            }
        }
    }
}

com.alicp.jetcache.embedded.LinkedHashMapCache自定義LRUMap繼承LinkedHashMap并實現(xiàn)InnerMap接口

  1. 自定義max字段,存儲元素個數(shù)的最大值,并設(shè)置初始容量為(max * 1.4f)
  2. 自定義lock字段,每個緩存實例的鎖,通過synchronized關(guān)鍵詞保證線程安全,所以性能相對來說不好
  3. 覆蓋LinkedHashMap的removeEldestEntry方法,當(dāng)元素大于最大值時移除最老的元素
  4. 自定義cleanExpiredEntry方法,遍歷Map,根據(jù)緩存value(被封裝成的com.alicp.jetcache.CacheValueHolder對象,包含緩存數(shù)據(jù)、失效時間戳和第一次訪問的時間),清理過期的元素
  5. 該對象初始化時會被添加至com.alicp.jetcache.embedded.Cleaner清理器中,Cleaner會周期性(每隔60秒)遍歷LinkedHashMapCache緩存實例,調(diào)用其cleanExpiredEntry方法

Cleaner清理器

com.alicp.jetcache.embedded.Cleaner用于清理緩存類型為LinkedHashMapCache的緩存數(shù)據(jù),請查看相應(yīng)注釋,代碼如下:

/**
 * 執(zhí)行任務(wù):定時清理(每分鐘) LinkedHashMapCache 緩存實例中過期的緩存數(shù)據(jù)
 */
class Cleaner {
    /**
     * 存放弱引用對象,以防內(nèi)存溢出
     * 如果被弱引用的對象只被當(dāng)前弱引用對象關(guān)聯(lián)時,gc 時被弱引用的對象則會被回收(取決于被弱引用的對象是否還與其他強(qiáng)引用對象關(guān)聯(lián))
     *
     * 個人理解:當(dāng)某個 LinkedHashMapCache 強(qiáng)引用對象沒有被其他對象(除了這里)引用時,我們應(yīng)該讓這個對象被回收,
     * 但是由于這里使用的也是強(qiáng)引用,這個對象被其他強(qiáng)引用對象關(guān)聯(lián)了,不可能被回收,存在內(nèi)存溢出的危險,
     * 所以這里使用了弱引用對象,如果被弱引用的對象沒有被其他對象(除了這里)引用時,這個對象會被回收
     *
     * 舉個例子:如果我們往一個 Map<Object, Object> 中存放一個key-value鍵值對
     * 假設(shè)對應(yīng)的鍵已經(jīng)不再使用被回收了,那我們無法再獲取到對應(yīng)的值,也無法被回收,占有一定的內(nèi)存,存在風(fēng)險
     */
    static LinkedList<WeakReference<LinkedHashMapCache>> linkedHashMapCaches = new LinkedList<>();
    static {
        // 創(chuàng)建一個線程池,1個核心線程
        ScheduledExecutorService executorService = JetCacheExecutor.defaultExecutor();
        // 起一個循環(huán)任務(wù)一直清理 linkedHashMapCaches 過期的數(shù)據(jù)(每隔60秒)
        executorService.scheduleWithFixedDelay(() -> run(), 60, 60, TimeUnit.SECONDS);
    }
    static void add(LinkedHashMapCache cache) {
        synchronized (linkedHashMapCaches) {
            // 創(chuàng)建一個弱引用對象,并添加到清理對象中
            linkedHashMapCaches.add(new WeakReference<>(cache));
        }
    }
    static void run() {
        synchronized (linkedHashMapCaches) {
            Iterator<WeakReference<LinkedHashMapCache>> it = linkedHashMapCaches.iterator();
            while (it.hasNext()) {
                WeakReference<LinkedHashMapCache> ref = it.next();
                // 獲取被弱引用的對象(強(qiáng)引用)
                LinkedHashMapCache c = ref.get();
                if (c == null) { // 表示被弱引用的對象被標(biāo)記成了垃圾,則移除
                    it.remove();
                } else {
                    c.cleanExpiredEntry();
                }
            }
        }
    }
}

CaffeineCache

com.alicp.jetcache.embedded.CaffeineCache基于Caffeine完成緩存實例對象本地緩存基于內(nèi)存操作緩存數(shù)據(jù)的InnerMap對象的初始化工作,主要代碼如下:

public class CaffeineCache<K, V> extends AbstractEmbeddedCache<K, V> {
    /**
     * 緩存實例對象
     */
    private com.github.benmanes.caffeine.cache.Cache cache;
    public CaffeineCache(EmbeddedCacheConfig<K, V> config) {
        super(config);
    }
    /**
     * 初始化本地緩存的容器
     *
     * @return Map對象
     */
    @Override
    @SuppressWarnings("unchecked")
    protected InnerMap createAreaCache() {
        Caffeine<Object, Object> builder = Caffeine.newBuilder();
        // 設(shè)置緩存實例的最大緩存數(shù)量
        builder.maximumSize(config.getLimit());
        final boolean isExpireAfterAccess = config.isExpireAfterAccess();
        final long expireAfterAccess = config.getExpireAfterAccessInMillis();
        // 設(shè)置緩存實例的緩存數(shù)據(jù)的失效策略
        builder.expireAfter(new Expiry<Object, CacheValueHolder>() {
            /**
             * 獲取緩存的有效時間
             *
             * @param value 緩存數(shù)據(jù)
             * @return 有效時間
             */
            private long getRestTimeInNanos(CacheValueHolder value) {
                long now = System.currentTimeMillis();
                long ttl = value.getExpireTime() - now;
                /*
                 * 如果本地緩存設(shè)置了多長時間沒訪問緩存則失效
                 */
                if(isExpireAfterAccess){
                    // 設(shè)置緩存的失效時間
                    // 多長時間沒訪問緩存則失效 and 緩存的有效時長取 min
                    ttl = Math.min(ttl, expireAfterAccess);
                }
                return TimeUnit.MILLISECONDS.toNanos(ttl);
            }
            @Override
            public long expireAfterCreate(Object key, CacheValueHolder value, long currentTime) {
                return getRestTimeInNanos(value);
            }
            @Override
            public long expireAfterUpdate(Object key, CacheValueHolder value,
                                          long currentTime, long currentDuration) {
                return currentDuration;
            }
            @Override
            public long expireAfterRead(Object key, CacheValueHolder value,
                                        long currentTime, long currentDuration) {
                return getRestTimeInNanos(value);
            }
        });
        // 構(gòu)建 Cache 緩存實例
        cache = builder.build();
        return new InnerMap() {
            @Override
            public Object getValue(Object key) {
                return cache.getIfPresent(key);
            }
            @Override
            public Map getAllValues(Collection keys) {
                return cache.getAllPresent(keys);
            }
            @Override
            public void putValue(Object key, Object value) {
                cache.put(key, value);
            }
            @Override
            public void putAllValues(Map map) {
                cache.putAll(map);
            }
            @Override
            public boolean removeValue(Object key) {
                return cache.asMap().remove(key) != null;
            }
            @Override
            public void removeAllValues(Collection keys) {
                cache.invalidateAll(keys);
            }
            @Override
            public boolean putIfAbsentValue(Object key, Object value) {
                return cache.asMap().putIfAbsent(key, value) == null;
            }
        };
    }
}

com.alicp.jetcache.embedded.CaffeineCache通過Caffeine構(gòu)建一個com.github.benmanes.caffeine.cache.Cache緩存對象,然后實現(xiàn)InnerMap接口,調(diào)用這個緩存對象的相關(guān)方法

  1. 構(gòu)建時設(shè)置每個元素的過期時間,也就是根據(jù)每個元素(com.alicp.jetcache.CacheValueHolder)的失效時間戳來設(shè)置,底層如何實現(xiàn)的可以參考Caffeine官方地址
  2. 調(diào)用com.github.benmanes.caffeine.cache.Cache的put方法我有遇到過'unable to create native thread'內(nèi)存溢出的問題,所以請結(jié)合實際業(yè)務(wù)場景合理的設(shè)置緩存相關(guān)配置

AbstractExternalCache遠(yuǎn)程緩存

com.alicp.jetcache.embedded.AbstractExternalCache抽象類繼承AbstractCache抽象類,定義了緩存實例對象遠(yuǎn)程緩存的配置信息ExternalCacheConfig對象,提供了將緩存key轉(zhuǎn)換成字節(jié)數(shù)組的方法,代碼比較簡單。

RedisCache

com.alicp.jetcache.redis.RedisCache使用Jedis連接Redis,對遠(yuǎn)程的緩存數(shù)據(jù)進(jìn)行操作,代碼沒有很復(fù)雜,可查看我的注釋

  1. 定義了com.alicp.jetcache.redis.RedisCacheConfig配置對象,包含Redis連接池的相關(guān)信息
  2. 實現(xiàn)了以DO_開頭的方法,也就是通過Jedis操作緩存數(shù)據(jù)

RedisLettuceCache

com.alicp.jetcache.redis.lettuce.RedisLettuceCache使用Lettuce連接Redis,對遠(yuǎn)程的緩存數(shù)據(jù)進(jìn)行操作,代碼沒有很復(fù)雜,可查看我的注釋

  1. 定義了com.alicp.jetcache.redis.lettuce.RedisLettuceCacheConfig配置對象,包含Redis客戶端、與Redis建立的安全連接等信息,因為底層是基于Netty實現(xiàn)的,所以無需配置線程池
  2. 使用com.alicp.jetcache.redis.lettuce.LettuceConnectionManager自定義管理器將與Redis連接的相關(guān)信息封裝成LettuceObjects對象,并管理RedisClient與LettuceObjects對應(yīng)關(guān)系
  3. 相比Jedis更加安全高效
  4. Lettuce不了解的可以參考我寫的測試類com.alicp.jetcache.test.external.LettuceTest

MultiLevelCache兩級緩存

當(dāng)你設(shè)置了緩存類型為BOTH兩級緩存,那么創(chuàng)建的實例對象會被封裝成com.alicp.jetcache.MultiLevelCache對象

  1. 定義了caches字段類型為Cache[],用于保存AbstractEmbeddedCache本地緩存實例和AbstractExternalCache遠(yuǎn)程緩存實例,本地緩存存放于遠(yuǎn)程緩存前面
  2. 實現(xiàn)了do_GET方法,遍歷caches數(shù)組,也就是先從本地緩存獲取,如果獲取緩存不成功則從遠(yuǎn)程緩存獲取,成功獲取到緩存后會調(diào)用checkResultAndFillUpperCache方法
  3. checkResultAndFillUpperCache方法的邏輯可以看到,將獲取到的緩存數(shù)據(jù)更新至更底層的緩存中,也就是說如果緩存數(shù)據(jù)是從遠(yuǎn)程獲取到的,那么進(jìn)入這個方法后會將獲取到的緩存數(shù)據(jù)更新到本地緩存中去,這樣下次請求可以直接從本地緩存獲取,避免與Redis之間的網(wǎng)絡(luò)消耗
  4. 實現(xiàn)了do_PUT方法,遍歷caches數(shù)組,通過CompletableFuture進(jìn)行異步編程,將所有的操作綁定在一條鏈上執(zhí)行。
  5. 實現(xiàn)的了PUT(K key, V value)方法,會先判斷是否單獨配置了本地緩存時間localExipre,配置了則單獨為本地緩存設(shè)置過期時間,沒有配置則到期時間和遠(yuǎn)程緩存的一樣
  6. 覆蓋tryLock方法,調(diào)用caches[caches.length-1].tryLock方法,也就是只會調(diào)用最頂層遠(yuǎn)程緩存的這個方法

主要代碼如下:

public class MultiLevelCache<K, V> extends AbstractCache<K, V> {
    private Cache[] caches;
    private MultiLevelCacheConfig<K, V> config;
    @SuppressWarnings("unchecked")
    @Deprecated
    public MultiLevelCache(Cache... caches) throws CacheConfigException {
        this.caches = caches;
        checkCaches();
        CacheConfig lastConfig = caches[caches.length - 1].config();
        config = new MultiLevelCacheConfig<>();
        config.setCaches(Arrays.asList(caches));
        config.setExpireAfterWriteInMillis(lastConfig.getExpireAfterWriteInMillis());
        config.setCacheNullValue(lastConfig.isCacheNullValue());
    }
    @SuppressWarnings("unchecked")
    public MultiLevelCache(MultiLevelCacheConfig<K, V> cacheConfig) throws CacheConfigException {
        this.config = cacheConfig;
        this.caches = cacheConfig.getCaches().toArray(new Cache[]{});
        checkCaches();
    }
    private void checkCaches() {
        if (caches == null || caches.length == 0) {
            throw new IllegalArgumentException();
        }
        for (Cache c : caches) {
            if (c.config().getLoader() != null) {
                throw new CacheConfigException("Loader on sub cache is not allowed, set the loader into MultiLevelCache.");
            }
        }
    }
    public Cache[] caches() {
        return caches;
    }
    @Override
    public MultiLevelCacheConfig<K, V> config() {
        return config;
    }
    @Override
    public CacheResult PUT(K key, V value) {
        if (config.isUseExpireOfSubCache()) { // 本地緩存使用自己的失效時間
            // 設(shè)置了TimeUnit為null,本地緩存則使用自己的到期時間
            return PUT(key, value, 0, null);
        } else {
            return PUT(key, value, config().getExpireAfterWriteInMillis(), TimeUnit.MILLISECONDS);
        }
    }
    @Override
    public CacheResult PUT_ALL(Map<? extends K, ? extends V> map) {
        if (config.isUseExpireOfSubCache()) {
            return PUT_ALL(map, 0, null);
        } else {
            return PUT_ALL(map, config().getExpireAfterWriteInMillis(), TimeUnit.MILLISECONDS);
        }
    }
    @Override
    protected CacheGetResult<V> do_GET(K key) {
    	// 遍歷多級緩存(遠(yuǎn)程緩存排在后面)
        for (int i = 0; i < caches.length; i++) {
            Cache cache = caches[i];
            CacheGetResult result = cache.GET(key);
            if (result.isSuccess()) {
                CacheValueHolder<V> holder = unwrapHolder(result.getHolder());
                /*
                 * 這個遍歷是從低層的緩存開始獲取,獲取成功則將該值設(shè)置到更低層的緩存中
                 * 情景:
                 * 本地沒有獲取到緩存,遠(yuǎn)程獲取到了緩存,這里會將遠(yuǎn)程的緩存數(shù)據(jù)設(shè)置到本地中,
                 * 這樣下次請求則直接從本次獲取,減少了遠(yuǎn)程獲取的時間
                 */
                checkResultAndFillUpperCache(key, i, holder);
                return new CacheGetResult(CacheResultCode.SUCCESS, null, holder);
            }
        }
        return CacheGetResult.NOT_EXISTS_WITHOUT_MSG;
    }
    private CacheValueHolder<V> unwrapHolder(CacheValueHolder<V> h) {
        // if @Cached or @CacheCache change type from REMOTE to BOTH (or from BOTH to REMOTE),
        // during the dev/publish process, the value type which different application server put into cache server will be different
        // (CacheValueHolder<V> and CacheValueHolder<CacheValueHolder<V>>, respectively).
        // So we need correct the problem at here and in CacheGetResult.
        Objects.requireNonNull(h);
        if (h.getValue() instanceof CacheValueHolder) {
            return (CacheValueHolder<V>) h.getValue();
        } else {
            return h;
        }
    }
    private void checkResultAndFillUpperCache(K key, int i, CacheValueHolder<V> h) {
        Objects.requireNonNull(h);
        long currentExpire = h.getExpireTime();
        long now = System.currentTimeMillis();
        if (now <= currentExpire) {
            if(config.isUseExpireOfSubCache()){ // 如果使用本地自己的緩存過期時間
                // 使用本地緩存自己的過期時間
                PUT_caches(i, key, h.getValue(), 0, null);
            } else { // 使用遠(yuǎn)程緩存的過期時間
                long restTtl = currentExpire - now;
                if (restTtl > 0) { // 遠(yuǎn)程緩存數(shù)據(jù)還未失效,則重新設(shè)置本地的緩存
                    PUT_caches(i, key, h.getValue(), restTtl, TimeUnit.MILLISECONDS);
                }
            }
        }
    }
    @Override
    protected MultiGetResult<K, V> do_GET_ALL(Set<? extends K> keys) {
        HashMap<K, CacheGetResult<V>> resultMap = new HashMap<>();
        Set<K> restKeys = new HashSet<>(keys);
        for (int i = 0; i < caches.length; i++) {
            if (restKeys.size() == 0) {
                break;
            }
            Cache<K, CacheValueHolder<V>> c = caches[i];
            MultiGetResult<K, CacheValueHolder<V>> allResult = c.GET_ALL(restKeys);
            if (allResult.isSuccess() && allResult.getValues() != null) {
                for (Map.Entry<K, CacheGetResult<CacheValueHolder<V>>> en : allResult.getValues().entrySet()) {
                    K key = en.getKey();
                    CacheGetResult result = en.getValue();
                    if (result.isSuccess()) {
                        CacheValueHolder<V> holder = unwrapHolder(result.getHolder());
                        checkResultAndFillUpperCache(key, i, holder);
                        resultMap.put(key, new CacheGetResult(CacheResultCode.SUCCESS, null, holder));
                        restKeys.remove(key);
                    }
                }
            }
        }
        for (K k : restKeys) {
            resultMap.put(k, CacheGetResult.NOT_EXISTS_WITHOUT_MSG);
        }
        return new MultiGetResult<>(CacheResultCode.SUCCESS, null, resultMap);
    }
    @Override
    protected CacheResult do_PUT(K key, V value, long expireAfterWrite, TimeUnit timeUnit) {
        return PUT_caches(caches.length, key, value, expireAfterWrite, timeUnit);
    }
    @Override
    protected CacheResult do_PUT_ALL(Map<? extends K, ? extends V> map, long expireAfterWrite, TimeUnit timeUnit) {
        CompletableFuture<ResultData> future = CompletableFuture.completedFuture(null);
        for (Cache c : caches) {
            CacheResult r;
            if(timeUnit == null) {
                r = c.PUT_ALL(map);
            } else {
                r = c.PUT_ALL(map, expireAfterWrite, timeUnit);
            }
            future = combine(future, r);
        }
        return new CacheResult(future);
    }
    private CacheResult PUT_caches(int lastIndex, K key, V value, long expire, TimeUnit timeUnit) {
        CompletableFuture<ResultData> future = CompletableFuture.completedFuture(null);
        for (int i = 0; i < lastIndex; i++) {
            Cache cache = caches[i];
            CacheResult r;
            if (timeUnit == null) { // 表示本地緩存使用自己過期時間
                r = cache.PUT(key, value);
            } else {
                r = cache.PUT(key, value, expire, timeUnit);
            }
            // 將多個 PUT 操作放在一條鏈上
            future = combine(future, r);
        }
        return new CacheResult(future);
    }
    private CompletableFuture<ResultData> combine(CompletableFuture<ResultData> future, CacheResult result) {
        return future.thenCombine(result.future(), (d1, d2) -> {
            if (d1 == null) {
                return d2;
            }
            if (d1.getResultCode() != d2.getResultCode()) {
                return new ResultData(CacheResultCode.PART_SUCCESS, null, null);
            }
            return d1;
        });
    }
    @Override
    protected CacheResult do_REMOVE(K key) {
        CompletableFuture<ResultData> future = CompletableFuture.completedFuture(null);
        for (Cache cache : caches) {
            CacheResult r = cache.REMOVE(key);
            future = combine(future, r);
        }
        return new CacheResult(future);
    }
    @Override
    protected CacheResult do_REMOVE_ALL(Set<? extends K> keys) {
        CompletableFuture<ResultData> future = CompletableFuture.completedFuture(null);
        for (Cache cache : caches) {
            CacheResult r = cache.REMOVE_ALL(keys);
            future = combine(future, r);
        }
        return new CacheResult(future);
    }
    @Override
    public <T> T unwrap(Class<T> clazz) {
        Objects.requireNonNull(clazz);
        for (Cache cache : caches) {
            try {
                T obj = (T) cache.unwrap(clazz);
                if (obj != null) {
                    return obj;
                }
            } catch (IllegalArgumentException e) {
                // ignore
            }
        }
        throw new IllegalArgumentException(clazz.getName());
    }
    @Override
    public AutoReleaseLock tryLock(K key, long expire, TimeUnit timeUnit) {
        if (key == null) {
            return null;
        }
        return caches[caches.length - 1].tryLock(key, expire, timeUnit);
    }
    @Override
    public boolean putIfAbsent(K key, V value) {
        throw new UnsupportedOperationException("putIfAbsent is not supported by MultiLevelCache");
    }
    @Override
    protected CacheResult do_PUT_IF_ABSENT(K key, V value, long expireAfterWrite, TimeUnit timeUnit) {
        throw new UnsupportedOperationException("PUT_IF_ABSENT is not supported by MultiLevelCache");
    }
    @Override
    public void close() {
        for (Cache c : caches) {
            c.close();
        }
    }
}

RefreshCache

com.alicp.jetcache.RefreshCache為緩存實例添加刷新任務(wù),前面在AbstractCache抽象類中講到了,在com.alicp.jetcache.anno.support.CacheContext.buildCache方法中會將cache包裝成CacheHandlerRefreshCache,所以說每個緩存實例都會調(diào)用一下addOrUpdateRefreshTask方法,代碼如下:

public class RefreshCache<K, V> extends LoadingCache<K, V> {
    
    protected CacheConfig<K, V> config;
    
    /**
	 * 用于保存刷新任務(wù)
	 */
	private ConcurrentHashMap<Object, RefreshTask> taskMap = new ConcurrentHashMap<>();
    
    protected void addOrUpdateRefreshTask(K key, CacheLoader<K, V> loader) {
		// 獲取緩存刷新策略
		RefreshPolicy refreshPolicy = config.getRefreshPolicy();
		if (refreshPolicy == null) { // 沒有則不進(jìn)行刷新
			return;
		}
		// 獲取刷新時間間隔
		long refreshMillis = refreshPolicy.getRefreshMillis();
		if (refreshMillis > 0) {
			// 獲取線程任務(wù)的ID
			Object taskId = getTaskId(key);
			// 獲取對應(yīng)的RefreshTask,不存在則創(chuàng)建一個
			RefreshTask refreshTask = taskMap.computeIfAbsent(taskId, tid -> {
				logger.debug("add refresh task. interval={},  key={}", refreshMillis, key);
				RefreshTask task = new RefreshTask(taskId, key, loader);
				task.lastAccessTime = System.currentTimeMillis();
				/*
				 * 獲取 ScheduledExecutorService 周期/延遲線程池,10個核心線程,創(chuàng)建的線程都是守護(hù)線程
				 * scheduleWithFixedDelay(Runnable command, long initialDelay, long period, TimeUnit unit)
				 * 運行的任務(wù)task、多久延遲后開始執(zhí)行、后續(xù)執(zhí)行的周期間隔多長,時間單位
				 * 通過其創(chuàng)建一個循環(huán)任務(wù),用于刷新緩存數(shù)據(jù)
				 */
				ScheduledFuture<?> future = JetCacheExecutor.heavyIOExecutor().scheduleWithFixedDelay(task,
						refreshMillis, refreshMillis, TimeUnit.MILLISECONDS);
				task.future = future;
				return task;
			});
			// 設(shè)置最后一次訪問時間
			refreshTask.lastAccessTime = System.currentTimeMillis();
		}
	}
}

如果緩存實例配置了刷新策略并且刷新間隔大于0,則會從taskMap(線程安全)中嘗試獲取對應(yīng)的刷新任務(wù)RefreshTask,如果不存在則創(chuàng)建一個任務(wù)放入線程池周期性的執(zhí)行

com.alicp.jetcache.RefreshCache.RefreshTask代碼如下:

public class RefreshCache<K, V> extends LoadingCache<K, V> {
    
    protected Cache concreteCache() {
		Cache c = getTargetCache();
		while (true) {
			if (c instanceof ProxyCache) {
				c = ((ProxyCache) c).getTargetCache();
			} else if (c instanceof MultiLevelCache) {
				Cache[] caches = ((MultiLevelCache) c).caches();
				// 如果是兩級緩存則返回遠(yuǎn)程緩存
				c = caches[caches.length - 1];
			} else {
				return c;
			}
		}
	}
    
    class RefreshTask implements Runnable {
		/**
		 * 唯一標(biāo)志符,也就是Key轉(zhuǎn)換后的值
		 */
		private Object taskId;
		/**
		 * 緩存的Key
		 */
		private K key;
		/**
		 * 執(zhí)行方法的CacheLoader對象
		 */
		private CacheLoader<K, V> loader;
		/**
		 * 最后一次訪問時間
		 */
		private long lastAccessTime;
		/**
		 * 該 Task 的執(zhí)行策略
		 */
		private ScheduledFuture future;
		RefreshTask(Object taskId, K key, CacheLoader<K, V> loader) {
			this.taskId = taskId;
			this.key = key;
			this.loader = loader;
		}
		private void cancel() {
			logger.debug("cancel refresh: {}", key);
			// 嘗試中斷當(dāng)前任務(wù)
			future.cancel(false);
			// 從任務(wù)列表中刪除
			taskMap.remove(taskId);
		}
		/**
		 * 重新加載數(shù)據(jù)
		 *
		 * @throws Throwable 異常
		 */
		private void load() throws Throwable {
			CacheLoader<K, V> l = loader == null ? config.getLoader() : loader;
			if (l != null) {
				// 封裝 CacheLoader 成 ProxyLoader,加載后會發(fā)起 Load 事件
				l = CacheUtil.createProxyLoader(cache, l, eventConsumer);
				// 加載
				V v = l.load(key);
				if (needUpdate(v, l)) {
					// 將重新加載的數(shù)據(jù)放入緩存
					cache.PUT(key, v);
				}
			}
		}
		/**
		 * 遠(yuǎn)程加載數(shù)據(jù)
		 *
		 * @param concreteCache 緩存對象
		 * @param currentTime   當(dāng)前時間
		 * @throws Throwable 異常
		 */
		private void externalLoad(final Cache concreteCache, final long currentTime) throws Throwable {
			// 獲取 Key 轉(zhuǎn)換后的值
			byte[] newKey = ((AbstractExternalCache) concreteCache).buildKey(key);
			// 創(chuàng)建分布式鎖對應(yīng)的Key
			byte[] lockKey = combine(newKey, "_#RL#".getBytes());
			// 分布式鎖的存在時間
			long loadTimeOut = RefreshCache.this.config.getRefreshPolicy().getRefreshLockTimeoutMillis();
			// 刷新間隔
			long refreshMillis = config.getRefreshPolicy().getRefreshMillis();
			// Key對應(yīng)的時間戳Key(用于存放上次刷新時間)
			byte[] timestampKey = combine(newKey, "_#TS#".getBytes());
			// AbstractExternalCache buildKey method will not convert byte[]
			// 獲取Key上一次刷新時間
			CacheGetResult refreshTimeResult = concreteCache.GET(timestampKey);
			boolean shouldLoad = false; // 是否需要重新加載
			if (refreshTimeResult.isSuccess()) {
				// 當(dāng)前時間與上一次刷新的時間間隔是否大于或等于刷新間隔
				shouldLoad = currentTime >= Long.parseLong(refreshTimeResult.getValue().toString()) + refreshMillis;
			} else if (refreshTimeResult.getResultCode() == CacheResultCode.NOT_EXISTS) { // 無緩存
				shouldLoad = true;
			}
			if (!shouldLoad) {
				if (multiLevelCache) {
					// 將頂層的緩存數(shù)據(jù)更新至低層的緩存中,例如將遠(yuǎn)程的緩存數(shù)據(jù)放入本地緩存
					// 因為如果是多級緩存,創(chuàng)建刷新任務(wù)后,我們只需更新遠(yuǎn)程的緩存,然后從遠(yuǎn)程緩存獲取緩存數(shù)據(jù)更新低層的緩存,保證緩存一致
					refreshUpperCaches(key);
				}
				return;
			}
			// 重新加載
			Runnable r = () -> {
				try {
					load();
					// AbstractExternalCache buildKey method will not convert byte[]
					// 保存一個key-value至redis,其中的信息為該value的生成時間,刷新緩存
					concreteCache.put(timestampKey, String.valueOf(System.currentTimeMillis()));
				} catch (Throwable e) {
					throw new CacheException("refresh error", e);
				}
			};
			// AbstractExternalCache buildKey method will not convert byte[]
			// 分布式緩存沒有一個全局分配的功能,這里嘗試獲取一把非嚴(yán)格的分布式鎖,獲取鎖的超時時間默認(rèn)60秒,也就是獲取到這把鎖最多可以擁有60秒
			// 只有獲取Key對應(yīng)的這把分布式鎖,才執(zhí)行重新加載的操作
			boolean lockSuccess = concreteCache.tryLockAndRun(lockKey, loadTimeOut, TimeUnit.MILLISECONDS, r);
			if (!lockSuccess && multiLevelCache) { // 沒有獲取到鎖并且是多級緩存
				// 這個時候應(yīng)該有其他實例在刷新緩存,所以這里設(shè)置過一會直接獲取遠(yuǎn)程的緩存數(shù)據(jù)更新到本地
				// 創(chuàng)建一個延遲任務(wù)(1/5刷新間隔后),將最頂層的緩存數(shù)據(jù)更新至每一層
				JetCacheExecutor.heavyIOExecutor().schedule(() -> refreshUpperCaches(key), (long) (0.2 * refreshMillis),
						TimeUnit.MILLISECONDS);
			}
		}
		private void refreshUpperCaches(K key) {
			MultiLevelCache<K, V> targetCache = (MultiLevelCache<K, V>) getTargetCache();
			Cache[] caches = targetCache.caches();
			int len = caches.length;
			// 獲取多級緩存中頂層的緩存數(shù)據(jù)
			CacheGetResult cacheGetResult = caches[len - 1].GET(key);
			if (!cacheGetResult.isSuccess()) {
				return;
			}
			// 將緩存數(shù)據(jù)重新放入低層緩存
			for (int i = 0; i < len - 1; i++) {
				caches[i].PUT(key, cacheGetResult.getValue());
			}
		}
		/**
		 * 刷新任務(wù)的具體執(zhí)行
		 */
		@Override
		public void run() {
			try {
				if (config.getRefreshPolicy() == null || (loader == null && !hasLoader())) {
					// 取消執(zhí)行
					cancel();
					return;
				}
				long now = System.currentTimeMillis();
				long stopRefreshAfterLastAccessMillis = config.getRefreshPolicy().getStopRefreshAfterLastAccessMillis();
				if (stopRefreshAfterLastAccessMillis > 0) {
					// 最后一次訪問到現(xiàn)在時間的間隔超過了設(shè)置的 stopRefreshAfterLastAccessMillis,則取消當(dāng)前任務(wù)執(zhí)行
					if (lastAccessTime + stopRefreshAfterLastAccessMillis < now) {
						logger.debug("cancel refresh: {}", key);
						cancel();
						return;
					}
				}
				logger.debug("refresh key: {}", key);
				// 獲取緩存實例對象,如果是多層則返回頂層,也就是遠(yuǎn)程緩存
				Cache concreteCache = concreteCache();
				if (concreteCache instanceof AbstractExternalCache) { // 遠(yuǎn)程緩存刷新
					externalLoad(concreteCache, now);
				} else { // 本地緩存刷新
					load();
				}
			} catch (Throwable e) {
				logger.error("refresh error: key=" + key, e);
			}
		}
	}
}

刷新邏輯:

  1. 判斷是否需要停止刷新了,需要的話調(diào)用其future的cancel方法取消執(zhí)行,并從taskMap中刪除
  2. 獲取緩存實例對象,如果是多層則返回頂層,也就是遠(yuǎn)程緩存實例對象
  3. 如果是本地緩存,則調(diào)用load方法,也就是執(zhí)行l(wèi)oader函數(shù)加載原有方法,將獲取到的數(shù)據(jù)更新至緩存實例中(如果是多級緩存,則每級緩存都會更新)
  4. 如果是遠(yuǎn)程緩存對象,則調(diào)用externalLoad方法,刷新后會往Redis中存放一個鍵值對,key為key_#TS#,value為上一次刷新時間

先從Redis中獲取上一次刷新時間的鍵值對,根據(jù)上一次刷新的時間判斷是否大于刷新間隔,大于(或者沒有上一次刷新時間)表示需要重新加載數(shù)據(jù),否則不需要重新加載數(shù)據(jù)

如果不需要重新加載數(shù)據(jù),但是又是多級緩存,則獲取遠(yuǎn)程緩存數(shù)據(jù)更新至本地緩存,保證兩級緩存的一致性

如果需要重新加載數(shù)據(jù),則調(diào)用tryLockAndRun方法,嘗試獲取分布式鎖,執(zhí)行刷新任務(wù)(調(diào)用load方法,并往Redis中重新設(shè)置上一次的刷新時間),如果沒有獲取到分布式鎖,則創(chuàng)建一個延遲任務(wù)(1/5刷新間隔后)將最頂層的緩存數(shù)據(jù)更新至每一層

解析配置

主要查看jetcache-autoconfigure子模塊,解析application.yml中jetcache相關(guān)配置,初始化不同緩存類型的CacheBuilder構(gòu)造器,用于生產(chǎn)緩存實例,也初始化以下對象:

com.alicp.jetcache.anno.support.ConfigProvider:緩存管理器,注入了全局配置GlobalCacheConfig、緩存實例管理器SimpleCacheManager、緩存上下文CacheContext等大量信息

com.alicp.jetcache.autoconfigure.AutoConfigureBeans:存儲CacheBuilder構(gòu)造器以及Redis的相關(guān)信息

com.alicp.jetcache.anno.support.GlobalCacheConfig:全局配置類,保存了一些全局信息

初始化構(gòu)造器

通過@Conditional注解將需要使用到的緩存類型對應(yīng)的構(gòu)造器初始化類注入到Spring容器并執(zhí)行初始化過程,也就是創(chuàng)建CacheBuilder構(gòu)造器

初始化構(gòu)造器類的類型結(jié)構(gòu)如下圖所示:

主要對象描述:

AbstractCacheAutoInit:抽象類,實現(xiàn)Spring的InitializingBean接口,注入至Spring容器時完成初始化

EmbeddedCacheAutoInit:抽象類,繼承AbstractCacheAutoInit,解析本地緩存獨有的配置

LinkedHashMapAutoConfiguration:初始化LinkedHashMapCacheBuilder構(gòu)造器

CaffeineAutoConfiguration:初始化CaffeineCacheBuilder構(gòu)造器

ExternalCacheAutoInit:抽象類,繼承AbstractCacheAutoInit,解析遠(yuǎn)程緩存獨有的配置

RedisAutoInit:初始化RedisCacheBuilder構(gòu)造器

RedisLettuceAutoInit:初始化RedisLettuceCacheBuilder構(gòu)造器

AbstractCacheAutoInit

com.alicp.jetcache.autoconfigure.AbstractCacheAutoInit抽象類主要實現(xiàn)了Spring的InitializingBean接口,在注入Spring容器時,Spring會調(diào)用其afterPropertiesSet方法,完成本地緩存類型和遠(yuǎn)程緩存類型CacheBuilder構(gòu)造器的初始化,主要代碼如下:

public abstract class AbstractCacheAutoInit implements InitializingBean {
    @Autowired
    protected ConfigurableEnvironment environment;
    @Autowired
    protected AutoConfigureBeans autoConfigureBeans;
    @Autowired
    protected ConfigProvider configProvider;
    protected String[] typeNames;
    private boolean inited = false;
    public AbstractCacheAutoInit(String... cacheTypes) {
        Objects.requireNonNull(cacheTypes,"cacheTypes can't be null");
        Assert.isTrue(cacheTypes.length > 0, "cacheTypes length is 0");
        this.typeNames = cacheTypes;
    }
    /**
     * 初始化方法
     */
    @Override
    public void afterPropertiesSet() {
        if (!inited) {
            synchronized (this) {
                if (!inited) {
                    // 這里我們有兩個指定前綴 'jetcache.local' 'jetcache.remote'
                    process("jetcache.local.", autoConfigureBeans.getLocalCacheBuilders(), true);
                    process("jetcache.remote.", autoConfigureBeans.getRemoteCacheBuilders(), false);
                    inited = true;
                }
            }
        }
    }
    private void process(String prefix, Map cacheBuilders, boolean local) {
        // 創(chuàng)建一個配置對象(本地或者遠(yuǎn)程)
        ConfigTree resolver = new ConfigTree(environment, prefix);
        // 獲取本地或者遠(yuǎn)程的配置項
        Map<String, Object> m = resolver.getProperties();
        // 獲取本地或者遠(yuǎn)程的 area ,這里我一般只有默認(rèn)的 default
        Set<String> cacheAreaNames = resolver.directChildrenKeys();
        for (String cacheArea : cacheAreaNames) {
            // 獲取本地或者遠(yuǎn)程存儲類型,例如 caffeine,redis.lettuce
            final Object configType = m.get(cacheArea + ".type");
            // 緩存類型是否和當(dāng)前 CacheAutoInit 的某一個 typeName 匹配(不同的 CacheAutoInit 會設(shè)置一個或者多個 typename)
            boolean match = Arrays.stream(typeNames).anyMatch((tn) -> tn.equals(configType));
            /*
             * 因為有很多 CacheAutoInit 繼承者,都會執(zhí)行這個方法,不同的繼承者解析不同的配置
             * 例如 CaffeineAutoConfiguration 只解析 jetcache.local.default.type=caffeine 即可
             * RedisLettuceAutoInit 只解析 jetcache.remote.default.type=redis.lettuce 即可
             */
            if (!match) {
                continue;
            }
            // 獲取本地或者遠(yuǎn)程的 area 的子配置項
            ConfigTree ct = resolver.subTree(cacheArea + ".");
            logger.info("init cache area {} , type= {}", cacheArea, typeNames[0]);
            // 根據(jù)配置信息構(gòu)建本地或者遠(yuǎn)程緩存的 CacheBuilder 構(gòu)造器
            CacheBuilder c = initCache(ct, local ? "local." + cacheArea : "remote." + cacheArea);
            // 將 CacheBuilder 構(gòu)造器存放至 AutoConfigureBeans
            cacheBuilders.put(cacheArea, c);
        }
    }
    /**
     * 設(shè)置公共的配置到 CacheBuilder 構(gòu)造器中
     *
     * @param builder 構(gòu)造器
     * @param ct      配置信息
     */
    protected void parseGeneralConfig(CacheBuilder builder, ConfigTree ct) {
        AbstractCacheBuilder acb = (AbstractCacheBuilder) builder;
        // 設(shè)置 Key 的轉(zhuǎn)換函數(shù)
        acb.keyConvertor(configProvider.parseKeyConvertor(ct.getProperty("keyConvertor")));
        // 設(shè)置超時時間
        String expireAfterWriteInMillis = ct.getProperty("expireAfterWriteInMillis");
        if (expireAfterWriteInMillis == null) {
            // compatible with 2.1 兼容老版本
            expireAfterWriteInMillis = ct.getProperty("defaultExpireInMillis");
        }
        if (expireAfterWriteInMillis != null) {
            acb.setExpireAfterWriteInMillis(Long.parseLong(expireAfterWriteInMillis));
        }
        // 多長時間沒有訪問就讓緩存失效,0表示不使用該功能(注意:只支持本地緩存)
        String expireAfterAccessInMillis = ct.getProperty("expireAfterAccessInMillis");
        if (expireAfterAccessInMillis != null) {
            acb.setExpireAfterAccessInMillis(Long.parseLong(expireAfterAccessInMillis));
        }
    }
    /**
     * 初始化 CacheBuilder 構(gòu)造器交由子類去實現(xiàn)
     *
     * @param ct                  配置信息
     * @param cacheAreaWithPrefix 配置前綴
     * @return CacheBuilder 構(gòu)造器
     */
    protected abstract CacheBuilder initCache(ConfigTree ct, String cacheAreaWithPrefix);
}

1.在afterPropertiesSet()方法中可以看到會調(diào)用process方法分別初始化本地緩存和遠(yuǎn)程緩存的構(gòu)造器

2.定義的process方法:

  1. 首先會從當(dāng)前環(huán)境中解析出JetCache的相關(guān)配置到ConfigTree對象中
  2. 然后遍歷緩存區(qū)域,獲取對應(yīng)的緩存類型type,進(jìn)行不同類型的緩存實例CacheBuilder構(gòu)造器初始化過程
  3. 不同CacheBuilder構(gòu)造器的初始化方法initCache交由子類實現(xiàn)
  4. 獲取到CacheBuilder構(gòu)造器后會將其放入AutoConfigureBeans對象中去

3.另外也定義了parseGeneralConfig方法解析本地緩存和遠(yuǎn)程緩存都有的配置至CacheBuilder構(gòu)造器中

EmbeddedCacheAutoInit

com.alicp.jetcache.autoconfigure.EmbeddedCacheAutoInit抽象類繼承了AbstractCacheAutoInit,主要是覆蓋父類的parseGeneralConfig,解析本地緩存單有的配置limit,代碼如下:

public abstract class EmbeddedCacheAutoInit extends AbstractCacheAutoInit {
    public EmbeddedCacheAutoInit(String... cacheTypes) {
        super(cacheTypes);
    }
    @Override
    protected void parseGeneralConfig(CacheBuilder builder, ConfigTree ct) {
        super.parseGeneralConfig(builder, ct);
        EmbeddedCacheBuilder ecb = (EmbeddedCacheBuilder) builder;
        // 設(shè)置本地緩存每個緩存實例的緩存數(shù)量個數(shù)限制(默認(rèn)100)
        ecb.limit(Integer.parseInt(ct.getProperty("limit", String.valueOf(CacheConsts.DEFAULT_LOCAL_LIMIT))));
    }
}

LinkedHashMapAutoConfiguration

com.alicp.jetcache.autoconfigure.LinkedHashMapAutoConfiguration繼承了EmbeddedCacheAutoInit,實現(xiàn)了initCache方法,先通過LinkedHashMapCacheBuilder創(chuàng)建一個默認(rèn)實現(xiàn)類,然后解析相關(guān)配置至構(gòu)造器中完成初始化,代碼如下:

@Component
@Conditional(LinkedHashMapAutoConfiguration.LinkedHashMapCondition.class)
public class LinkedHashMapAutoConfiguration extends EmbeddedCacheAutoInit {
    public LinkedHashMapAutoConfiguration() {
        super("linkedhashmap");
    }
    @Override
    protected CacheBuilder initCache(ConfigTree ct, String cacheAreaWithPrefix) {
        // 創(chuàng)建一個 LinkedHashMapCacheBuilder 構(gòu)造器
        LinkedHashMapCacheBuilder builder = LinkedHashMapCacheBuilder.createLinkedHashMapCacheBuilder();
        // 解析相關(guān)配置至 LinkedHashMapCacheBuilder 的 CacheConfig 中
        parseGeneralConfig(builder, ct);
        return builder;
    }
    public static class LinkedHashMapCondition extends JetCacheCondition {
        // 配置了緩存類型為 linkedhashmap 當(dāng)前類才會被注入 Spring 容器
        public LinkedHashMapCondition() {
            super("linkedhashmap");
        }
    }
}
  1. 這里我們注意到@Conditional注解,這個注解的作用是:滿足SpringBootCondition條件這個Bean才會被Spring容器管理
  2. 他的條件是LinkedHashMapCondition,繼承了JetCacheCondition,也就是說配置文件中配置了緩存類型為linkedhashmap時這個類才會被Spring容器管理,才會完成LinkedHashMapCacheBuilder構(gòu)造器的初始化
  3. JetCacheCondition邏輯并不復(fù)雜,可自行查看

CaffeineAutoConfiguration

com.alicp.jetcache.autoconfigure.CaffeineAutoConfiguration繼承了EmbeddedCacheAutoInit,實現(xiàn)了initCache方法,先通過CaffeineCacheBuilder創(chuàng)建一個默認(rèn)實現(xiàn)類,然后解析相關(guān)配置至構(gòu)造器中完成初始化,代碼如下:

@Component
@Conditional(CaffeineAutoConfiguration.CaffeineCondition.class)
public class CaffeineAutoConfiguration extends EmbeddedCacheAutoInit {
    public CaffeineAutoConfiguration() {
        super("caffeine");
    }
    @Override
    protected CacheBuilder initCache(ConfigTree ct, String cacheAreaWithPrefix) {
        // 創(chuàng)建一個 CaffeineCacheBuilder 構(gòu)造器
        CaffeineCacheBuilder builder = CaffeineCacheBuilder.createCaffeineCacheBuilder();
        // 解析相關(guān)配置至 CaffeineCacheBuilder 的 CacheConfig 中
        parseGeneralConfig(builder, ct);
        return builder;
    }
    public static class CaffeineCondition extends JetCacheCondition {
        // 配置了緩存類型為 caffeine 當(dāng)前類才會被注入 Spring 容器
        public CaffeineCondition() {
            super("caffeine");
        }
    }
}
  1. 同樣使用了@Conditional注解,這個注解的作用是:滿足SpringBootCondition條件這個Bean才會被Spring容器管理
  2. 他的條件是CaffeineCondition,繼承了JetCacheCondition,也就是說配置文件中配置了緩存類型為caffeine時這個類才會被Spring容器管理,才會完成LinkedHashMapCacheBuilder構(gòu)造器的初始化

ExternalCacheAutoInit

com.alicp.jetcache.autoconfigure.ExternalCacheAutoInit抽象類繼承了AbstractCacheAutoInit,主要是覆蓋父類的parseGeneralConfig,解析遠(yuǎn)程緩存單有的配置keyPrefix、valueEncodervalueDecoder,代碼如下:

public abstract class ExternalCacheAutoInit extends AbstractCacheAutoInit {
    public ExternalCacheAutoInit(String... cacheTypes) {
        super(cacheTypes);
    }
    /**
     * 設(shè)置遠(yuǎn)程緩存 CacheBuilder 構(gòu)造器的相關(guān)配置
     *
     * @param builder 構(gòu)造器
     * @param ct      配置信息
     */
    @Override
    protected void parseGeneralConfig(CacheBuilder builder, ConfigTree ct) {
        super.parseGeneralConfig(builder, ct);
        ExternalCacheBuilder ecb = (ExternalCacheBuilder) builder;
        // 設(shè)置遠(yuǎn)程緩存 key 的前綴
        ecb.setKeyPrefix(ct.getProperty("keyPrefix"));
        /*
         * 根據(jù)配置創(chuàng)建緩存數(shù)據(jù)的編碼函數(shù)和解碼函數(shù)
         */
        ecb.setValueEncoder(configProvider.parseValueEncoder(ct.getProperty("valueEncoder", CacheConsts.DEFAULT_SERIAL_POLICY)));
        ecb.setValueDecoder(configProvider.parseValueDecoder(ct.getProperty("valueDecoder", CacheConsts.DEFAULT_SERIAL_POLICY)));
    }
}

RedisAutoInit

com.alicp.jetcache.autoconfigure.RedisAutoInit繼承了ExternalCacheAutoInit,實現(xiàn)initCache方法,完成了通過Jedis連接Redis的初始化操作,主要代碼如下:

@Configuration
@Conditional(RedisAutoConfiguration.RedisCondition.class)
public class RedisAutoConfiguration {
    public static final String AUTO_INIT_BEAN_NAME = "redisAutoInit";
    @Bean(name = AUTO_INIT_BEAN_NAME)
    public RedisAutoInit redisAutoInit() {
        return new RedisAutoInit();
    }
    public static class RedisCondition extends JetCacheCondition {
        // 配置了緩存類型為 redis 當(dāng)前類才會被注入 Spring 容器
        public RedisCondition() {
            super("redis");
        }
    }
    public static class RedisAutoInit extends ExternalCacheAutoInit {
        public RedisAutoInit() {
            // 設(shè)置緩存類型
            super("redis");
        }
        @Autowired
        private AutoConfigureBeans autoConfigureBeans;
        @Override
        protected CacheBuilder initCache(ConfigTree ct, String cacheAreaWithPrefix) {
            Pool jedisPool = parsePool(ct);
            Pool[] slavesPool = null;
            int[] slavesPoolWeights = null;
            // 是否只從 Redis 的從節(jié)點讀取數(shù)據(jù)
            boolean readFromSlave = Boolean.parseBoolean(ct.getProperty("readFromSlave", "False"));
            // 獲取從節(jié)點的配置信息
            ConfigTree slaves = ct.subTree("slaves.");
            Set<String> slaveNames = slaves.directChildrenKeys();
            // 依次創(chuàng)建每個從節(jié)點的連接池
            if (slaveNames.size() > 0) {
                slavesPool = new Pool[slaveNames.size()];
                slavesPoolWeights = new int[slaveNames.size()];
                int i = 0;
                for (String slaveName: slaveNames) {
                    ConfigTree slaveConfig = slaves.subTree(slaveName + ".");
                    slavesPool[i] = parsePool(slaveConfig);
                    slavesPoolWeights[i] = Integer.parseInt(slaveConfig.getProperty("weight","100"));
                    i++;
                }
            }
            // 創(chuàng)建一個 RedisCacheBuilder 構(gòu)造器
            ExternalCacheBuilder externalCacheBuilder = RedisCacheBuilder.createRedisCacheBuilder()
                    .jedisPool(jedisPool)
                    .readFromSlave(readFromSlave)
                    .jedisSlavePools(slavesPool)
                    .slaveReadWeights(slavesPoolWeights);
            // 解析相關(guān)配置至 RedisCacheBuilder 的 CacheConfig 中
            parseGeneralConfig(externalCacheBuilder, ct);
            // eg: "jedisPool.remote.default"
            autoConfigureBeans.getCustomContainer().put("jedisPool." + cacheAreaWithPrefix, jedisPool);
            return externalCacheBuilder;
        }
        /**
         * 創(chuàng)建 Redis 連接池
         *
         * @param ct 配置信息
         * @return 連接池
         */
        private Pool<Jedis> parsePool(ConfigTree ct) {
            // 創(chuàng)建連接池配置對象
            GenericObjectPoolConfig poolConfig = parsePoolConfig(ct);
            String host = ct.getProperty("host", (String) null);
            int port = Integer.parseInt(ct.getProperty("port", "0"));
            int timeout = Integer.parseInt(ct.getProperty("timeout", String.valueOf(Protocol.DEFAULT_TIMEOUT)));
            String password = ct.getProperty("password", (String) null);
            int database = Integer.parseInt(ct.getProperty("database", String.valueOf(Protocol.DEFAULT_DATABASE)));
            String clientName = ct.getProperty("clientName", (String) null);
            boolean ssl = Boolean.parseBoolean(ct.getProperty("ssl", "false"));
            String masterName = ct.getProperty("masterName", (String) null);
            String sentinels = ct.getProperty("sentinels", (String) null);//ip1:port,ip2:port
            Pool<Jedis> jedisPool;
            if (sentinels == null) {
                Objects.requireNonNull(host, "host/port or sentinels/masterName is required");
                if (port == 0) {
                    throw new IllegalStateException("host/port or sentinels/masterName is required");
                }
                // 創(chuàng)建一個 Jedis 連接池
                jedisPool = new JedisPool(poolConfig, host, port, timeout, password, database, clientName, ssl);
            } else {
                Objects.requireNonNull(masterName, "host/port or sentinels/masterName is required");
                String[] strings = sentinels.split(",");
                HashSet<String> sentinelsSet = new HashSet<>();
                for (String s : strings) {
                    if (s != null && !s.trim().equals("")) {
                        sentinelsSet.add(s.trim());
                    }
                }
                // 創(chuàng)建一個 Jedis Sentine 連接池
                jedisPool = new JedisSentinelPool(masterName, sentinelsSet, poolConfig, timeout, password, database, clientName);
            }
            return jedisPool;
        }
    }
}

com.alicp.jetcache.autoconfigure.RedisAutoInitcom.alicp.jetcache.autoconfigure.RedisAutoConfiguration內(nèi)部的靜態(tài)類,在RedisAutoConfiguration內(nèi)通過redisAutoInit()方法定義RedisAutoInit作為Spring Bean

同樣RedisAutoConfiguration使用了@Conditional注解,滿足SpringBootCondition條件這個Bean才會被Spring容器管理,內(nèi)部的RedisAutoInit也不會被管理,也就是說配置文件中配置了緩存類型為redis時RedisLettuceAutoInit才會被Spring容器管理,才會完成RedisLettuceCacheBuilder構(gòu)造器的初始化

實現(xiàn)了initCache方法

  • 先解析Redis的相關(guān)配置
  • 通過Jedis創(chuàng)建Redis連接池
  • 通過RedisCacheBuilder創(chuàng)建一個默認(rèn)實現(xiàn)類
  • 解析相關(guān)配置至構(gòu)造器中完成初始化
  • 將Redis連接保存至AutoConfigureBeans

RedisLettuceAutoInit

com.alicp.jetcache.autoconfigure.RedisLettuceAutoInit繼承了ExternalCacheAutoInit,實現(xiàn)initCache方法,完成了通過Lettuce連接Redis的初始化操作,主要代碼如下:

@Configuration
@Conditional(RedisLettuceAutoConfiguration.RedisLettuceCondition.class)
public class RedisLettuceAutoConfiguration {
    public static final String AUTO_INIT_BEAN_NAME = "redisLettuceAutoInit";
    /**
     * 注入 spring 容器的條件
     */
    public static class RedisLettuceCondition extends JetCacheCondition {
        // 配置了緩存類型為 redis.lettuce 當(dāng)前類才會被注入 Spring 容器
        public RedisLettuceCondition() {
            super("redis.lettuce");
        }
    }
    @Bean(name = {AUTO_INIT_BEAN_NAME})
    public RedisLettuceAutoInit redisLettuceAutoInit() {
        return new RedisLettuceAutoInit();
    }
    public static class RedisLettuceAutoInit extends ExternalCacheAutoInit {
        public RedisLettuceAutoInit() {
            // 設(shè)置緩存類型
            super("redis.lettuce");
        }
        /**
         * 初始化 RedisLettuceCacheBuilder 構(gòu)造器
         *
         * @param ct                  配置信息
         * @param cacheAreaWithPrefix 配置前綴
         * @return 構(gòu)造器
         */
        @Override
        protected CacheBuilder initCache(ConfigTree ct, String cacheAreaWithPrefix) {
            Map<String, Object> map = ct.subTree("uri"/*there is no dot*/).getProperties();
            // 數(shù)據(jù)節(jié)點偏好設(shè)置
            String readFromStr = ct.getProperty("readFrom");
            // 集群模式
            String mode = ct.getProperty("mode");
            // 異步獲取結(jié)果的超時時間,默認(rèn)1s
            long asyncResultTimeoutInMillis = Long.parseLong(
                    ct.getProperty("asyncResultTimeoutInMillis", Long.toString(CacheConsts.ASYNC_RESULT_TIMEOUT.toMillis())));
            ReadFrom readFrom = null;
            if (readFromStr != null) {
                /*
                 * MASTER:只從Master節(jié)點中讀取。
                 * MASTER_PREFERRED:優(yōu)先從Master節(jié)點中讀取。
                 * SLAVE_PREFERRED:優(yōu)先從Slave節(jié)點中讀取。
                 * SLAVE:只從Slave節(jié)點中讀取。
                 * NEAREST:使用最近一次連接的Redis實例讀取。
                 */
                readFrom = ReadFrom.valueOf(readFromStr.trim());
            }
            AbstractRedisClient client;
            StatefulConnection connection = null;
            if (map == null || map.size() == 0) {
                throw new CacheConfigException("lettuce uri is required");
            } else {
                // 創(chuàng)建對應(yīng)的 RedisURI
                List<RedisURI> uriList = map.values().stream().map((k) -> RedisURI.create(URI.create(k.toString())))
                        .collect(Collectors.toList());
                if (uriList.size() == 1) { // 只有一個 URI,集群模式只給一個域名怎么辦 TODO 疑問??
                    RedisURI uri = uriList.get(0);
                    if (readFrom == null) {
                        // 創(chuàng)建一個 Redis 客戶端
                        client = RedisClient.create(uri);
                        // 設(shè)置失去連接時的行為,拒絕命令,默認(rèn)為 DEFAULT
                        ((RedisClient) client).setOptions(ClientOptions.builder().
                                disconnectedBehavior(ClientOptions.DisconnectedBehavior.REJECT_COMMANDS).build());
                    } else {
                        // 創(chuàng)建一個 Redis 客戶端
                        client = RedisClient.create();
                        ((RedisClient) client).setOptions(ClientOptions.builder().
                                disconnectedBehavior(ClientOptions.DisconnectedBehavior.REJECT_COMMANDS).build());
                        // 創(chuàng)建一個安全連接并設(shè)置數(shù)據(jù)節(jié)點偏好
                        StatefulRedisMasterSlaveConnection c = MasterSlave.connect(
                                (RedisClient) client, new JetCacheCodec(), uri);
                        c.setReadFrom(readFrom);
                        connection = c;
                    }
                } else { // 多個 URI,集群模式
                    if (mode != null && mode.equalsIgnoreCase("MasterSlave")) {
                        client = RedisClient.create();
                        ((RedisClient) client).setOptions(ClientOptions.builder().
                                disconnectedBehavior(ClientOptions.DisconnectedBehavior.REJECT_COMMANDS).build());
                        StatefulRedisMasterSlaveConnection c = MasterSlave.connect(
                                (RedisClient) client, new JetCacheCodec(), uriList);
                        if (readFrom != null) {
                            c.setReadFrom(readFrom);
                        }
                        connection = c;
                    } else {
                        // 創(chuàng)建一個 Redis 客戶端
                        client = RedisClusterClient.create(uriList);
                        ((RedisClusterClient) client).setOptions(ClusterClientOptions.builder().
                                disconnectedBehavior(ClientOptions.DisconnectedBehavior.REJECT_COMMANDS).build());
                        if (readFrom != null) {
                            StatefulRedisClusterConnection c = ((RedisClusterClient) client).connect(new JetCacheCodec());
                            c.setReadFrom(readFrom);
                            connection = c;
                        }
                    }
                }
            }
            // 創(chuàng)建一個 RedisLettuceCacheBuilder 構(gòu)造器
            ExternalCacheBuilder externalCacheBuilder = RedisLettuceCacheBuilder.createRedisLettuceCacheBuilder()
                    .connection(connection)
                    .redisClient(client)
                    .asyncResultTimeoutInMillis(asyncResultTimeoutInMillis);
            // 解析相關(guān)配置至 RedisLettuceCacheBuilder 的 CacheConfig 中
            parseGeneralConfig(externalCacheBuilder, ct);
            // eg: "remote.default.client"
            autoConfigureBeans.getCustomContainer().put(cacheAreaWithPrefix + ".client", client);
            // 開始將 Redis 客戶端和安全連接保存至 LettuceConnectionManager 管理器中
            LettuceConnectionManager m = LettuceConnectionManager.defaultManager();
            // 初始化 Lettuce 連接 Redis
            m.init(client, connection);
            // 初始化 Redis 連接的相關(guān)信息保存至 LettuceObjects 中,并將相關(guān)信息保存至 AutoConfigureBeans.customContainer
            autoConfigureBeans.getCustomContainer().put(cacheAreaWithPrefix + ".connection", m.connection(client));
            autoConfigureBeans.getCustomContainer().put(cacheAreaWithPrefix + ".commands", m.commands(client));
            autoConfigureBeans.getCustomContainer().put(cacheAreaWithPrefix + ".asyncCommands", m.asyncCommands(client));
            autoConfigureBeans.getCustomContainer().put(cacheAreaWithPrefix + ".reactiveCommands", m.reactiveCommands(client));
            return externalCacheBuilder;
        }
    }
}

1.com.alicp.jetcache.autoconfigure.RedisLettuceAutoInitcom.alicp.jetcache.autoconfigure.RedisLettuceAutoConfiguration內(nèi)部的靜態(tài)類,在RedisLettuceAutoConfiguration內(nèi)通過redisLettuceAutoInit()方法定義RedisLettuceAutoInit作為Spring Bean

2.同樣RedisLettuceAutoConfiguration使用了@Conditional注解,滿足SpringBootCondition條件這個Bean才會被Spring容器管理,內(nèi)部的RedisLettuceAutoInit也不會被管理,也就是說配置文件中配置了緩存類型為redis.lettuce時RedisLettuceAutoInit才會被Spring容器管理,才會完成RedisLettuceCacheBuilder構(gòu)造器的初始化

3.實現(xiàn)了initCache方法

  • 先解析Redis的相關(guān)配置
  • 通過Lettuce創(chuàng)建Redis客戶端和與Redis的連接
  • 通過RedisLettuceCacheBuilder創(chuàng)建一個默認(rèn)實現(xiàn)類
  • 解析相關(guān)配置至構(gòu)造器中完成初始化
  • 獲取LettuceConnectionManager管理器,將通過Lettuce創(chuàng)建Redis客戶端和與Redis的連接保存
  • 將Redis客戶端、與Redis的連接、同步命令、異步命令和反應(yīng)式命令相關(guān)保存至AutoConfigureBeans

JetCacheAutoConfiguration自動配置

上面的初始化構(gòu)造器的類需要被Spring容器管理,就需被掃描到,我們一般會設(shè)置掃描路徑,但是別人引入JetCache肯定是作為其他包不能夠被掃描到的,這些Bean也就不會被Spring管理,這里我們查看jetcache-autoconfigure模塊下src/main/resources/META-INF/spring.factories文件,內(nèi)容如下:

org.springframework.boot.autoconfigure.EnableAutoConfiguration=com.alicp.jetcache.autoconfigure.JetCacheAutoConfiguration

這應(yīng)該是一種SPI機(jī)制,這樣這個項目以外的JetCache包里面的com.alicp.jetcache.autoconfigure.JetCacheAutoConfiguration就會被Spring容器掃描到,我們來看看他的代碼:

/**
 * 該 Bean 將會被 Spring 容器注入,依次注入下面幾個 Bean
 * SpringConfigProvider -> AutoConfigureBeans -> BeanDependencyManager(為 GlobalCacheConfig 添加 CacheAutoInit 依賴) -> GlobalCacheConfig
 * 由此會完成初始化配置操作,緩存實例構(gòu)造器 CacheBuilder 也會被注入容器
 *
 * Created on 2016/11/17.
 *
 * @author <a href="mailto:areyouok@gmail.com">huangli</a>
 */
@Configuration
@ConditionalOnClass(GlobalCacheConfig.class)
@ConditionalOnMissingBean(GlobalCacheConfig.class)
@EnableConfigurationProperties(JetCacheProperties.class)
@Import({RedisAutoConfiguration.class,
        CaffeineAutoConfiguration.class,
        MockRemoteCacheAutoConfiguration.class,
        LinkedHashMapAutoConfiguration.class,
        RedisLettuceAutoConfiguration.class,
        RedisSpringDataAutoConfiguration.class})
public class JetCacheAutoConfiguration {
    public static final String GLOBAL_CACHE_CONFIG_NAME = "globalCacheConfig";
    private SpringConfigProvider _springConfigProvider = new SpringConfigProvider();
    private AutoConfigureBeans _autoConfigureBeans = new AutoConfigureBeans();
    private GlobalCacheConfig _globalCacheConfig;
    @Bean
    @ConditionalOnMissingBean
    public SpringConfigProvider springConfigProvider() {
        return _springConfigProvider;
    }
    @Bean
    public AutoConfigureBeans autoConfigureBeans() {
        return _autoConfigureBeans;
    }
    @Bean
    public static BeanDependencyManager beanDependencyManager(){
        return new BeanDependencyManager();
    }
    @Bean(name = GLOBAL_CACHE_CONFIG_NAME)
    public GlobalCacheConfig globalCacheConfig(SpringConfigProvider configProvider,
                                                            AutoConfigureBeans autoConfigureBeans,
                                                            JetCacheProperties props) {
        if (_globalCacheConfig != null) {
            return _globalCacheConfig;
        }
        _globalCacheConfig = new GlobalCacheConfig();
        _globalCacheConfig.setHiddenPackages(props.getHiddenPackages());
        _globalCacheConfig.setStatIntervalMinutes(props.getStatIntervalMinutes());
        _globalCacheConfig.setAreaInCacheName(props.isAreaInCacheName());
        _globalCacheConfig.setPenetrationProtect(props.isPenetrationProtect());
        _globalCacheConfig.setEnableMethodCache(props.isEnableMethodCache());
        _globalCacheConfig.setLocalCacheBuilders(autoConfigureBeans.getLocalCacheBuilders());
        _globalCacheConfig.setRemoteCacheBuilders(autoConfigureBeans.getRemoteCacheBuilders());
        return _globalCacheConfig;
    }
}
  1. 可以看到通過@Import注解,初始化構(gòu)造器的那些類會被加入到Spring容器,加上@Condotional注解,只有我們配置過的緩存類型的構(gòu)造器才會被加入,然后保存至AutoConfigureBeans對象中
  2. 注意到這里我們注入的是SpringConfigProvider對象,加上@ConditionalOnMissingBean注解,無法再次注冊該對象至Spring容器,相比ConfigProvider對象,它的區(qū)別是設(shè)置了EncoderParser為DefaultSpringEncoderParser,設(shè)置了KeyConvertorParser為DefaultSpringKeyConvertorParser,目的是支持兩個解析器能夠解析自定義bean
  3. BeanDependencyManager中可以看到它是一個BeanFactoryPostProcessor,用于BeanFactory容器初始后執(zhí)行操作,目的是往JetCacheAutoConfiguration的BeanDefinition的依賴中添加幾個AbstractCacheAutoInit類型的beanName,保證幾個CacheBuilder構(gòu)造器已經(jīng)初始化
  4. globalCacheConfig方法中設(shè)置全局的相關(guān)配置并添加已經(jīng)初始化的CacheBuilder構(gòu)造器,然后返回GlobalCacheConfig讓Spring容器管理,這樣一來就完成了JetCache的解析配置并初始化的功能

CacheBuilder構(gòu)造器

構(gòu)造器的作用就是根據(jù)配置構(gòu)建一個對應(yīng)類型的緩存實例

CacheBuilder的子類結(jié)構(gòu)如下:

根據(jù)類名就可以知道其作用

CacheBuilder接口只定義了一個buildCache()方法,用于構(gòu)建緩存實例,交由不同的實現(xiàn)類

AbstractCacheBuilder抽象類實現(xiàn)了buildCache()方法,主要代碼如下:

public abstract class AbstractCacheBuilder<T extends AbstractCacheBuilder<T>> implements CacheBuilder, Cloneable {
    /**
     * 該緩存實例的配置
     */
    protected CacheConfig config;
    /**
     * 創(chuàng)建緩存實例函數(shù)
     */
    private Function<CacheConfig, Cache> buildFunc;
    public abstract CacheConfig getConfig();
    protected T self() {
        return (T) this;
    }
    public T buildFunc(Function<CacheConfig, Cache> buildFunc) {
        this.buildFunc = buildFunc;
        return self();
    }
    protected void beforeBuild() {
    }
    @Deprecated
    public final <K, V> Cache<K, V> build() {
        return buildCache();
    }
    @Override
    public final <K, V> Cache<K, V> buildCache() {
        if (buildFunc == null) {
            throw new CacheConfigException("no buildFunc");
        }
        beforeBuild();
        // 克隆一份配置信息,因為這里獲取到的是全局配置信息,以防后續(xù)被修改
        CacheConfig c = getConfig().clone();
        // 通過構(gòu)建函數(shù)創(chuàng)建一個緩存實例
        Cache<K, V> cache = buildFunc.apply(c);
        /*
         * 目前發(fā)現(xiàn) c.getLoader() 都是 null,后續(xù)都會把 cache 封裝成 CacheHandlerRefreshCache
         * TODO 疑問????
         */
        if (c.getLoader() != null) {
            if (c.getRefreshPolicy() == null) {
                cache = new LoadingCache<>(cache);
            } else {
                cache = new RefreshCache<>(cache);
            }
        }
        return cache;
    }
    @Override
    public Object clone() {
        AbstractCacheBuilder copy = null;
        try {
            copy = (AbstractCacheBuilder) super.clone();
            copy.config = getConfig().clone();
            return copy;
        } catch (CloneNotSupportedException e) {
            throw new CacheException(e);
        }
    }
}
  1. 實現(xiàn)了java.lang.Cloneable的clone方法,支持克隆該對象,因為每個緩存實例的配置不一定相同,這個構(gòu)造器中保存的是全局的一些配置,所以需要克隆一個構(gòu)造器出來為每個緩存實例設(shè)置其自己的配置而不影響這個最初始的構(gòu)造器
  2. 定義CacheConfig對象存放緩存配置,構(gòu)建緩存實例需要根據(jù)這些配置
  3. 定義的buildFunc函數(shù)用于構(gòu)建緩存實例,我們在初始化構(gòu)造器中可以看到,不同的構(gòu)造器設(shè)置的該函數(shù)都是new一個緩存實例并傳入配置信息,例如:
// 設(shè)置構(gòu)建 CaffeineCache 緩存實例的函數(shù)
buildFunc((c) -> new CaffeineCache((EmbeddedCacheConfig) c));
// 進(jìn)入CaffeineCache的構(gòu)造器你就可以看到會根據(jù)配置完成緩存實例的初始化

不同類型的構(gòu)造器區(qū)別在于CacheConfig類型不同,因為遠(yuǎn)程和本地的配置是有所區(qū)別的,還有就是設(shè)置的buildFunc函數(shù)不同,因為需要構(gòu)建不同的緩存實例,和上面的例子差不多,都是new一個緩存實例并傳入配置信息,這里就不一一講述了

AOP

主要查看jetcache-anno子模塊,提供AOP功能

啟用JetCache

JetCache可以通過@EnableMethodCache和@EnableCreateCacheAnnotation注解完成AOP的初始化工作,我們在Spring Boot工程中的啟動類上面添加這兩個注解即可啟用JetCache緩存。

@EnableMethodCache

@Target(ElementType.TYPE)
@Retention(RetentionPolicy.RUNTIME)
@Documented
@Import({CommonConfiguration.class, ConfigSelector.class})
public @interface EnableMethodCache {
    boolean proxyTargetClass() default false;
    AdviceMode mode() default AdviceMode.PROXY;
    int order() default Ordered.LOWEST_PRECEDENCE;
    String[] basePackages();
}

注解的相關(guān)配置在上面的'如何使用'中已經(jīng)講過了,這里我們關(guān)注@Import注解中的CommonConfigurationConfigSelector兩個類,將會被Spring容器管理

  1. com.alicp.jetcache.anno.config.CommonConfiguration上面有@Configuration注解,所以會被作為一個Spring Bean,里面定義了一個Bean為ConfigMap,所以這個Bean也會被Spring容器管理,com.alicp.jetcache.anno.support.ConfigMap中保存方法與緩存注解配置信息的映射關(guān)系
  2. com.alicp.jetcache.anno.config.ConfigSelector繼承了AdviceModeImportSelector,通過@Import注解他的selectImports方法會被調(diào)用,根據(jù)不同的AdviceMode導(dǎo)入不同的配置類,可以看到會返回一個JetCacheProxyConfiguration類名稱,那么它也會被注入

com.alicp.jetcache.anno.config.JetCacheProxyConfiguration是配置AOP的配置類,代碼如下:

@Configuration
public class JetCacheProxyConfiguration implements ImportAware, ApplicationContextAware {
    protected AnnotationAttributes enableMethodCache;
    private ApplicationContext applicationContext;
    @Override
    public void setImportMetadata(AnnotationMetadata importMetadata) {
        // 獲取 @EnableMethodCache 注解信息
        this.enableMethodCache = AnnotationAttributes.fromMap(
                importMetadata.getAnnotationAttributes(EnableMethodCache.class.getName(), false));
        if (this.enableMethodCache == null) {
            throw new IllegalArgumentException(
                    "@EnableMethodCache is not present on importing class " + importMetadata.getClassName());
        }
    }
    @Override
    public void setApplicationContext(ApplicationContext applicationContext) throws BeansException {
        this.applicationContext = applicationContext;
    }
    @Bean(name = CacheAdvisor.CACHE_ADVISOR_BEAN_NAME)
    @Role(BeanDefinition.ROLE_INFRASTRUCTURE)
    public CacheAdvisor jetcacheAdvisor(JetCacheInterceptor jetCacheInterceptor) {
        CacheAdvisor advisor = new CacheAdvisor();
        // bean的名稱:jetcache2.internalCacheAdvisor
        advisor.setAdviceBeanName(CacheAdvisor.CACHE_ADVISOR_BEAN_NAME);
        // 設(shè)置緩存攔截器為 JetCacheInterceptor
        advisor.setAdvice(jetCacheInterceptor);
        // 設(shè)置需要掃描的包
        advisor.setBasePackages(this.enableMethodCache.getStringArray("basePackages"));
        // 設(shè)置優(yōu)先級,默認(rèn) Integer 的最大值,最低優(yōu)先級
        advisor.setOrder(this.enableMethodCache.<Integer>getNumber("order"));
        return advisor;
    }
    /**
     * 注入一個 JetCacheInterceptor 攔截器,設(shè)置為框架內(nèi)部的角色
     *
     * @return JetCacheInterceptor
     */
    @Bean
    @Role(BeanDefinition.ROLE_INFRASTRUCTURE)
    public JetCacheInterceptor jetCacheInterceptor() {
        return new JetCacheInterceptor();
    }
}

因為JetCacheProxyConfiguration是通過@Import注解注入的并且實現(xiàn)了ImportAware接口,當(dāng)被注入Bean的時候會先調(diào)用其setImportMetadata方法(這里好像必須添加@Configuration注解,不然無法被Spring識別出來)獲取到@EnableMethodCache注解的元信息

其中定義了兩個Bean:

com.alicp.jetcache.anno.aop.JetCacheInterceptor:實現(xiàn)了aop中的MethodInterceptor方法攔截器,可用于aop攔截方法后執(zhí)行相關(guān)處理

com.alicp.jetcache.anno.aop.CacheAdvisor

1.繼承了org.springframework.aop.support.AbstractBeanFactoryPointcutAdvisor,將會作為一個AOP切面

2.設(shè)置了通知advice為JetCacheInterceptor,也就是說被攔截的方法都會進(jìn)入JetCacheInterceptor,JetCacheInterceptor就作為JetCache的入口了

3.根據(jù)注解設(shè)置了需要掃描的包路徑以及優(yōu)先級,默認(rèn)是最低優(yōu)先級

4.CacheAdvisor實現(xiàn)了org.springframework.aopPointcutAdvisor接口的getPointcut()方法,設(shè)置這個切面的切入點為com.alicp.jetcache.anno.aop.CachePointcut

5.從CachePointcut作為切入點

  • 實現(xiàn)了org.springframework.aop.ClassFilter接口,用于判斷哪些類需要被攔截
  • 實現(xiàn)了org.springframework.aop.MethodMatcher接口,用于判斷哪些類中的哪些方法會被攔截
  • 在判斷方法是否需要進(jìn)入JetCache的JetCacheInterceptor過程中,會解析方法上面的JetCache相關(guān)緩存注解,將配置信息封裝com.alicp.jetcache.anno.methodCacheInvokeConfig對象中,并把它保存至com.alicp.jetcache.anno.support.ConfigMap對象中

總結(jié):@EnableMethodCache注解主要就是生成一個AOP切面用于攔截帶有緩存注解的方法

@EnableCreateCacheAnnotation

@Target(ElementType.TYPE)
@Retention(RetentionPolicy.RUNTIME)
@Documented
@Import({CommonConfiguration.class, CreateCacheAnnotationBeanPostProcessor.class})
public @interface EnableCreateCacheAnnotation {
}

相比@EnableMethodCache注解,沒有相關(guān)屬性,同樣會導(dǎo)入CommonConfiguration類

不同的是將導(dǎo)入com.alicp.jetcache.anno.field.CreateCacheAnnotationBeanPostProcessor類,它繼承了org.springframework.beans.factory.annotation.AutowiredAnnotationBeanPostProcessor

作為一個BeanPostProcessor,用于在Spring初始化bean的時候做一些操作

從代碼中可以看到他的作用是:如果這個bean內(nèi)部存在添加了帶有@CreateCache注解的字段(沒有添加static),會將這個字段作為需要注入的對象,解析成 com.alicp.jetcache.anno.field.LazyInitCache緩存實例

LazyInitCache的主要代碼如下:

class LazyInitCache implements ProxyCache {
    /**
     * 是否初始化,用于懶加載
     */
    private boolean inited;
    /**
     * 緩存實例
     */
    private Cache cache;
    /**
     * 所處上下文
     */
    private ConfigurableListableBeanFactory beanFactory;
    /**
     * CreateCache 注解元信息
     */
    private CreateCache ann;
    /**
     * 字段
     */
    private Field field;
    /**
     * 刷新策略
     */
    private RefreshPolicy refreshPolicy;
    /**
     * 保護(hù)策略
     */
    private PenetrationProtectConfig protectConfig;
    public LazyInitCache(ConfigurableListableBeanFactory beanFactory, CreateCache ann, Field field) {
        this.beanFactory = beanFactory;
        this.ann = ann;
        this.field = field;
        CacheRefresh cr = field.getAnnotation(CacheRefresh.class);
        if (cr != null) {
            refreshPolicy = CacheConfigUtil.parseRefreshPolicy(cr);
        }
        CachePenetrationProtect penetrateProtect = field.getAnnotation(CachePenetrationProtect.class);
        if (penetrateProtect != null) {
            protectConfig = CacheConfigUtil.parsePenetrationProtectConfig(penetrateProtect);
        }
    }
    private void checkInit() {
        if (!inited) {
            synchronized (this) {
                if (!inited) {
                    init();
                    inited = true;
                }
            }
        }
    }
    /**
     * 獲取緩存實例,不存在則新建
     *
     * @return 緩存實例
     */
    @Override
    public Cache getTargetCache() {
        checkInit();
        return cache;
    }
    private void init() {
        if (inited) {
            throw new IllegalStateException();
        }
        // 從 spring 的容器中獲取全局緩存配置 GlobalCacheConfig 對象
        GlobalCacheConfig globalCacheConfig = beanFactory.getBean(GlobalCacheConfig.class);
        ConfigProvider configProvider = beanFactory.getBean(ConfigProvider.class);
        // 將注解信息封裝到 CachedAnnoConfig 對象中
        CachedAnnoConfig cac = new CachedAnnoConfig();
        cac.setArea(ann.area());
        cac.setName(ann.name());
        cac.setTimeUnit(ann.timeUnit());
        cac.setExpire(ann.expire());
        cac.setLocalExpire(ann.localExpire());
        cac.setCacheType(ann.cacheType());
        cac.setLocalLimit(ann.localLimit());
        cac.setSerialPolicy(ann.serialPolicy());
        cac.setKeyConvertor(ann.keyConvertor());
        cac.setRefreshPolicy(refreshPolicy);
        cac.setPenetrationProtectConfig(protectConfig);
        String cacheName = cac.getName();
        if (CacheConsts.isUndefined(cacheName)) {
            String[] hiddenPackages = globalCacheConfig.getHiddenPackages();
            CacheNameGenerator g = configProvider.createCacheNameGenerator(hiddenPackages);
            cacheName = g.generateCacheName(field);
        }
        // 從緩存實例管理器中獲取或者創(chuàng)建對應(yīng)的緩存實例
        cache = configProvider.getCacheContext().__createOrGetCache(cac, ann.area(), cacheName);
    }
}

1.可以看到通過@CreateCache創(chuàng)建的緩存實例也可以添加@CacheRefresh@CachePenetrationProtect注解

2.在AbstractCache抽象類的computeIfAbsentImpl方法中我們有講到,如果緩存實例是ProxyCache類型,則會先調(diào)用其getTargetCache()方法獲取緩存實例對象,所以LazyInitCache在第一次訪問的時候才進(jìn)行初始化,并根據(jù)緩存注解配置信息創(chuàng)建(存在則直接獲?。┮粋€緩存實例

總結(jié):@EnableCreateCacheAnnotation注解主要是支持@CreateCache能夠創(chuàng)建緩存實例

通過@EnableMethodCache@EnableCreateCacheAnnotation兩個注解,加上前面的解析配置過程,已經(jīng)完成的JetCache的解析與初始化過程,那么接下來我們來看看JetCache如何處理被攔截的方法。

攔截器

com.alicp.jetcache.anno.aop.CachePointcut切入點判斷方法是否需要攔截的邏輯:

1.方法所在的類對象是否匹配,除去以"java"、"org.springframework"開頭和包含"$$EnhancerBySpringCGLIB$$"、"$$FastClassBySpringCGLIB$$"的類,該類是否在我們通過@EnableMethodCache注解配置的basePackages中

2.從ConfigMap獲取方法對應(yīng)的CacheInvokeConfig對象,也就是獲取緩存配置信息

  • 如果是一個空對象,那么不需要被攔截,因為前面已經(jīng)判斷了所在的類是否需要被攔截,而這個類中并不是所有的方法都會添加緩存注解,所以這一類的方法會設(shè)置一個空對象(定義在CacheInvokeConfig內(nèi)部的一個靜態(tài)對象添加了final修飾),保存在ConfigMap中
  • 如果不為null,則需被攔截
  • 通過CacheConfigUtil解析這個方法的緩存注解,如果有@Cached注解或者@CacheInvalidate注解或者@CacheUpdate注解,先解析注解生成CacheInvokeConfig對象保存至ConfigMap中,然后該方法會被攔截,否在保存一個空對象不會被攔截

ConfigProvider

com.alicp.jetcache.anno.support.ConfigProvide是一個配置提供者對象,包含了JetCache的全局配置、緩存實例管理器、緩存value轉(zhuǎn)換器、緩存key轉(zhuǎn)換器、上下文和監(jiān)控指標(biāo)相關(guān)信息,主要代碼如下:

public class ConfigProvider extends AbstractLifecycle {
    /**
     * 緩存的全局配置
     */
    @Resource
    protected GlobalCacheConfig globalCacheConfig;
    /**
     * 緩存實例管理器
     */
    protected SimpleCacheManager cacheManager;
    /**
     * 根據(jù)不同類型生成緩存數(shù)據(jù)轉(zhuǎn)換函數(shù)的轉(zhuǎn)換器
     */
    protected EncoderParser encoderParser;
    /**
     * 根據(jù)不同類型生成緩存 Key 轉(zhuǎn)換函數(shù)的轉(zhuǎn)換器
     */
    protected KeyConvertorParser keyConvertorParser;
    /**
     * 緩存監(jiān)控指標(biāo)管理器
     */
    protected CacheMonitorManager cacheMonitorManager;
    /**
     * 打印緩存各項指標(biāo)的函數(shù)
     */
    private Consumer<StatInfo> metricsCallback = new StatInfoLogger(false);
    /**
     * 緩存更新事件(REMOVE OR PUT)消息接收者,無實現(xiàn)類
     * 我們可以自己實現(xiàn) CacheMessagePublisher 用于統(tǒng)計一些緩存的命中信息
     */
    private CacheMessagePublisher cacheMessagePublisher;
    /**
     * 默認(rèn)的緩存監(jiān)控指標(biāo)管理器
     */
    private CacheMonitorManager defaultCacheMonitorManager = new DefaultCacheMonitorManager();
    /**
     * 緩存上下文
     */
    private CacheContext cacheContext;
    public ConfigProvider() {
        cacheManager = SimpleCacheManager.defaultManager;
        encoderParser = new DefaultEncoderParser();
        keyConvertorParser = new DefaultKeyConvertorParser();
        cacheMonitorManager = defaultCacheMonitorManager;
    }
    @Override
    public void doInit() {
        // 啟動緩存指標(biāo)監(jiān)控器,周期性打印各項指標(biāo)
        initDefaultCacheMonitorInstaller();
        // 初始化緩存上下文
        cacheContext = newContext();
    }
    protected void initDefaultCacheMonitorInstaller() {
        if (cacheMonitorManager == defaultCacheMonitorManager) {
            DefaultCacheMonitorManager installer = (DefaultCacheMonitorManager) cacheMonitorManager;
            installer.setGlobalCacheConfig(globalCacheConfig);
            installer.setMetricsCallback(metricsCallback);
            if (cacheMessagePublisher != null) {
                installer.setCacheMessagePublisher(cacheMessagePublisher);
            }
            // 啟動緩存指標(biāo)監(jiān)控器
            installer.init();
        }
    }
    @Override
    public void doShutdown() {
        shutdownDefaultCacheMonitorInstaller();
        cacheManager.rebuild();
    }
    protected void shutdownDefaultCacheMonitorInstaller() {
        if (cacheMonitorManager == defaultCacheMonitorManager) {
            ((DefaultCacheMonitorManager) cacheMonitorManager).shutdown();
        }
    }
    /**
     * 根據(jù)編碼類型通過緩存value轉(zhuǎn)換器生成編碼函數(shù)
     *
     * @param valueEncoder 編碼類型
     * @return 編碼函數(shù)
     */
    public Function<Object, byte[]> parseValueEncoder(String valueEncoder) {
        return encoderParser.parseEncoder(valueEncoder);
    }
    /**
     * 根據(jù)解碼類型通過緩存value轉(zhuǎn)換器生成解碼函數(shù)
     *
     * @param valueDecoder 解碼類型
     * @return 解碼函數(shù)
     */
    public Function<byte[], Object> parseValueDecoder(String valueDecoder) {
        return encoderParser.parseDecoder(valueDecoder);
    }
    /**
     * 根據(jù)轉(zhuǎn)換類型通過緩存key轉(zhuǎn)換器生成轉(zhuǎn)換函數(shù)
     *
     * @param convertor 轉(zhuǎn)換類型
     * @return 轉(zhuǎn)換函數(shù)
     */
    public Function<Object, Object> parseKeyConvertor(String convertor) {
        return keyConvertorParser.parseKeyConvertor(convertor);
    }
    public CacheNameGenerator createCacheNameGenerator(String[] hiddenPackages) {
        return new DefaultCacheNameGenerator(hiddenPackages);
    }
    protected CacheContext newContext() {
        return new CacheContext(this, globalCacheConfig);
    }
}

繼承了com.alicp.jetcache.anno.support.AbstractLifecycle,查看其代碼可以看到有兩個方法,分別為init()初始化方法和shutdown()銷毀方法,因為分別添加了@PostConstruct注解和@PreDestroy注解,所以在Spring初始化時會調(diào)用init(),在Spring容器銷毀時會調(diào)用shutdown()方法,內(nèi)部分別調(diào)用doInit()和doShutdown(),這兩個方法交由子類實現(xiàn)

在doInit()方法中先啟動緩存指標(biāo)監(jiān)控器,用于周期性打印各項緩存指標(biāo),然后初始化CacheContext緩存上下文,SpringConfigProvider返回的是SpringConfigContext

在doShutdown()方法中關(guān)閉緩存指標(biāo)監(jiān)控器,清除緩存實例

CacheContext

com.alicp.jetcache.anno.support.CacheContext緩存上下文主要為每一個被攔截的請求創(chuàng)建緩存上下文,構(gòu)建對應(yīng)的緩存實例,主要代碼如下:

public class CacheContext {
    private static Logger logger = LoggerFactory.getLogger(CacheContext.class);
    private static ThreadLocal<CacheThreadLocal> cacheThreadLocal = new ThreadLocal<CacheThreadLocal>() {
        @Override
        protected CacheThreadLocal initialValue() {
            return new CacheThreadLocal();
        }
    };
    /**
     * JetCache 緩存的管理器(包含很多信息)
     */
    private ConfigProvider configProvider;
    /**
     * 緩存的全局配置
     */
    private GlobalCacheConfig globalCacheConfig;
    /**
     * 緩存實例管理器
     */
    protected SimpleCacheManager cacheManager;
    public CacheContext(ConfigProvider configProvider, GlobalCacheConfig globalCacheConfig) {
        this.globalCacheConfig = globalCacheConfig;
        this.configProvider = configProvider;
        cacheManager = configProvider.getCacheManager();
    }
    public CacheInvokeContext createCacheInvokeContext(ConfigMap configMap) {
    	// 創(chuàng)建一個本次調(diào)用的上下文
        CacheInvokeContext c = newCacheInvokeContext();
        // 添加一個函數(shù),后續(xù)用于獲取緩存實例
        // 根據(jù)注解配置信息獲取緩存實例對象,不存在則創(chuàng)建并設(shè)置到緩存注解配置類中
        c.setCacheFunction((invokeContext, cacheAnnoConfig) -> {
            Cache cache = cacheAnnoConfig.getCache();
            if (cache == null) {
                if (cacheAnnoConfig instanceof CachedAnnoConfig) { // 緩存注解
                    // 根據(jù)配置創(chuàng)建一個緩存實例對象,通過 CacheBuilder
                    cache = createCacheByCachedConfig((CachedAnnoConfig) cacheAnnoConfig, invokeContext);
                } else if ((cacheAnnoConfig instanceof CacheInvalidateAnnoConfig) || (cacheAnnoConfig instanceof CacheUpdateAnnoConfig)) { // 更新/使失效緩存注解
                    CacheInvokeConfig cacheDefineConfig = configMap.getByCacheName(cacheAnnoConfig.getArea(), cacheAnnoConfig.getName());
                    if (cacheDefineConfig == null) {
                        String message = "can't find @Cached definition with area=" + cacheAnnoConfig.getArea()
                                + " name=" + cacheAnnoConfig.getName() +
                                ", specified in " + cacheAnnoConfig.getDefineMethod();
                        CacheConfigException e = new CacheConfigException(message);
                        logger.error("Cache operation aborted because can't find @Cached definition", e);
                        return null;
                    }
                    cache = createCacheByCachedConfig(cacheDefineConfig.getCachedAnnoConfig(), invokeContext);
                }
                cacheAnnoConfig.setCache(cache);
            }
            return cache;
        });
        return c;
    }
    private Cache createCacheByCachedConfig(CachedAnnoConfig ac, CacheInvokeContext invokeContext) {
    	// 緩存區(qū)域
        String area = ac.getArea();
        // 緩存實例名稱
        String cacheName = ac.getName();
        if (CacheConsts.isUndefined(cacheName)) { // 沒有定義緩存實例名稱
        	// 生成緩存實例名稱:類名+方法名+(參數(shù)類型)
            cacheName = configProvider.createCacheNameGenerator(invokeContext.getHiddenPackages())
                    .generateCacheName(invokeContext.getMethod(), invokeContext.getTargetObject());
        }
        // 創(chuàng)建緩存實例對象
        Cache cache = __createOrGetCache(ac, area, cacheName);
        return cache;
    }
    @Deprecated
    public <K, V> Cache<K, V> getCache(String cacheName) {
        return getCache(CacheConsts.DEFAULT_AREA, cacheName);
    }
    @Deprecated
    public <K, V> Cache<K, V> getCache(String area, String cacheName) {
        Cache cache = cacheManager.getCacheWithoutCreate(area, cacheName);
        return cache;
    }
    public Cache __createOrGetCache(CachedAnnoConfig cachedAnnoConfig, String area, String cacheName) {
    	// 緩存名稱拼接
        String fullCacheName = area + "_" + cacheName;
        // 從緩存實例管理器中根據(jù)緩存區(qū)域和緩存實例名稱獲取緩存實例
        Cache cache = cacheManager.getCacheWithoutCreate(area, cacheName);
        if (cache == null) {
            synchronized (this) { // 加鎖
                // 再次確認(rèn)
                cache = cacheManager.getCacheWithoutCreate(area, cacheName);
                if (cache == null) {
                    /*
                     * 緩存區(qū)域的名稱是否作為緩存 key 名稱前綴,默認(rèn)為 true ,我一般設(shè)置為 false
                     */
                    if (globalCacheConfig.isAreaInCacheName()) {
                        // for compatible reason, if we use default configuration, the prefix should same to that version <=2.4.3
                        cache = buildCache(cachedAnnoConfig, area, fullCacheName);
                    } else {
                        // 構(gòu)建一個緩存實例
                        cache = buildCache(cachedAnnoConfig, area, cacheName);
                    }
                    cacheManager.putCache(area, cacheName, cache);
                }
            }
        }
        return cache;
    }
    protected Cache buildCache(CachedAnnoConfig cachedAnnoConfig, String area, String cacheName) {
        Cache cache;
        if (cachedAnnoConfig.getCacheType() == CacheType.LOCAL) { // 本地緩存
            cache = buildLocal(cachedAnnoConfig, area);
        } else if (cachedAnnoConfig.getCacheType() == CacheType.REMOTE) { // 遠(yuǎn)程緩存
            cache = buildRemote(cachedAnnoConfig, area, cacheName);
        } else { // 兩級緩存
        	// 構(gòu)建本地緩存實例
            Cache local = buildLocal(cachedAnnoConfig, area);
            // 構(gòu)建遠(yuǎn)程緩存實例
            Cache remote = buildRemote(cachedAnnoConfig, area, cacheName);
            // 兩級緩存時是否單獨設(shè)置了本地緩存失效時間 localExpire
            boolean useExpireOfSubCache = cachedAnnoConfig.getLocalExpire() > 0;
            // 創(chuàng)建一個兩級緩存CacheBuilder
            cache = MultiLevelCacheBuilder.createMultiLevelCacheBuilder()
                    .expireAfterWrite(remote.config().getExpireAfterWriteInMillis(), TimeUnit.MILLISECONDS)
                    .addCache(local, remote)
                    .useExpireOfSubCache(useExpireOfSubCache)
                    .cacheNullValue(cachedAnnoConfig.isCacheNullValue())
                    .buildCache();
        }
        // 設(shè)置緩存刷新策略
        cache.config().setRefreshPolicy(cachedAnnoConfig.getRefreshPolicy());
        // 將 cache 封裝成 CacheHandlerRefreshCache,也就是 RefreshCache 類型
        // 后續(xù)添加刷新任務(wù)時會判斷是否為 RefreshCache 類型,然后決定是否執(zhí)行 addOrUpdateRefreshTask 方法,添加刷新任務(wù),沒有刷新策略不會添加
        cache = new CacheHandler.CacheHandlerRefreshCache(cache);
        // 設(shè)置緩存未命中時,JVM是否只允許一個線程執(zhí)行方法,其他線程等待,全局配置默認(rèn)為false
        cache.config().setCachePenetrationProtect(globalCacheConfig.isPenetrationProtect());
        PenetrationProtectConfig protectConfig = cachedAnnoConfig.getPenetrationProtectConfig();
        if (protectConfig != null) { // 方法的@CachePenetrationProtect注解
            cache.config().setCachePenetrationProtect(protectConfig.isPenetrationProtect());
            cache.config().setPenetrationProtectTimeout(protectConfig.getPenetrationProtectTimeout());
        }
        if (configProvider.getCacheMonitorManager() != null) {
        	// 添加監(jiān)控統(tǒng)計配置
            configProvider.getCacheMonitorManager().addMonitors(area, cacheName, cache);
        }
        return cache;
    }
    protected Cache buildRemote(CachedAnnoConfig cachedAnnoConfig, String area, String cacheName) {
        // 獲取緩存區(qū)域?qū)?yīng)的 CacheBuilder 構(gòu)造器
        ExternalCacheBuilder cacheBuilder = (ExternalCacheBuilder) globalCacheConfig.getRemoteCacheBuilders().get(area);
        if (cacheBuilder == null) {
            throw new CacheConfigException("no remote cache builder: " + area);
        }
        // 克隆一個 CacheBuilder 構(gòu)造器,因為不同緩存實例有不同的配置
        cacheBuilder = (ExternalCacheBuilder) cacheBuilder.clone();
        if (cachedAnnoConfig.getExpire() > 0 ) {
        	// 設(shè)置失效時間
            cacheBuilder.expireAfterWrite(cachedAnnoConfig.getExpire(), cachedAnnoConfig.getTimeUnit());
        }
        // 設(shè)置緩存 key 的前綴
        if (cacheBuilder.getConfig().getKeyPrefix() != null) {
            // 配置文件中配置了 prefix,則設(shè)置為 prefix+cacheName
            cacheBuilder.setKeyPrefix(cacheBuilder.getConfig().getKeyPrefix() + cacheName);
        } else { // 設(shè)置為 cacheName
            cacheBuilder.setKeyPrefix(cacheName);
        }
        if (!CacheConsts.isUndefined(cachedAnnoConfig.getKeyConvertor())) { // 如果注解中設(shè)置了Key的轉(zhuǎn)換方式則替換,否則還是使用全局的
        	// 設(shè)置 key 的轉(zhuǎn)換器,只支持 FASTJSON
            cacheBuilder.setKeyConvertor(configProvider.parseKeyConvertor(cachedAnnoConfig.getKeyConvertor()));
        }
        if (!CacheConsts.isUndefined(cachedAnnoConfig.getSerialPolicy())) {
        	// 緩存數(shù)據(jù)保存至遠(yuǎn)程需要進(jìn)行編碼和解碼,所以這里設(shè)置其編碼和解碼方式,KRYO 和 JAVA 可選擇
            cacheBuilder.setValueEncoder(configProvider.parseValueEncoder(cachedAnnoConfig.getSerialPolicy()));
            cacheBuilder.setValueDecoder(configProvider.parseValueDecoder(cachedAnnoConfig.getSerialPolicy()));
        }
        // 設(shè)置是否緩存 null 值
        cacheBuilder.setCacheNullValue(cachedAnnoConfig.isCacheNullValue());
        return cacheBuilder.buildCache();
    }
    protected Cache buildLocal(CachedAnnoConfig cachedAnnoConfig, String area) {
    	// 獲取緩存區(qū)域?qū)?yīng)的 CacheBuilder 構(gòu)造器
        EmbeddedCacheBuilder cacheBuilder = (EmbeddedCacheBuilder) globalCacheConfig.getLocalCacheBuilders().get(area);
        if (cacheBuilder == null) {
            throw new CacheConfigException("no local cache builder: " + area);
        }
        // 克隆一個 CacheBuilder 構(gòu)造器,因為不同緩存實例有不同的配置
        cacheBuilder = (EmbeddedCacheBuilder) cacheBuilder.clone();
        if (cachedAnnoConfig.getLocalLimit() != CacheConsts.UNDEFINED_INT) {
            // 本地緩存數(shù)量限制
            cacheBuilder.setLimit(cachedAnnoConfig.getLocalLimit());
        }
        if (cachedAnnoConfig.getCacheType() == CacheType.BOTH && cachedAnnoConfig.getLocalExpire() > 0) {
        	// 設(shè)置本地緩存失效時間,前提是多級緩存,一般和遠(yuǎn)程緩存保持一致不設(shè)置
            cacheBuilder.expireAfterWrite(cachedAnnoConfig.getLocalExpire(), cachedAnnoConfig.getTimeUnit());
        } else if (cachedAnnoConfig.getExpire() > 0) {
        	// 設(shè)置失效時間
            cacheBuilder.expireAfterWrite(cachedAnnoConfig.getExpire(), cachedAnnoConfig.getTimeUnit());
        }
        if (!CacheConsts.isUndefined(cachedAnnoConfig.getKeyConvertor())) {
            cacheBuilder.setKeyConvertor(configProvider.parseKeyConvertor(cachedAnnoConfig.getKeyConvertor()));
        }
        // 設(shè)置是否緩存 null 值
        cacheBuilder.setCacheNullValue(cachedAnnoConfig.isCacheNullValue());
        // 構(gòu)建一個緩存實例
        return cacheBuilder.buildCache();
    }
    protected CacheInvokeContext newCacheInvokeContext() {
        return new CacheInvokeContext();
    }
}

createCacheInvokeContext方法返回一個本次調(diào)用的上下文CacheInvokeContext,為這個上下文設(shè)置緩存函數(shù),用于獲取或者構(gòu)建緩存實例,這個函數(shù)在CacheHandler中會被調(diào)用,我們來看看這個函數(shù)的處理邏輯:有兩個入?yún)?,分別為本次調(diào)用的上下文和緩存注解的配置信息

首先從緩存注解的配置信息中獲取緩存實例,如果不為null則直接返回,否則調(diào)用createCacheByCachedConfig方法,根據(jù)配置通過CacheBuilder構(gòu)造器創(chuàng)建一個緩存實例對象

createCacheByCachedConfig方法:

1.如果沒有定義緩存實例名稱(@Cached注解中的name配置),則生成類名+方法名+(參數(shù)類型)作為緩存實例名稱

2.然后調(diào)用__createOrGetCache方法

__createOrGetCache方法:

1.通過緩存實例管理器SimpleCacheManager根據(jù)緩存區(qū)域area和緩存實例名稱cacheName獲取緩存實例對象,如果不為null則直接返回,判斷緩存實例對象是否為null為進(jìn)行兩次確認(rèn),第二次會給當(dāng)前CacheContext加鎖進(jìn)行判斷,避免線程不安全

2.緩存實例對象還是為null的話,先判斷緩存區(qū)域area是否添加至緩存實例名稱中,是的話"area_cacheName"為緩存實例名稱,然后調(diào)用buildCache方法創(chuàng)建一個緩存實例對象

buildCache方法:根據(jù)緩存實例類型構(gòu)建不同的緩存實例對象,處理邏輯如下:

CacheType為LOCAL則調(diào)用buildLocal方法:

  1. 1.1. 從GlobalCacheConfig全局配置的localCacheBuilders(保存本地緩存CacheBuilder構(gòu)造器的集合)中的獲取本地緩存該緩存區(qū)域的構(gòu)造器,在之前講到的'JetCacheAutoConfiguration自動配置'中有說到過,會將初始化好的構(gòu)造器從AutoConfigureBeans中添加至GlobalCacheConfig中
  2.   1.2. 克隆一個 CacheBuilder 構(gòu)造器,因為不同緩存實例有不同的配置
  3.   1.3. 將緩存注解的配置信息設(shè)置到構(gòu)造器中,有以下配置:

     - 如果配置了localLimit,則設(shè)置本地緩存最大數(shù)量limit的值

     - 如果CacheType為BOTH并且配置了localExpire(大于0),則設(shè)置有效時間expireAfterWrite的值為localExpire,否則如果配置的expire大于0,則設(shè)置其值為expire

     - 如果配置了keyConvertor,則根據(jù)該值生成一個轉(zhuǎn)換函數(shù),沒有配置的話在初始化構(gòu)造器的時候根據(jù)全局配置可能已經(jīng)生成了一個轉(zhuǎn)換函數(shù)(我一般在全局配置中設(shè)置)

     - 設(shè)置是否緩存null值

  1.  1.4. 通過調(diào)用構(gòu)造器的buildCache()方法構(gòu)建一個緩存實例對象,該方法在之前講到的'CacheBuilder構(gòu)造器'中有分析過

CacheType為REMOTE則調(diào)用buildRemote方法:

  1.   1.1. 從GlobalCacheConfig全局配置的remoteCacheBuilders(保存遠(yuǎn)程緩存CacheBuilder構(gòu)造器的集合)中的獲取遠(yuǎn)程緩存該緩存區(qū)域的構(gòu)造器
  2.   1.2. 克隆一個 CacheBuilder 構(gòu)造器,因為不同緩存實例有不同的配置
  3.   1.3. 將緩存注解的配置信息設(shè)置到構(gòu)造器中,有以下配置:

     - 如果配置了expire,則設(shè)置遠(yuǎn)程緩存有效時間expireAfterWrite的值

     - 如果全局設(shè)置遠(yuǎn)程緩存的緩存key的前綴keyPrefix,則設(shè)置緩存key的前綴為"keyPrefix+cacheName",否則我為"cacheName"

     - 如果配置了keyConvertor,則根據(jù)該值生成一個轉(zhuǎn)換函數(shù),沒有配置的話在初始化構(gòu)造器的時候根據(jù)全局配置可能已經(jīng)生成了一個轉(zhuǎn)換函數(shù)(我一般在全局配置中設(shè)置)

     - 如果設(shè)置了serialPolicy,則根據(jù)該值生成編碼和解碼函數(shù),沒有配置的話在初始化構(gòu)造器的時候根據(jù)全局配置可能已經(jīng)生成了編碼函數(shù)和解碼函數(shù)(我一般在全局配置中設(shè)置)

     - 設(shè)置是否緩存null值

  1.   1.4. 通過調(diào)用構(gòu)造器的buildCache()方法構(gòu)建一個緩存實例對象

CacheType為BOTH則調(diào)用buildLocal方法構(gòu)建本地緩存實例,調(diào)用buildRemote方法構(gòu)建遠(yuǎn)程緩存實例:

   1.1. 創(chuàng)建一個MultiLevelCacheBuilder構(gòu)造器

   1.2. 設(shè)置有效時間為遠(yuǎn)程緩存的有效時間、添加local和remote緩存實例、設(shè)置是否單獨配置了本地緩存的失效時間(是否有配置localExpire)、設(shè)置是否緩存null值

   1.3. 通過調(diào)用構(gòu)造器的buildCache()方法構(gòu)建一個緩存實例對象

2.設(shè)置刷新策略RefreshPolicy,沒有的話為null

3.將緩存實例對象封裝成CacheHandlerRefreshCache對象,用于后續(xù)的添加刷新任務(wù),在之前的'AbstractCache抽象類'有講到

4.設(shè)置是否開啟緩存未命中時加載方法的保護(hù)模式,全局默認(rèn)為false

5.將緩存實例添加至監(jiān)控管理器中

JetCacheInterceptor

被攔截后的處理在com.alicp.jetcache.anno.aop.JetCacheInterceptor中,代碼如下:

public class JetCacheInterceptor implements MethodInterceptor, ApplicationContextAware {
    private static final Logger logger = LoggerFactory.getLogger(JetCacheInterceptor.class);
    /**
     * 緩存實例注解信息
     */
    @Autowired
    private ConfigMap cacheConfigMap;
    /**
     * Spring 上下文
     */
    private ApplicationContext applicationContext;
    /**
     * 緩存的全局配置
     */
    private GlobalCacheConfig globalCacheConfig;
    /**
     * JetCache 緩存的管理器(包含很多信息)
     */
    ConfigProvider configProvider;
    @Override
    public void setApplicationContext(ApplicationContext applicationContext) throws BeansException {
        this.applicationContext = applicationContext;
    }
    @Override
    public Object invoke(final MethodInvocation invocation) throws Throwable {
        if (configProvider == null) {
            /**
             * 這里會獲取到 SpringConfigProvider 可查看 {@link com.alicp.jetcache.autoconfigure.JetCacheAutoConfiguration}
             */
            configProvider = applicationContext.getBean(ConfigProvider.class);
        }
        if (configProvider != null && globalCacheConfig == null) {
            globalCacheConfig = configProvider.getGlobalCacheConfig();
        }
        if (globalCacheConfig == null || !globalCacheConfig.isEnableMethodCache()) {
            return invocation.proceed();
        }
        // 獲取被攔截的方法
        Method method = invocation.getMethod();
        // 獲取被攔截的對象
        Object obj = invocation.getThis();
        CacheInvokeConfig cac = null;
        if (obj != null) {
        	// 獲取改方法的Key(方法所在類名+方法名+(參數(shù)類型)+方法返回類型+_被攔截的類名)
            String key = CachePointcut.getKey(method, obj.getClass());
            // 獲取該方法的緩存注解信息,在 Pointcut 中已經(jīng)對注解進(jìn)行解析并放入 ConfigMap 中
            cac  = cacheConfigMap.getByMethodInfo(key);
        }
        if(logger.isTraceEnabled()){
            logger.trace("JetCacheInterceptor invoke. foundJetCacheConfig={}, method={}.{}(), targetClass={}",
                    cac != null,
                    method.getDeclaringClass().getName(),
                    method.getName(),
                    invocation.getThis() == null ? null : invocation.getThis().getClass().getName());
        }
        // 無緩存相關(guān)注解配置信息表明無須緩存,直接執(zhí)行該方法
        if (cac == null || cac == CacheInvokeConfig.getNoCacheInvokeConfigInstance()) {
            return invocation.proceed();
        }
        // 為本次調(diào)用創(chuàng)建一個上下文對象,包含對應(yīng)的緩存實例
        CacheInvokeContext context = configProvider.getCacheContext().createCacheInvokeContext(cacheConfigMap);
        context.setTargetObject(invocation.getThis());
        context.setInvoker(invocation::proceed);
        context.setMethod(method);
        context.setArgs(invocation.getArguments());
        context.setCacheInvokeConfig(cac);
        context.setHiddenPackages(globalCacheConfig.getHiddenPackages());
        // 繼續(xù)往下執(zhí)行
        return CacheHandler.invoke(context);
    }
    public void setCacheConfigMap(ConfigMap cacheConfigMap) {
        this.cacheConfigMap = cacheConfigMap;
    }
}

ConfigMap中獲取被攔截的方法對象的緩存配置信息,如果沒有則直接執(zhí)行該方法,否則繼續(xù)往下執(zhí)行

根據(jù)CacheContext對象(SpringCacheContext,因為在之前講到的'JetCacheAutoConfiguration自動配置'中有說到注入的是SpringConfigProvider對象,在其初始化方法中調(diào)用newContext()方法生成SpringCacheContext)調(diào)用其createCacheInvokeContext方法為本次調(diào)用創(chuàng)建一個上下文CacheInvokeContext,并設(shè)置獲取緩存實例函數(shù),具體實現(xiàn)邏輯查看上面講到的CacheContext

設(shè)置本次調(diào)用上下文的targetObject為被攔截對象,invoker為被攔截對象的調(diào)用器,method為被攔截方法,args為方法入?yún)?,cacheInvokeConfig為緩存配置信息,hiddenPackages為緩存實例名稱需要截斷的包名

通過CacheHandler的invoke方法繼續(xù)往下執(zhí)行

CacheHandler

com.alicp.jetcache.anno.method.CacheHandler用于JetCache處理被攔截的方法,部分代碼如下:

public class CacheHandler implements InvocationHandler {
	public static Object invoke(CacheInvokeContext context) throws Throwable {
		if (context.getCacheInvokeConfig().isEnableCacheContext()) {
			try {
				CacheContextSupport._enable();
				return doInvoke(context);
			} finally {
				CacheContextSupport._disable();
			}
		} else {
			return doInvoke(context);
		}
	}
	private static Object doInvoke(CacheInvokeContext context) throws Throwable {
		// 獲取緩存實例配置
		CacheInvokeConfig cic = context.getCacheInvokeConfig();
		// 獲取注解配置信息
		CachedAnnoConfig cachedConfig = cic.getCachedAnnoConfig();
		if (cachedConfig != null && (cachedConfig.isEnabled() || CacheContextSupport._isEnabled())) {
			// 經(jīng)過緩存中獲取結(jié)果
			return invokeWithCached(context);
		} else if (cic.getInvalidateAnnoConfigs() != null || cic.getUpdateAnnoConfig() != null) {
			// 根據(jù)結(jié)果刪除或者更新緩存
			return invokeWithInvalidateOrUpdate(context);
		} else {
			// 執(zhí)行該方法
			return invokeOrigin(context);
		}
	}
	private static Object invokeWithCached(CacheInvokeContext context) throws Throwable {
		// 獲取本地調(diào)用的上下文
		CacheInvokeConfig cic = context.getCacheInvokeConfig();
		// 獲取注解配置信息
		CachedAnnoConfig cac = cic.getCachedAnnoConfig();
		// 獲取緩存實例對象(不存在則會創(chuàng)建并設(shè)置到 cac 中)
		// 可在 JetCacheInterceptor 創(chuàng)建本次調(diào)用的上下文時,調(diào)用 createCacheInvokeContext(cacheConfigMap) 方法中查看詳情
		Cache cache = context.getCacheFunction().apply(context, cac);
		if (cache == null) {
			logger.error("no cache with name: " + context.getMethod());
			// 無緩存實例對象,執(zhí)行原有方法
			return invokeOrigin(context);
		}
		// 生成緩存 Key 對象(注解中沒有配置的話就是入?yún)?,沒有入?yún)t為 "_$JETCACHE_NULL_KEY$_" )
		Object key = ExpressionUtil.evalKey(context, cic.getCachedAnnoConfig());
		if (key == null) {
			 // 生成緩存 Key 失敗則執(zhí)行原方法,并記錄 CacheLoadEvent 事件
			return loadAndCount(context, cache, key);
		}
		/*
		 * 根據(jù)配置的 condition 來決定是否走緩存
		 * 緩存注解中沒有配置 condition 表示所有請求都走緩存
		 * 配置了 condition 表示滿足條件的才走緩存
		 */
		if (!ExpressionUtil.evalCondition(context, cic.getCachedAnnoConfig())) {
			// 不滿足 condition 則直接執(zhí)行原方法,并記錄 CacheLoadEvent 事件
			return loadAndCount(context, cache, key);
		}
		try {
			// 創(chuàng)建一個執(zhí)行原有方法的函數(shù)
			CacheLoader loader = new CacheLoader() {
				@Override
				public Object load(Object k) throws Throwable {
					Object result = invokeOrigin(context);
					context.setResult(result);
					return result;
				}
				@Override
				public boolean vetoCacheUpdate() {
					// 本次執(zhí)行原方法后是否需要更新緩存
					return !ExpressionUtil.evalPostCondition(context, cic.getCachedAnnoConfig());
				}
			};
			// 獲取結(jié)果
			Object result = cache.computeIfAbsent(key, loader);
			return result;
		} catch (CacheInvokeException e) {
			throw e.getCause();
		}
	}
	private static Object loadAndCount(CacheInvokeContext context, Cache cache, Object key) throws Throwable {
		long t = System.currentTimeMillis();
		Object v = null;
		boolean success = false;
		try {
			// 調(diào)用原有方法
			v = invokeOrigin(context);
			success = true;
		} finally {
			t = System.currentTimeMillis() - t;
			// 發(fā)送 CacheLoadEvent 事件
			CacheLoadEvent event = new CacheLoadEvent(cache, t, key, v, success);
			while (cache instanceof ProxyCache) {
				cache = ((ProxyCache) cache).getTargetCache();
			}
			if (cache instanceof AbstractCache) {
				((AbstractCache) cache).notify(event);
			}
		}
		return v;
	}
	private static Object invokeOrigin(CacheInvokeContext context) throws Throwable {
		// 執(zhí)行被攔截的方法
		return context.getInvoker().invoke();
	}
}

直接查看invokeWithCached方法:

  1. 獲取緩存注解信息
  2. 根據(jù)本地調(diào)用的上下文CacheInvokeContext獲取緩存實例對象(調(diào)用其cacheFunction函數(shù)),在CacheContext中有講到
  3. 如果緩存實例不存在則直接調(diào)用invokeOrigin方法,執(zhí)行被攔截的對象的調(diào)用器
  4. 根據(jù)本次調(diào)用的上下文CacheInvokeContext生成緩存key,根據(jù)配置的緩存key的SpEL表達(dá)式生成,如果沒有配置則返回入?yún)ο?,如果沒有對象則返回"_ $JETCACHE_NULL_KEY$_"
  5. 根據(jù)配置condition表達(dá)式判斷是否需要走緩存
  6. 創(chuàng)建一個CacheLoader對象,用于執(zhí)行被攔截的對象的調(diào)用器,也就是加載原有方法
  7. 調(diào)用緩存實例的computeIfAbsent(key, loader)方法獲取結(jié)果,這個方法的處理過程可查看'緩存API'這一小節(jié)

到此這篇關(guān)于JetCache 緩存框架的使用以及源碼分析的文章就介紹到這了,更多相關(guān)JetCache 緩存框架內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • 使用Java實現(xiàn)一個解析CURL腳本小工具

    使用Java實現(xiàn)一個解析CURL腳本小工具

    文章介紹了如何使用Java實現(xiàn)一個解析CURL腳本的工具,該工具可以將CURL腳本中的Header解析為KV Map結(jié)構(gòu),獲取URL路徑、請求類型,解析URL參數(shù)列表和Body請求體,感興趣的小伙伴跟著小編一起來看看吧
    2025-02-02
  • Java編寫擲骰子游戲

    Java編寫擲骰子游戲

    這篇文章主要介紹了Java編寫擲骰子游戲,需要的朋友可以參考下
    2015-11-11
  • Java中的四種引用類型之強(qiáng)引用、軟引用、弱引用和虛引用及用法詳解

    Java中的四種引用類型之強(qiáng)引用、軟引用、弱引用和虛引用及用法詳解

    Java的四種引用類型(強(qiáng)、軟、弱、虛)控制對象生命周期,分別在不同內(nèi)存壓力下回收,適用于緩存、資源清理等場景,配合ReferenceQueue實現(xiàn)高效內(nèi)存管理,本文將全面剖析這四種引用類型的概念、用法、實現(xiàn)原理以及實際應(yīng)用場景,感興趣的朋友一起看看吧
    2025-08-08
  • 使用Spring源碼報錯java:找不到類 InstrumentationSavingAgent的問題

    使用Spring源碼報錯java:找不到類 InstrumentationSavingAgent的問題

    這篇文章主要介紹了使用Spring源碼報錯java:找不到類 InstrumentationSavingAgent的問題,本文給大家分享解決方法,對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下
    2020-10-10
  • 利用Java Apache POI 生成Word文檔示例代碼

    利用Java Apache POI 生成Word文檔示例代碼

    本篇文章主要介紹了利用Java Apache POI 生成Word文檔示例代碼,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2017-05-05
  • 攔截器獲取request的值之后,Controller拿不到值的解決

    攔截器獲取request的值之后,Controller拿不到值的解決

    這篇文章主要介紹了攔截器獲取request的值之后,Controller拿不到值的解決方案,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教
    2021-10-10
  • spring事務(wù)的propagation傳播屬性示例詳解

    spring事務(wù)的propagation傳播屬性示例詳解

    這篇文章主要為大家介紹了spring事務(wù)的propagation傳播屬性示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2023-09-09
  • Java實現(xiàn)經(jīng)典游戲Flappy Bird的示例代碼

    Java實現(xiàn)經(jīng)典游戲Flappy Bird的示例代碼

    Flappy?Bird是13年紅極一時的小游戲,即摁上鍵控制鳥的位置穿過管道間的縫隙。本文將用Java語言實現(xiàn)這一經(jīng)典的游戲,需要的可以參考一下
    2022-02-02
  • Java中對象、集合與JSON的轉(zhuǎn)換操作指南

    Java中對象、集合與JSON的轉(zhuǎn)換操作指南

    這篇文章給大家介紹Java中對象、集合與JSON的互轉(zhuǎn)技術(shù)指南,詳細(xì)介紹了如何使用Jackson、Gson等庫進(jìn)行基本的轉(zhuǎn)換操作,以及如何處理復(fù)雜類型和日期類型的數(shù)據(jù)轉(zhuǎn)換,感興趣的朋友跟隨小編一起看看吧
    2025-09-09
  • Java中關(guān)于二叉樹的概念以及搜索二叉樹詳解

    Java中關(guān)于二叉樹的概念以及搜索二叉樹詳解

    二叉樹是一種很有用的非線性結(jié)構(gòu),日常的開發(fā)中常會用到,關(guān)于二叉樹的概念以及搜索二叉樹本文給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下
    2021-09-09

最新評論

亚洲2021av天堂| 青青青青青手机视频| 男人天堂色男人av| 91久久精品色伊人6882| 无码中文字幕波多野不卡| 女同久久精品秋霞网| 风流唐伯虎电视剧在线观看| 黄色在线观看免费观看在线| 伊人综合aⅴ在线网| 成年人啪啪视频在线观看| 福利片区一区二体验区| 偷拍自拍亚洲视频在线观看| 国产使劲操在线播放| 99热久久这里只有精品| 九一传媒制片厂视频在线免费观看| 国产揄拍高清国内精品对白| 另类av十亚洲av| 岛国黄色大片在线观看| 亚洲天天干 夜夜操| 青青青青视频在线播放| 精品国产在线手机在线| 精品区一区二区三区四区人妻 | 精品首页在线观看视频| 亚洲欧美清纯唯美另类| 亚洲精品亚洲人成在线导航| 日韩北条麻妃一区在线| 日视频免费在线观看| 精品人人人妻人人玩日产欧| 啪啪啪操人视频在线播放| 真实国产乱子伦一区二区| 国产白嫩美女一区二区| 岛国黄色大片在线观看 | 中文字幕日韩人妻在线三区| 亚洲av日韩av网站| 天天日天天操天天摸天天舔| 99亚洲美女一区二区三区| 五月天中文字幕内射| 午夜精彩视频免费一区| 国产va在线观看精品| 夜夜嗨av蜜臀av| 亚洲国产欧美国产综合在线| nagger可以指黑人吗| 国产精品国产三级麻豆| 日本免费一级黄色录像| 好男人视频在线免费观看网站| 国产综合视频在线看片| 青青在线视频性感少妇和隔壁黑丝| 久青青草视频手机在线免费观看 | 18禁网站一区二区三区四区| 男人插女人视频网站| 最新国产精品网址在线观看| free性日本少妇| 人妻丰满熟妇综合网| 中国无遮挡白丝袜二区精品| 精品91自产拍在线观看一区| 这里有精品成人国产99| 青青青视频自偷自拍38碰| 天天日天天干天天舔天天射| 人人妻人人人操人人人爽| 日本一二三中文字幕| 直接观看免费黄网站| 黑人大几巴狂插日本少妇| 香蕉aⅴ一区二区三区| 97超碰人人搞人人| 久久尻中国美女视频| 18禁网站一区二区三区四区| 国产V亚洲V天堂无码欠欠| 国产日韩精品免费在线| 黑人借宿ntr人妻的沦陷2| 日本xx片在线观看| 青青青青青青青在线播放视频| 北条麻妃高跟丝袜啪啪| 午夜精品一区二区三区城中村| 3344免费偷拍视频| 人妻无码中文字幕专区| 99人妻视频免费在线| av中文字幕电影在线看| 中文字幕在线一区精品| 亚洲国产精品免费在线观看| 国产美女一区在线观看| asmr福利视频在线观看| 2018在线福利视频| av在线免费中文字幕| 国产精品熟女久久久久浪潮| 国产第一美女一区二区三区四区| 骚逼被大屌狂草视频免费看| av日韩在线观看大全| 免费黄页网站4188| 99精品免费久久久久久久久a| 不卡一不卡二不卡三| 亚洲成人免费看电影| av手机免费在线观看高潮| 大学生A级毛片免费视频| 久久久久久99国产精品| jul—619中文字幕在线| 欧美一区二区三区激情啪啪啪| 18禁美女羞羞免费网站| 久青青草视频手机在线免费观看 | 美女操逼免费短视频下载链接 | 久久久久久九九99精品| 三级等保密码要求条款| 55夜色66夜色国产精品站| 青青青青青青青青青青草青青| 免费成人av中文字幕| 亚洲av琪琪男人的天堂| 精品高跟鞋丝袜一区二区| 亚洲视频在线观看高清| 中文 成人 在线 视频| 美女日逼视频免费观看| 久久尻中国美女视频| 黄片色呦呦视频免费看| 午夜影院在线观看视频羞羞羞| 青青草在观免费国产精品| 亚洲男人让女人爽的视频| 超碰97免费人妻麻豆| 天干天天天色天天日天天射| 91色网站免费在线观看| www日韩毛片av| 人妻少妇中文有码精品| 久久久人妻一区二区| 亚洲av色香蕉一区二区三区| 青春草视频在线免费播放| 亚洲av日韩精品久久久| 亚洲精品av在线观看| 亚洲成人激情视频免费观看了| 蜜桃视频入口久久久| 最新91九色国产在线观看| 男人操女人的逼免费视频| 青草亚洲视频在线观看| 好了av中文字幕在线| 91福利视频免费在线观看| 一区二区三区四区五区性感视频| 国产一区二区欧美三区| 亚洲精品乱码久久久久久密桃明| 又色又爽又黄的美女裸体| 天天干天天爱天天色| 中文字幕日韩精品就在这里| 99精品免费久久久久久久久a| 美女张开两腿让男人桶av| 免费无毒热热热热热热久| 亚洲 欧美 自拍 偷拍 在线| 久久丁香花五月天色婷婷| 国产视频在线视频播放| 蜜臀成人av在线播放| 亚洲精品 日韩电影| 91麻豆精品秘密入口在线观看| av乱码一区二区三区| 91九色porny蝌蚪国产成人| 伊人开心婷婷国产av| 岛国青草视频在线观看| 国产成人一区二区三区电影网站| 91亚洲精品干熟女蜜桃频道| 中文字幕在线观看极品视频| 99热久久这里只有精品| 91亚洲手机在线视频播放| 十八禁在线观看地址免费 | 熟女人妻在线中出观看完整版| 91‖亚洲‖国产熟女| 91综合久久亚洲综合| 成人福利视频免费在线| 亚洲欧美在线视频第一页| 亚洲av无女神免非久久| 9l人妻人人爽人人爽| 久久精品国产23696| 337p日本大胆欧美人| 日韩中文字幕福利av| 欧美成人小视频在线免费看| av中文在线天堂精品| 熟女少妇激情五十路| 在线观看成人国产电影| 又色又爽又黄又刺激av网站| 91综合久久亚洲综合| 在线国产精品一区二区三区| 国产女孩喷水在线观看| 国产精品国产三级麻豆| 中文字幕综合一区二区| 97资源人妻免费在线视频| 性生活第二下硬不起来| 绯色av蜜臀vs少妇| 亚洲一区二区三区精品视频在线| 色综合久久五月色婷婷综合| 在线国产日韩欧美视频| 中国熟女@视频91| 日本阿v视频在线免费观看| 欧美精产国品一二三产品价格| asmr福利视频在线观看| 国产福利在线视频一区| 五十路人妻熟女av一区二区| 亚洲男人的天堂a在线| 中文字幕AV在线免费看 | 在线亚洲天堂色播av电影| 三级av中文字幕在线观看| 欧美黑人性暴力猛交喷水| 色秀欧美视频第一页| 成人24小时免费视频| 51国产偷自视频在线播放| 亚洲精品中文字幕下载| 亚洲综合乱码一区二区| av俺也去在线播放| 91欧美在线免费观看| 激情国产小视频在线| 欧美特级特黄a大片免费| 男人的天堂av日韩亚洲| 亚洲 国产 成人 在线| 一区二区三区四区五区性感视频| 亚洲一区二区三区久久受| 欧洲日韩亚洲一区二区三区| 亚洲成人情色电影在线观看| 日本xx片在线观看| 中国熟女@视频91| 婷婷午夜国产精品久久久| 免费一级黄色av网站| 亚洲丝袜老师诱惑在线观看| 日韩近亲视频在线观看| 精品日产卡一卡二卡国色天香| 美女少妇亚洲精选av| 啊啊好慢点插舔我逼啊啊啊视频| 婷婷六月天中文字幕| 综合一区二区三区蜜臀| 午夜免费体验区在线观看| 在线制服丝袜中文字幕| 18禁精品网站久久| 色综合久久久久久久久中文| 亚洲视频在线视频看视频在线| av森泽佳奈在线观看| 天天日天天操天天摸天天舔| 亚洲精品在线资源站| 高潮视频在线快速观看国家快速| 极品丝袜一区二区三区| 亚洲Av无码国产综合色区| 特黄老太婆aa毛毛片| 天天日天天日天天擦| 人妻无码色噜噜狠狠狠狠色| 大黑人性xxxxbbbb| 亚洲国产香蕉视频在线播放| 久草视频 久草视频2| 狠狠躁夜夜躁人人爽天天久天啪| 可以在线观看的av中文字幕| 最新中文字幕乱码在线| 综合精品久久久久97| 青青青青操在线观看免费| 亚洲熟女综合色一区二区三区四区| 熟女人妻在线观看视频| 青青青激情在线观看视频| 天天草天天色天天干| 欧美日韩一区二区电影在线观看| 亚洲区美熟妇久久久久| 久久一区二区三区人妻欧美| 欧美日韩激情啪啪啪| 女同久久精品秋霞网| 大香蕉大香蕉在线有码 av| 综合页自拍视频在线播放| 大陆胖女人与丈夫操b国语高清| 午夜精品福利91av| 19一区二区三区在线播放| 成人亚洲国产综合精品| 唐人色亚洲av嫩草| 久草极品美女视频在线观看| 亚洲av自拍偷拍综合| 中文字幕高清资源站| 亚洲2021av天堂| 国产女人被做到高潮免费视频| 精品区一区二区三区四区人妻| 毛片一级完整版免费| 亚洲综合图片20p| 欧美偷拍亚洲一区二区| 日韩激情文学在线视频| 精彩视频99免费在线| 任你操任你干精品在线视频| 亚洲精品一区二区三区老狼| 久久久久久cao我的性感人妻| 五月激情婷婷久久综合网| 亚洲一区二区三区久久受| 中国产一级黄片免费视频播放| 99久久中文字幕一本人| 天天日天天干天天要| 中字幕人妻熟女人妻a62v网| 天天操天天插天天色| 天天操天天干天天日狠狠插| 成人蜜桃美臀九一一区二区三区| 韩国男女黄色在线观看| 38av一区二区三区| 国产亚洲欧美视频网站| 久久久久久久久久久久久97| 夜色17s精品人妻熟女| 亚洲中文精品字幕在线观看| 天天操天天操天天碰| 日韩精品中文字幕在线| 一色桃子人妻一区二区三区| 天天日夜夜操天天摸| 免费在线黄色观看网站| 亚洲天堂成人在线观看视频网站| 色综合色综合色综合色| 免费看国产av网站| 1区2区3区不卡视频| 日韩欧美国产精品91| 91九色porny国产蝌蚪视频| 精品黑人巨大在线一区| 最新中文字幕乱码在线| 国产成人一区二区三区电影网站| 人妻少妇中文有码精品| 在线观看免费视频网| 亚洲一区自拍高清免费视频| 伊拉克及约旦宣布关闭领空| 国产使劲操在线播放| 免费手机黄页网址大全| 国产亚洲视频在线观看| 最新日韩av传媒在线| 91在线视频在线精品3| 精产国品久久一二三产区区别| 在线免费观看99视频| 亚洲一级 片内射视正片| 天天干夜夜操啊啊啊| 色综合久久五月色婷婷综合| 国产男女视频在线播放| 在线免费视频 自拍| 晚上一个人看操B片| 美女张开两腿让男人桶av| 99精品一区二区三区的区| 成年人啪啪视频在线观看| 韩国黄色一级二级三级| 久久香蕉国产免费天天| 欧美激情电影免费在线| 亚洲国产成人av在线一区| 免费一级特黄特色大片在线观看 | 精品国产亚洲av一淫| 中文字幕人妻被公上司喝醉在线| 国产精品精品精品999| 国产成人小视频在线观看无遮挡| 香港三日本三韩国三欧美三级| yy6080国产在线视频| 人妻自拍视频中国大陆| 青青青青青操视频在线观看| 欧美激情电影免费在线| 超碰在线观看免费在线观看| 亚洲区欧美区另类最新章节| 久久久久久久久久性潮| 男人操女人的逼免费视频| 日韩美女综合中文字幕pp| 91试看福利一分钟| 日本少妇的秘密免费视频| 老鸭窝在线观看一区| 天天日天天玩天天摸| 久久机热/这里只有| aiss午夜免费视频| 在线免费观看视频一二区| 久久久久久性虐视频| 人妻少妇中文有码精品| 护士小嫩嫩又紧又爽20p| 97精品综合久久在线| 特大黑人巨大xxxx| 免费在线观看视频啪啪| 欧美女同性恋免费a| 亚洲av一妻不如妾| 亚洲综合色在线免费观看| 国产成人无码精品久久久电影| 一级A一级a爰片免费免会员| 自拍偷拍亚洲另类色图| 红杏久久av人妻一区| 在线国产精品一区二区三区| 亚洲一区二区三区uij| 91极品新人『兔兔』精品新作| 2012中文字幕在线高清| 欧美精品久久久久久影院| 久久久麻豆精亚洲av麻花| 家庭女教师中文字幕在线播放| 狠狠鲁狠狠操天天晚上干干| 插逼视频双插洞国产操逼插洞| 亚洲 色图 偷拍 欧美| 99久久超碰人妻国产| 午夜国产福利在线观看| 日韩av免费观看一区| 无码日韩人妻精品久久| 在线国产中文字幕视频| 2021国产一区二区| 亚洲欧美另类手机在线| 亚洲高清国产自产av| 午夜极品美女福利视频| 青青社区2国产视频| 超碰在线观看免费在线观看| 国产性生活中老年人视频网站| 99久久成人日韩欧美精品| 日韩av有码一区二区三区4| 男人的天堂av日韩亚洲| 亚洲一区自拍高清免费视频| 3344免费偷拍视频| 国产成人综合一区2区| 五十路人妻熟女av一区二区| 九九视频在线精品播放| 98精产国品一二三产区区别| 国产成人自拍视频在线免费观看| 91九色porny蝌蚪国产成人| 偷拍自拍 中文字幕| 一级a看免费观看网站| 日韩加勒比东京热二区| 欧美激情电影免费在线| 91九色porny蝌蚪国产成人| 91免费放福利在线观看| 福利在线视频网址导航| 极品性荡少妇一区二区色欲| 欧美男同性恋69视频| 日韩欧美制服诱惑一区在线| 国产又粗又猛又爽又黄的视频在线| 天天操夜夜骑日日摸| 香蕉aⅴ一区二区三区| 97黄网站在线观看| 人妻丝袜av在线播放网址| 国产又粗又硬又大视频| 欧美日韩中文字幕欧美| 毛片一级完整版免费| 在线观看视频 你懂的| 操操网操操伊剧情片中文字幕网| 天堂女人av一区二区| 久久久制服丝袜中文字幕| 国产日韩一区二区在线看 | 欧美精产国品一二三产品价格| 人妻久久无码中文成人| 可以免费看的www视频你懂的| 国产精品国色综合久久| 精品国产成人亚洲午夜| 三级av中文字幕在线观看| 日本韩国亚洲综合日韩欧美国产| 91人妻精品久久久久久久网站| 特大黑人巨大xxxx| 绝色少妇高潮3在线观看| 家庭女教师中文字幕在线播放| 天堂av在线官网中文| 国产精品自拍偷拍a| 精品日产卡一卡二卡国色天香| 亚洲中文字幕综合小综合| 亚洲精品乱码久久久久久密桃明| av资源中文字幕在线观看| 99热碰碰热精品a中文| 国产卡一卡二卡三乱码手机| 狠狠鲁狠狠操天天晚上干干| 成人网18免费视频版国产| 亚洲精品福利网站图片| 青青草在观免费国产精品| 国产成人小视频在线观看无遮挡| 亚洲国产精品美女在线观看| 91精品国产高清自在线看香蕉网 | 成人免费毛片aaaa| 国产日本精品久久久久久久| 午夜在线精品偷拍一区二| 成年人黄色片免费网站| 最近中文2019年在线看| 涩爱综合久久五月蜜臀| 日韩一区二区电国产精品| 最新激情中文字幕视频| 九九视频在线精品播放| 99精品国产aⅴ在线观看| 亚洲国产美女一区二区三区软件| 欧美精产国品一二三区| 护士小嫩嫩又紧又爽20p| 天天日天天干天天爱| 欧美精品免费aaaaaa| 亚洲2021av天堂| 日韩a级黄色小视频| 成熟丰满熟妇高潮xx×xx| 日本性感美女写真视频| 久久三久久三久久三久久| 国内资源最丰富的网站| 日本一二三区不卡无| 2022精品久久久久久中文字幕| 热99re69精品8在线播放| 看一级特黄a大片日本片黑人| 精品国产成人亚洲午夜| 丰满的子国产在线观看| 黑人巨大的吊bdsm| 爆乳骚货内射骚货内射在线| 护士特殊服务久久久久久久| 天天日天天干天天干天天日| 天天干天天搞天天摸| 国产福利小视频二区| 亚洲福利午夜久久久精品电影网| 欧美精品国产综合久久| 日韩欧美中文国产在线| 我想看操逼黄色大片| 国产黄色高清资源在线免费观看| 成人国产激情自拍三区| 99视频精品全部15| 女人精品内射国产99| 午夜精品一区二区三区更新| 国产老熟女伦老熟妇ⅹ| 天天干狠狠干天天操| 日韩午夜福利精品试看| 人妻素人精油按摩中出| 色av色婷婷人妻久久久精品高清 | av乱码一区二区三区| 在线新三级黄伊人网| 国产自拍在线观看成人| 中文字幕欧美日韩射射一| 91人妻精品久久久久久久网站| 日韩加勒比东京热二区| 护士小嫩嫩又紧又爽20p| 国产之丝袜脚在线一区二区三区 | 孕妇奶水仑乱A级毛片免费看| 自拍偷拍日韩欧美亚洲| 亚洲欧美成人综合视频| 欧美亚洲免费视频观看| 偷拍美女一区二区三区| 男人的天堂一区二区在线观看| 色综合天天综合网国产成人| 午夜久久香蕉电影网| 超碰97人人澡人人| 夜夜躁狠狠躁日日躁麻豆内射 | 日本高清撒尿pissing| 制服丝袜在线人妻中文字幕| 欧美激情电影免费在线| 夫妻在线观看视频91| 天堂av狠狠操蜜桃| 狠狠躁夜夜躁人人爽天天久天啪| 丝袜长腿第一页在线| 福利片区一区二体验区| 欧美日韩高清午夜蜜桃大香蕉| 国产成人精品亚洲男人的天堂| 蜜桃臀av蜜桃臀av| 搡老妇人老女人老熟女| 91亚洲手机在线视频播放| 亚洲推理片免费看网站| 香蕉av影视在线观看| 熟女国产一区亚洲中文字幕| 啊慢点鸡巴太大了啊舒服视频| 91高清成人在线视频| 真实国产乱子伦一区二区| 久久热这里这里只有精品| 天天射夜夜操狠狠干| 大香蕉大香蕉在线有码 av| 国产卡一卡二卡三乱码手机| 中文字幕一区二区自拍| 日本成人一区二区不卡免费在线| 日韩欧美一级aa大片| 男女之间激情网午夜在线| 中文乱理伦片在线观看| 在线观看欧美黄片一区二区三区| 人妻少妇性色欲欧美日韩| 婷婷五月亚洲综合在线| 伊人日日日草夜夜草| 少妇ww搡性bbb91| 在线观看国产网站资源| 亚洲av香蕉一区区二区三区犇| 欧美一区二区三区在线资源 | 亚洲在线观看中文字幕av| 人人超碰国字幕观看97| 精品久久久久久久久久久a√国产 日本女大学生的黄色小视频 | 黄色av网站免费在线| 亚洲在线观看中文字幕av| 亚洲超碰97人人做人人爱| 亚洲av黄色在线网站| 性感美女福利视频网站| 天干天天天色天天日天天射| nagger可以指黑人吗| 熟女人妻三十路四十路人妻斩| 岛国一区二区三区视频在线| 中文字幕日本人妻中出| 五月天久久激情视频| 婷婷综合亚洲爱久久| 93精品视频在线观看| 国产又大又黄免费观看| 精品亚洲国产中文自在线| 亚洲另类伦春色综合小| 久久人人做人人妻人人玩精品vr| 中文字幕人妻三级在线观看| 五十路人妻熟女av一区二区| 51精品视频免费在线观看| 青青草成人福利电影| 国内精品在线播放第一页| 国产一区二区神马久久| 啪啪啪啪啪啪啪啪啪啪黄色| 亚洲欧美国产综合777| 啪啪啪啪啪啪啪免费视频| 99精品国产自在现线观看| 天天射夜夜操狠狠干| 99的爱精品免费视频| 国产janese在线播放| 夜色福利视频在线观看| 91av精品视频在线| 一区二区三区的久久的蜜桃的视频 | 亚洲天天干 夜夜操| 少妇深喉口爆吞精韩国| 深田咏美亚洲一区二区| 黑人解禁人妻叶爱071| 蜜桃专区一区二区在线观看| 亚洲日产av一区二区在线| 最新日韩av传媒在线| 免费手机黄页网址大全| 男生舔女生逼逼视频| 天干天天天色天天日天天射| 蜜臀av久久久久久久| 日韩美在线观看视频黄| 亚洲老熟妇日本老妇| 国产在线免费观看成人| 非洲黑人一级特黄片| 亚洲国产欧美一区二区三区久久| 中文字幕一区二区三区人妻大片 | 做爰视频毛片下载蜜桃视频1| av天堂中文字幕最新| 亚洲公开视频在线观看| 亚洲最大黄了色网站| 日本美女成人在线视频| 久久久噜噜噜久久熟女av| 天天日天天日天天射天天干| 免费在线福利小视频| 日本人妻欲求不满中文字幕| 日本一二三中文字幕| 日韩欧美国产一区ab| 国产精品久久久黄网站| 性生活第二下硬不起来| 人人妻人人爽人人澡人人精品| 手机看片福利盒子日韩在线播放| 亚洲一级av大片免费观看| 日本韩国免费一区二区三区视频| 欧美怡红院视频在线观看| 99精品视频之69精品视频| 极品性荡少妇一区二区色欲| 哥哥姐姐综合激情小说| 国产午夜激情福利小视频在线| 中文人妻AV久久人妻水| 精品人人人妻人人玩日产欧| 亚洲超碰97人人做人人爱| 动漫av网站18禁| 亚洲第一黄色在线观看| 早川濑里奈av黑人番号| 黄色无码鸡吧操逼视频| 日本精品视频不卡一二三| 亚洲国产精品美女在线观看| 欧美麻豆av在线播放| 免费在线看的黄网站| 亚洲欧美成人综合视频| 日本免费午夜视频网站| 精内国产乱码久久久久久| 在线亚洲天堂色播av电影| 青青青青草手机在线视频免费看| 青青青视频自偷自拍38碰| 亚洲熟妇x久久av久久| 男人天堂最新地址av| 丰满的继坶3中文在线观看| 在线观看视频污一区| 欧美精品黑人性xxxx| 国产乱弄免费视频观看| 五十路熟女人妻一区二区9933| 亚洲精品乱码久久久久久密桃明| 自拍偷拍亚洲欧美在线视频| 青青在线视频性感少妇和隔壁黑丝| 不戴胸罩引我诱的隔壁的人妻| 国产一区二区视频观看| 人妻少妇av在线观看| 巨乳人妻日下部加奈被邻居中出 | 亚洲av人人澡人人爽人人爱| 免费黄页网站4188| 日本美女成人在线视频| 亚洲精品亚洲人成在线导航 | 中文字幕在线观看国产片| 日本阿v视频在线免费观看| 国产精品国产三级国产精东| av中文字幕电影在线看| 午夜久久香蕉电影网| 青青草国内在线视频精选| 888亚洲欧美国产va在线播放| 一区二区三区另类在线| 少妇高潮一区二区三区| 亚洲高清视频在线不卡| 喷水视频在线观看这里只有精品| 在线免费观看欧美小视频| 91欧美在线免费观看| 91麻豆精品传媒国产黄色片| 2022中文字幕在线| av在线观看网址av| 午夜激情高清在线观看| brazzers欧熟精品系列| 人妻少妇中文有码精品| 啪啪啪啪啪啪啪免费视频| 亚洲精品欧美日韩在线播放| 人妻自拍视频中国大陆| 欧美80老妇人性视频| 91麻豆精品秘密入口在线观看| 亚洲1069综合男同| 精品一区二区三四区| 亚洲免费成人a v| 顶级尤物粉嫩小尤物网站| 精品人妻伦一二三区久| 国产视频网站一区二区三区| 久久机热/这里只有| 欧美国产亚洲中英文字幕| 天天日天天天天天天天天天天| 蜜桃精品久久久一区二区| 人妻久久无码中文成人| av在线播放国产不卡| av天堂加勒比在线| 中文字幕日韩无敌亚洲精品| 亚洲欧美成人综合视频| 青青色国产视频在线| 亚洲精品乱码久久久久久密桃明| 日韩av有码中文字幕| aⅴ精产国品一二三产品| yellow在线播放av啊啊啊| 天天操天天插天天色| 久久久久久97三级| 9久在线视频只有精品| 国产性色生活片毛片春晓精品 | 蜜桃专区一区二区在线观看| 久久久久久9999久久久久| 国产欧美精品一区二区高清| 国产性生活中老年人视频网站| 色av色婷婷人妻久久久精品高清 | 欧美少妇性一区二区三区| 黑人3p华裔熟女普通话| 啊啊好大好爽啊啊操我啊啊视频| 玩弄人妻熟妇性色av少妇| 亚洲精品亚洲人成在线导航| xxx日本hd高清| 久久一区二区三区人妻欧美| av高潮迭起在线观看| 黑人进入丰满少妇视频| 噜噜色噜噜噜久色超碰| 亚洲精品精品国产综合| AV天堂一区二区免费试看| 欧美日韩国产一区二区三区三州| 精品av久久久久久久| 午夜福利人人妻人人澡人人爽| 真实国模和老外性视频| 日本女人一级免费片| 熟女国产一区亚洲中文字幕| 成年人午夜黄片视频资源| 国产日韩欧美视频在线导航| 亚洲激情偷拍一区二区| av在线免费中文字幕| 国产又粗又硬又猛的毛片视频| 国产伦精品一区二区三区竹菊| 亚洲精品久久综合久| 天天日天天天天天天天天天天 | 免费在线观看视频啪啪| 97人妻总资源视频| 欧美一区二区三区激情啪啪啪| okirakuhuhu在线观看| 日美女屁股黄邑视频| 成人av亚洲一区二区| ka0ri在线视频| 欧美女同性恋免费a| 在线免费观看日本伦理| av在线免费中文字幕| 日韩av熟妇在线观看| 青青青青草手机在线视频免费看 | 福利一二三在线视频观看| 日韩人妻在线视频免费| 99久久99一区二区三区| 成人av免费不卡在线观看| 香蕉片在线观看av| 又大又湿又爽又紧A视频| 亚洲特黄aaaa片| 晚上一个人看操B片| www久久久久久久久久久| 四川五十路熟女av| 老师啊太大了啊啊啊尻视频| 国产之丝袜脚在线一区二区三区 | 久久久精品999精品日本| 亚洲精品中文字幕下载| 国产a级毛久久久久精品| 人妻熟女中文字幕aⅴ在线| 日日摸夜夜添夜夜添毛片性色av| 日本阿v视频在线免费观看| 精品人妻伦一二三区久| 中国产一级黄片免费视频播放| 日韩一个色综合导航| 国产一区二区三免费视频| 岛国青草视频在线观看| 国产刺激激情美女网站| 午夜av一区二区三区| 香港一级特黄大片在线播放| 久久久久久久久久久久久97| 97国产在线av精品| 青青青青青青草国产| 天天操天天干天天日狠狠插| 在线观看av观看av| 岛国毛片视频免费在线观看| 久久久精品精品视频视频| 五十路丰满人妻熟妇| 一区二区三区在线视频福利| 国产日本欧美亚洲精品视| 97少妇精品在线观看| 亚洲Av无码国产综合色区| 深田咏美亚洲一区二区| 人妻丝袜榨强中文字幕| 国产chinesehd精品麻豆| 日韩av免费观看一区| 日韩视频一区二区免费观看| 亚洲男人让女人爽的视频| 一区二区三区欧美日韩高清播放| 国产一区二区三免费视频| 在线观看av2025| av无限看熟女人妻另类av| 亚洲精品福利网站图片| 亚洲国产免费av一区二区三区| 亚洲图库另类图片区| 黄色的网站在线免费看| 中文 成人 在线 视频| 韩国AV无码不卡在线播放| 久草福利电影在线观看| 韩国AV无码不卡在线播放| 国产日本精品久久久久久久| 久久www免费人成一看片| 欧洲国产成人精品91铁牛tv| 国产精彩对白一区二区三区 | 中文字幕综合一区二区| 天天日天天日天天擦| 绝顶痉挛大潮喷高潮无码| 亚洲欧美综合另类13p| av森泽佳奈在线观看| 美女福利写真在线观看视频| 人人人妻人人澡人人| 视频二区在线视频观看| 精品亚洲国产中文自在线| 中文字幕一区二区人妻电影冢本| 自拍偷拍日韩欧美亚洲| av大全在线播放免费| 成人蜜桃美臀九一一区二区三区| 亚洲欧美激情中文字幕| 国产亚洲国产av网站在线| 18禁污污污app下载| 视频久久久久久久人妻| 黑人大几巴狂插日本少妇| 在线网站你懂得老司机| eeuss鲁片一区二区三区| 日本美女成人在线视频| 色综合天天综合网国产成人| 午夜在线观看岛国av,com| 精品久久久久久久久久久久人妻 | 天天通天天透天天插| 国产麻豆乱子伦午夜视频观看| 亚洲无线观看国产高清在线| 93精品视频在线观看| 日本av在线一区二区三区| 亚洲天堂av最新网址| 人妻丝袜诱惑我操她视频| 搡老妇人老女人老熟女| 动漫黑丝美女的鸡巴| 最新黄色av网站在线观看| 偷拍自拍 中文字幕| 精品一线二线三线日本| 亚洲熟色妇av日韩熟色妇在线 | 大屁股熟女一区二区三区| 97精品综合久久在线| 2021天天色天天干| av中文字幕在线导航| 久久永久免费精品人妻专区| 可以在线观看的av中文字幕| 午夜精品九一唐人麻豆嫩草成人| av网站色偷偷婷婷网男人的天堂| 亚洲中文字幕校园春色| 午夜福利资源综合激情午夜福利资 | 亚洲欧美激情中文字幕| 青青青青青青青在线播放视频| 亚洲 清纯 国产com| 精品日产卡一卡二卡国色天香| 91国产资源在线视频| 亚洲1卡2卡三卡4卡在线观看 | 超碰在线观看免费在线观看 | 欧美亚洲自偷自拍 在线| 一区二区熟女人妻视频| 国产在线免费观看成人| 日本人妻少妇18—xx| 青青青国产免费视频| 欧美天堂av无线av欧美| 五月激情婷婷久久综合网| 啊啊好大好爽啊啊操我啊啊视频 | 国产在线一区二区三区麻酥酥| 都市家庭人妻激情自拍视频| 人人妻人人人操人人人爽| 成人乱码一区二区三区av| 国产亚州色婷婷久久99精品| 国产va在线观看精品| 岛国青草视频在线观看| huangse网站在线观看| 丰满的继坶3中文在线观看| 亚洲少妇高潮免费观看| 青青草精品在线视频观看| 少妇人妻久久久久视频黄片| av成人在线观看一区| 欧美亚洲国产成人免费在线| 五十路熟女人妻一区二| 91在线免费观看成人| 可以在线观看的av中文字幕| 啪啪啪18禁一区二区三区| 午夜福利资源综合激情午夜福利资| 黄色在线观看免费观看在线| 国产美女一区在线观看| 护士特殊服务久久久久久久| 天天色天天爱天天爽| 中文字幕在线观看极品视频| 午夜av一区二区三区| 国产丰满熟女成人视频| 馒头大胆亚洲一区二区| 国产日本欧美亚洲精品视| 任你操任你干精品在线视频| 国产日韩精品一二三区久久久| 在线观看的黄色免费网站| 北条麻妃高跟丝袜啪啪| 沙月文乃人妻侵犯中文字幕在线| 日本熟妇色熟妇在线观看| 欧美黑人性暴力猛交喷水| 超碰在线观看免费在线观看| 三上悠亚和黑人665番号| 欧美另类z0z变态| 国产亚洲精品品视频在线| 超碰中文字幕免费观看| 97国产在线av精品| 国产欧美日韩第三页| 人妻无码中文字幕专区| 五月天色婷婷在线观看视频免费| 亚洲av无码成人精品区辽| 粗大的内捧猛烈进出爽大牛汉子| 天天做天天干天天操天天射| 人人妻人人爽人人澡人人精品| 人妻3p真实偷拍一二区| 欧美视频不卡一区四区| 亚洲伊人久久精品影院一美女洗澡 | 啊啊好慢点插舔我逼啊啊啊视频| 黑人变态深video特大巨大| 精品美女在线观看视频在线观看| 自拍偷拍一区二区三区图片| 国产大鸡巴大鸡巴操小骚逼小骚逼| 97人人妻人人澡人人爽人人精品| 亚洲午夜在线视频福利| 午夜成午夜成年片在线观看 | 亚洲国产免费av一区二区三区 | 久久午夜夜伦痒痒想咳嗽P| 婷婷午夜国产精品久久久| 国产视频一区在线观看| 国产精品伦理片一区二区| 天天色天天操天天透| 一区二区视频在线观看视频在线| 色综合久久无码中文字幕波多| 亚洲国产精品中文字幕网站| 欧美成一区二区三区四区| 人妻无码色噜噜狠狠狠狠色| 国产女人被做到高潮免费视频| 一区二区三区在线视频福利| 久久精品国产23696| 久久久久久久精品成人热| 国产麻豆国语对白露脸剧情| 88成人免费av网站| 92福利视频午夜1000看| 毛茸茸的大外阴中国视频| 青青草亚洲国产精品视频| 92福利视频午夜1000看| 亚洲av人人澡人人爽人人爱| 91p0rny九色露脸熟女| 国产精品久久久黄网站| 久久久久久久99精品| 成年女人免费播放视频| jul—619中文字幕在线| 92福利视频午夜1000看| 天天日天天鲁天天操| 少妇深喉口爆吞精韩国| 成人蜜桃美臀九一一区二区三区| 一区二区三区美女毛片| 久久久久91精品推荐99| 久草福利电影在线观看| 91综合久久亚洲综合| 国产福利小视频二区| 亚洲精品国产久久久久久| 制丝袜业一区二区三区| 日日操综合成人av| 亚洲乱码中文字幕在线| 夫妻在线观看视频91| 不戴胸罩引我诱的隔壁的人妻| 自拍偷拍亚洲另类色图| 天天躁日日躁狠狠躁躁欧美av | 中文字幕人妻一区二区视频 | 久久久久久久久久久久久97| 成人av在线资源网站| 日韩a级精品一区二区| 丰满的子国产在线观看| 日韩美在线观看视频黄| av天堂加勒比在线| 99亚洲美女一区二区三区| 51国产成人精品视频| 中文字幕国产专区欧美激情| 亚洲精品精品国产综合| 欧美成人猛片aaaaaaa| 欧美一区二区中文字幕电影| mm131美女午夜爽爽爽| 亚洲一区久久免费视频| 色婷婷六月亚洲综合香蕉| 亚洲欧美一卡二卡三卡| 18禁美女羞羞免费网站| av中文字幕网址在线| 久久久人妻一区二区| 香蕉91一区二区三区| 香港一级特黄大片在线播放| 久青青草视频手机在线免费观看| 日视频免费在线观看| 午夜福利资源综合激情午夜福利资| 91免费观看国产免费| 特大黑人巨大xxxx| 国产实拍勾搭女技师av在线| 自拍偷拍亚洲另类色图| 欧美另类z0z变态| 婷婷午夜国产精品久久久| 日韩特级黄片高清在线看| 欧美精品久久久久久影院| 亚洲精品国产久久久久久| 婷婷色国产黑丝少妇勾搭AV| 黄色的网站在线免费看| 操的小逼流水的文章| 国产日韩精品免费在线| 日日夜夜精品一二三| 男人在床上插女人视频| 91快播视频在线观看| 国产真实乱子伦a视频| 精品成人午夜免费看| eeuss鲁片一区二区三区| 中文字幕国产专区欧美激情| 午夜在线观看一区视频| 超污视频在线观看污污污| www久久久久久久久久久| 最新97国产在线视频| 午夜av一区二区三区| 中文字幕中文字幕 亚洲国产| 成人区人妻精品一区二视频| 国产欧美日韩在线观看不卡| 18禁美女黄网站色大片下载| 国产一区二区视频观看| 日本一区精品视频在线观看| 久久亚洲天堂中文对白| 日本后入视频在线观看| 亚洲一级特黄特黄黄色录像片| 精品一区二区亚洲欧美| 一区二区三区四区中文| 精品av国产一区二区三区四区 | 爆乳骚货内射骚货内射在线| 精品一区二区三区欧美| 中文乱理伦片在线观看| 午夜精品久久久久久99热| 黄色中文字幕在线播放| 欧洲国产成人精品91铁牛tv| 天天躁夜夜躁日日躁a麻豆| 国产在线观看黄色视频| 黄色的网站在线免费看| 在线国产日韩欧美视频| 99热久久极品热亚洲| 绯色av蜜臀vs少妇| 日日爽天天干夜夜操| 国产欧美日韩在线观看不卡| 美女张开腿让男生操在线看| 美女张开两腿让男人桶av| 亚洲天天干 夜夜操| 国产在线一区二区三区麻酥酥 | 激情人妻校园春色亚洲欧美| 日韩二区视频一线天婷婷五| 亚欧在线视频你懂的| 亚洲av无码成人精品区辽| 国产福利小视频大全| 精品国产污污免费网站入口自| 天天操夜夜骑日日摸| 啊啊好大好爽啊啊操我啊啊视频 | 亚洲国产精品免费在线观看| 大骚逼91抽插出水视频| 韩国亚洲欧美超一级在线播放视频| 久久久久久99国产精品| aaa久久久久久久久| 综合精品久久久久97| 精品亚洲国产中文自在线| 少妇人妻100系列| 国产乱子伦精品视频潮优女| 中文字幕av男人天堂| 欧美成人精品欧美一级黄色| 好吊视频—区二区三区| 国产麻豆剧果冻传媒app| 中文字幕高清在线免费播放| 在线观看911精品国产| 欧美久久一区二区伊人| 久碰精品少妇中文字幕av| 91片黄在线观看喷潮| 国产精品久久9999| 国产精品系列在线观看一区二区| 2021久久免费视频| 熟女俱乐部一二三区| 家庭女教师中文字幕在线播放| 国产精品黄大片在线播放| 色婷婷精品大在线观看| 中文字幕日韩无敌亚洲精品| 青青草精品在线视频观看| 女生被男生插的视频网站| 亚洲专区激情在线观看视频| 亚洲综合图片20p| 亚洲第一黄色在线观看| 日韩av有码一区二区三区4 | 一区二区三区久久中文字幕| 久碰精品少妇中文字幕av| 好吊视频—区二区三区| 亚洲美女美妇久久字幕组| 亚洲综合色在线免费观看| 亚洲的电影一区二区三区| 久久久久91精品推荐99| 日本a级视频老女人| 午夜精品一区二区三区福利视频| 韩国男女黄色在线观看| 免费69视频在线看| v888av在线观看视频| 久久热这里这里只有精品| 天天日天天玩天天摸| 农村胖女人操逼视频| 护士特殊服务久久久久久久| 国产在线自在拍91国语自产精品| 91‖亚洲‖国产熟女| 日韩欧美在线观看不卡一区二区| 3344免费偷拍视频| 日本熟妇一区二区x x| 欧美日韩精品永久免费网址| 又粗又长 明星操逼小视频 | 精品欧美一区二区vr在线观看| 国产福利小视频二区| 91福利在线视频免费观看| 东京热男人的av天堂| 黄色成年网站午夜在线观看| 99热国产精品666| 大鸡吧插逼逼视频免费看 | 日本欧美视频在线观看三区| 国产精品一二三不卡带免费视频| 十八禁在线观看地址免费| 精品高潮呻吟久久av| 欧美伊人久久大香线蕉综合| 午夜福利资源综合激情午夜福利资 | 日韩亚洲高清在线观看| 亚洲另类综合一区小说| 日日夜夜大香蕉伊人| 国产综合精品久久久久蜜臀| av俺也去在线播放| 欧美亚洲自偷自拍 在线| 国产黄网站在线观看播放| 亚洲成人情色电影在线观看| 中国熟女一区二区性xx| 免费黄色成人午夜在线网站| 动漫美女的小穴视频| 亚洲免费va在线播放| 99精品视频在线观看免费播放| 青青操免费日综合视频观看| 国产janese在线播放| 久久香蕉国产免费天天| 久久精品亚洲国产av香蕉| 久久精品视频一区二区三区四区| 97国产在线观看高清| 成年人黄视频在线观看| 香蕉91一区二区三区| 国产成人精品一区在线观看 | 精品国产乱码一区二区三区乱| 伊人精品福利综合导航| 亚洲天堂精品久久久| 国产精品视频资源在线播放 | 黑人性生活视频免费看| 久久久久久九九99精品| 国产美女精品福利在线| 一区二区在线视频中文字幕| 久久精品美女免费视频| 日本免费视频午夜福利视频| 人妻丝袜精品中文字幕| av中文在线天堂精品| 久久免看30视频口爆视频| 人人妻人人爱人人草| 亚洲综合在线视频可播放| 青青青青视频在线播放| 亚洲午夜精品小视频| 天天干天天日天天干天天操| 天天日天天敢天天干| 2022国产综合在线干| 国产精品久久久久久久女人18| 免费在线观看污污视频网站| 日日夜夜精品一二三| 精品黑人一区二区三区久久国产| 五十路丰满人妻熟妇| 国产精品久久久黄网站| 人妻丝袜精品中文字幕| 亚洲成人av在线一区二区| 午夜的视频在线观看| 国产综合高清在线观看| 91精品国产观看免费| 中文字幕欧美日韩射射一| 欧美一区二区三区久久久aaa| 青青青青青青草国产| 91亚洲手机在线视频播放| 久草视频在线免播放| 特黄老太婆aa毛毛片| 97年大学生大白天操逼| 中文字幕 人妻精品| 日本xx片在线观看| 99久久激情婷婷综合五月天| 青青青青青免费视频| 亚洲视频在线观看高清| 亚洲免费av在线视频| 91欧美在线免费观看| 熟女妇女老妇一二三区| 日韩无码国产精品强奸乱伦| 日韩美女搞黄视频免费| 欧美精品 日韩国产| 91麻豆精品91久久久久同性| 中文字幕人妻被公上司喝醉在线 | 久草电影免费在线观看| 亚洲精品高清自拍av| 视频在线亚洲一区二区| 久久精品美女免费视频| 国产免费高清视频视频| 中文字幕av一区在线观看| 沙月文乃人妻侵犯中文字幕在线| 亚洲精品 日韩电影| 天天干天天日天天谢综合156| 人妻久久久精品69系列| 不卡精品视频在线观看| 国产成人精品一区在线观看| 2020av天堂网在线观看| 国产又色又刺激在线视频| 天天干天天操天天玩天天射| 操人妻嗷嗷叫视频一区二区| 天天爽夜夜爽人人爽QC| 国产女孩喷水在线观看| 97色视频在线观看| 二区中出在线观看老师| 国产一区二区在线欧美| 午夜精品久久久久久99热| 一区二区视频视频视频| 久久热这里这里只有精品| 美女视频福利免费看| 1区2区3区不卡视频| 一二三中文乱码亚洲乱码one| 啊啊好慢点插舔我逼啊啊啊视频| 18禁污污污app下载| 亚洲中文字幕乱码区| 欧美精品一二三视频| 人妻久久无码中文成人| 久久久久久久久久久久久97| 欧美日韩一级黄片免费观看| 中文字幕在线视频一区二区三区 | 国产亚洲欧美45p| 国产伊人免费在线播放| 欧美精产国品一二三产品价格| 中文字幕乱码av资源| 亚洲欧美国产麻豆综合| 91国内视频在线观看| 精品久久久久久久久久久99| 老司机福利精品视频在线| 国产成人精品福利短视频| 无码精品一区二区三区人| 大鸡吧插逼逼视频免费看| 1000部国产精品成人观看视频| 国产一区av澳门在线观看| 3344免费偷拍视频| 亚洲免费成人a v| 色哟哟在线网站入口| 中文字日产幕乱六区蜜桃| 久久精品久久精品亚洲人| 美女在线观看日本亚洲一区| 亚洲免费视频欧洲免费视频| 国产成人一区二区三区电影网站 | 亚洲精品午夜久久久久| 日韩av有码中文字幕| 亚洲精品无码久久久久不卡 | 日日夜夜大香蕉伊人| 亚洲福利天堂久久久久久| 亚洲精品ww久久久久久| 黄色资源视频网站日韩| 国产精选一区在线播放| 亚洲最大免费在线观看| 国产aⅴ一线在线观看| 97人妻色免费视频| 日本三极片中文字幕| 午夜精品一区二区三区4| 中国把吊插入阴蒂的视频| 成年人中文字幕在线观看| 亚洲精品欧美日韩在线播放| 欧亚日韩一区二区三区观看视频| 制丝袜业一区二区三区| 人人在线视频一区二区| 国产+亚洲+欧美+另类| 91免费观看在线网站| 亚洲精品无码久久久久不卡| 亚洲综合图片20p| 国产精品人妻熟女毛片av久| 中文字幕免费福利视频6| 国产又粗又黄又硬又爽| 亚洲午夜伦理视频在线| 大鸡巴操娇小玲珑的女孩逼| 激情五月婷婷综合色啪| 99热这里只有国产精品6| 97青青青手机在线视频| 天天干天天操天天爽天天摸| 国产黄色大片在线免费播放| 久久这里只有精彩视频免费| 专门看国产熟妇的网站| 性感美女高潮视频久久久| 日本一二三区不卡无| 美女日逼视频免费观看| 欧美日本国产自视大全| 天天色天天爱天天爽| 欧美视频一区免费在线| 国产揄拍高清国内精品对白| 亚洲va国产va欧美精品88| 亚洲午夜在线视频福利| 欧美精品欧美极品欧美视频| 国产女人露脸高潮对白视频| 免费男阳茎伸入女阳道视频| 亚洲 中文字幕在线 日韩| 最新欧美一二三视频| 家庭女教师中文字幕在线播放| 国语对白xxxx乱大交| 欧美日本在线视频一区| 在线新三级黄伊人网| 人妻av无码专区久久绿巨人| 年轻的人妻被夫上司侵犯| 青青尤物在线观看视频网站| 五十路熟女人妻一区二| 性欧美激情久久久久久久| okirakuhuhu在线观看| 日本午夜久久女同精女女| 综合页自拍视频在线播放| 天天操天天弄天天射| 一区国内二区日韩三区欧美| 蜜桃专区一区二区在线观看| 红桃av成人在线观看| 亚洲1卡2卡三卡4卡在线观看 | 91快播视频在线观看| 99热99re在线播放| 91精品国产观看免费| 自拍偷拍日韩欧美亚洲| 国产日本精品久久久久久久| 欧美男人大鸡吧插女人视频| 丝袜肉丝一区二区三区四区在线| 夜色17s精品人妻熟女| 国产刺激激情美女网站| 天天做天天干天天舔| 久草福利电影在线观看| av日韩在线免费播放| 成人影片高清在线观看| 国产九色91在线视频| 精品高跟鞋丝袜一区二区| 噜噜色噜噜噜久色超碰| 午夜精品一区二区三区福利视频| 午夜在线观看一区视频| 亚洲色偷偷综合亚洲AV伊人| 红杏久久av人妻一区| heyzo蜜桃熟女人妻| 亚洲老熟妇日本老妇| 99久久成人日韩欧美精品| 亚洲欧美综合在线探花| 抽查舔水白紧大视频| 粉嫩欧美美人妻小视频| 91天堂精品一区二区| 阴茎插到阴道里面的视频| 超级福利视频在线观看| 老司机深夜免费福利视频在线观看| 国产一区成人在线观看视频 | 久久永久免费精品人妻专区| 成年人免费看在线视频| 久久久久五月天丁香社区| 最新国产亚洲精品中文在线| 日本美女成人在线视频| 日本韩国在线观看一区二区| 免费看国产av网站| 日韩精品激情在线观看| 中国老熟女偷拍第一页| 国产极品美女久久久久久| 国产麻豆国语对白露脸剧情| 欧美激情精品在线观看| 精品suv一区二区69| 99国内小视频在现欢看| 强行扒开双腿猛烈进入免费版| 少妇高潮无套内谢麻豆| 国产精品午夜国产小视频| 欧美女同性恋免费a| 亚洲精品国产综合久久久久久久久| 人妻少妇亚洲精品中文字幕| 国产麻豆乱子伦午夜视频观看| 女同互舔一区二区三区| 手机看片福利盒子日韩在线播放| 在线观看免费av网址大全| 91精品综合久久久久3d动漫| 国产精品国产三级国产午| 在线免费视频 自拍| 在线免费观看亚洲精品电影| 精品日产卡一卡二卡国色天香| 人妻3p真实偷拍一二区| 人人超碰国字幕观看97| 国产又大又黄免费观看| 超级福利视频在线观看| 成人亚洲精品国产精品| 青青草人人妻人人妻| 福利片区一区二体验区| 午夜久久久久久久精品熟女| 日韩av中文在线免费观看| 青青青青青免费视频| 欧美日本aⅴ免费视频| 中文字幕最新久久久| 亚洲成人激情视频免费观看了| 天天干天天操天天爽天天摸| ka0ri在线视频| 国产自拍在线观看成人| 国产精品久久久黄网站| 亚洲激情av一区二区| 国产大学生援交正在播放| 高潮喷水在线视频观看| 无码精品一区二区三区人| 亚洲偷自拍高清视频| 农村胖女人操逼视频| 福利视频一区二区三区筱慧| 欧美亚洲中文字幕一区二区三区| 天堂av中文在线最新版| 日本熟妇丰满厨房55| 综合页自拍视频在线播放| 欧美麻豆av在线播放| 韩国黄色一级二级三级| 国产又粗又猛又爽又黄的视频在线| 欧美亚洲国产成人免费在线| 五十路息与子猛烈交尾视频| 国产第一美女一区二区三区四区| 人妻3p真实偷拍一二区| 九九热99视频在线观看97| 在线视频国产欧美日韩| 人人妻人人人操人人人爽| 啊啊啊想要被插进去视频| lutube在线成人免费看| 97人妻无码AV碰碰视频| 88成人免费av网站| 青草青永久在线视频18| 在线可以看的视频你懂的| 午夜dv内射一区区| 国产精品久久久久久久女人18| 桃色视频在线观看一区二区| 亚洲欧美日韩视频免费观看| 国产janese在线播放| 日本成人不卡一区二区| 日比视频老公慢点好舒服啊| 欧美精品中文字幕久久二区| 一个色综合男人天堂| 天天日天天透天天操| 中文字幕一区二区亚洲一区| 六月婷婷激情一区二区三区| 好了av中文字幕在线| av黄色成人在线观看| 久草极品美女视频在线观看| 一本一本久久a久久精品综合不卡| 中国黄片视频一区91| 伊人成人在线综合网| 久久久久久国产精品| 久久久久久久一区二区三| 国产麻豆剧传媒精品国产av蜜桃| 亚洲精品国品乱码久久久久| 国产91精品拍在线观看| 国产精品视频一区在线播放| 五十路av熟女松本翔子| 欧美一区二区三区激情啪啪啪| 欧美少妇性一区二区三区| 色在线观看视频免费的| 国产视频一区在线观看| 久久久久五月天丁香社区| 亚洲av可乐操首页| 日美女屁股黄邑视频| 色婷婷六月亚洲综合香蕉| 人妻熟女中文字幕aⅴ在线| 丝袜美腿视频诱惑亚洲无| 揄拍成人国产精品免费看视频 | 国际av大片在线免费观看| 国产欧美精品一区二区高清| 日日爽天天干夜夜操| 人妻熟女中文字幕aⅴ在线| 熟女俱乐部一二三区| 91桃色成人网络在线观看| 亚洲熟妇x久久av久久| 哥哥姐姐综合激情小说| 亚洲一区二区三区久久午夜 | 性生活第二下硬不起来| 大香蕉伊人国产在线| 98精产国品一二三产区区别| 亚洲精品中文字幕下载| 日韩美女福利视频网| 亚洲另类在线免费观看| 亚洲最大免费在线观看| 国产自拍在线观看成人| 亚洲欧美久久久久久久久| 亚洲一区av中文字幕在线观看| 亚洲伊人久久精品影院一美女洗澡| 免费在线看的黄网站| 日韩近亲视频在线观看| 1024久久国产精品| 粉嫩小穴流水视频在线观看| 中文字幕av一区在线观看| 国产日韩欧美美利坚蜜臀懂色| 啪啪啪啪啪啪啪啪啪啪黄色| 欧美成人精品欧美一级黄色| 中国熟女一区二区性xx| 精品一区二区三区欧美| 男女啪啪视频免费在线观看| 国产V亚洲V天堂无码欠欠| 国产午夜无码福利在线看| 成人av天堂丝袜在线观看| 久久精品在线观看一区二区 | 2022国产精品视频| 日韩美女搞黄视频免费| 在线观看免费视频网| av一区二区三区人妻| 99热色原网这里只有精品| 天天日天天爽天天干| 中文字幕一区二区三区蜜月| 男女第一次视频在线观看| av一本二本在线观看| 北条麻妃高跟丝袜啪啪| 天天日天天日天天射天天干| 成人av天堂丝袜在线观看| 亚洲自拍偷拍综合色| 98精产国品一二三产区区别| 男人天堂最新地址av| 97精品人妻一区二区三区精品| 日本高清在线不卡一区二区| 国产污污污污网站在线| av森泽佳奈在线观看| 亚洲激情,偷拍视频| 亚洲 图片 欧美 图片| 99精品国自产在线人| xxx日本hd高清| 91中文字幕免费在线观看| 久久精品亚洲国产av香蕉| 丰满少妇翘臀后进式| 天天艹天天干天天操| 日韩伦理短片在线观看| 青草青永久在线视频18| 国产va在线观看精品| 97人妻夜夜爽二区欧美极品| 粉嫩av蜜乳av蜜臀| 999久久久久999| 国产女孩喷水在线观看| 老司机福利精品视频在线| 91国产在线视频免费观看| 国产不卡av在线免费| av成人在线观看一区| 中国黄片视频一区91| 亚洲av日韩高清hd| 亚洲嫩模一区二区三区| 久久尻中国美女视频| 国产老熟女伦老熟妇ⅹ| 2022国产综合在线干| 成年人中文字幕在线观看| 97青青青手机在线视频| 骚货自慰被发现爆操| 丝袜美腿欧美另类 中文字幕| 蜜桃精品久久久一区二区| 粉嫩av蜜乳av蜜臀| 国产精品一二三不卡带免费视频 | 人妻丝袜精品中文字幕| 早川濑里奈av黑人番号| 懂色av之国产精品| 国产剧情演绎系列丝袜高跟| 国产一区二区三免费视频| 国产精品久久久久久久女人18| 天天日天天干天天插舔舔| 在线免费观看日本片| 午夜精品久久久久麻豆影视| 2020韩国午夜女主播在线| 天天摸天天日天天操| 亚洲成人情色电影在线观看| 100%美女蜜桃视频| 黑人3p华裔熟女普通话| 78色精品一区二区三区| 天天日天天透天天操| 一区二区三区四区视频在线播放| 57pao国产一区二区| 五月天中文字幕内射| 19一区二区三区在线播放| 中文字幕之无码色多多| 搞黄色在线免费观看| 中文字幕一区二区人妻电影冢本| 无码精品一区二区三区人| 福利一二三在线视频观看| 红杏久久av人妻一区| 久久久久久久一区二区三| 天天操天天射天天操天天天| 中文乱理伦片在线观看| 视频一区二区在线免费播放| 岛国免费大片在线观看| 日本性感美女写真视频| 国产亚洲天堂天天一区| 天天插天天色天天日| 涩爱综合久久五月蜜臀| av一本二本在线观看| 中文字幕 亚洲av| 2021年国产精品自拍| 天天日天天干天天舔天天射| 18禁免费av网站| 美女被肏内射视频网站| jiuse91九色视频| 亚洲狠狠婷婷综合久久app| 粉嫩av懂色av蜜臀av | 2020久久躁狠狠躁夜夜躁| 日韩av免费观看一区| 天天日天天舔天天射进去| 国产实拍勾搭女技师av在线| 人妻丝袜av在线播放网址| 91极品大一女神正在播放| 日本乱人一区二区三区| 91桃色成人网络在线观看| 人人爱人人妻人人澡39| 亚洲va国产va欧美va在线| 国产精品一二三不卡带免费视频 | 中文字幕—97超碰网| 老司机99精品视频在线观看| 91免费观看在线网站| 亚洲va欧美va人人爽3p| 好了av中文字幕在线| 久久三久久三久久三久久| 人人在线视频一区二区| 福利午夜视频在线观看| 亚洲美女自偷自拍11页| 2021久久免费视频| 中文 成人 在线 视频| 日韩精品一区二区三区在线播放| 福利在线视频网址导航| 91快播视频在线观看| 亚洲中文精品人人免费| 97精品视频在线观看| av一本二本在线观看| 亚洲综合一区成人在线| 亚洲另类综合一区小说| 伊人开心婷婷国产av| 这里有精品成人国产99| 视频一区二区综合精品| 午夜精彩视频免费一区| 免费一级特黄特色大片在线观看 | 啊啊好大好爽啊啊操我啊啊视频| 天天干天天日天天干天天操| 日本午夜福利免费视频| 中文字幕人妻熟女在线电影| 午夜dv内射一区区| 国产在线91观看免费观看| 中文字幕日韩精品就在这里| 国产普通话插插视频| 一级黄片久久久久久久久| 99一区二区在线观看| 5528327男人天堂| 中文字幕一区二区人妻电影冢本| 中国产一级黄片免费视频播放| 在线观看亚洲人成免费网址| 99精品国产aⅴ在线观看| 夜夜嗨av蜜臀av| 91亚洲精品干熟女蜜桃频道 | 五十路息与子猛烈交尾视频| 蜜桃臀av蜜桃臀av| 青青青青青青青青青青草青青| 天天干天天操天天玩天天射| 久精品人妻一区二区三区| 日比视频老公慢点好舒服啊| 午夜国产福利在线观看| 大白屁股精品视频国产| 青青草原网站在线观看| 欧美偷拍自拍色图片| 19一区二区三区在线播放| 久久www免费人成一看片| 国产超码片内射在线| 国产剧情演绎系列丝袜高跟| 91国内视频在线观看| 国产污污污污网站在线| 免费高清自慰一区二区三区网站| 99热这里只有精品中文| 中文字幕视频一区二区在线观看| 专门看国产熟妇的网站| 91传媒一区二区三区| 国产V亚洲V天堂无码欠欠| 插小穴高清无码中文字幕| 日韩一区二区电国产精品| 啊啊啊想要被插进去视频| 大鸡巴操娇小玲珑的女孩逼| 精品视频中文字幕在线播放 | 欧美精品一区二区三区xxxx| 天天摸天天亲天天舔天天操天天爽| 人妻少妇一区二区三区蜜桃| 91精品视频在线观看免费| 在线观看视频一区麻豆| 精品一区二区三区三区88| 六月婷婷激情一区二区三区| 视频 一区二区在线观看| 又粗又硬又猛又黄免费30| 国产精品成人xxxx| 激情内射在线免费观看| 国产成人精品av网站| 小泽玛利亚视频在线观看| 2o22av在线视频| 亚洲蜜臀av一区二区三区九色 | 亚洲女人的天堂av| 亚洲一区二区三区五区| 成人区人妻精品一区二视频| 888欧美视频在线| 黄色成人在线中文字幕| 国产麻豆乱子伦午夜视频观看| 成人av亚洲一区二区| 蜜桃色婷婷久久久福利在线| sspd152中文字幕在线| 亚洲成a人片777777| 大香蕉伊人中文字幕| 亚洲精品无码色午夜福利理论片| 天天操天天弄天天射| 无忧传媒在线观看视频| 五月婷婷在线观看视频免费| 婷婷五月亚洲综合在线| 午夜蜜桃一区二区三区| 黄色大片男人操女人逼| 日本在线不卡免费视频| avjpm亚洲伊人久久| 亚洲在线观看中文字幕av| 女同久久精品秋霞网| av久久精品北条麻妃av观看| av男人天堂狠狠干| 亚洲精品一线二线在线观看 | 日本一区二区三区免费小视频| 91免费观看国产免费| 男女第一次视频在线观看| 桃色视频在线观看一区二区| 欧美精品黑人性xxxx| 自拍偷区二区三区麻豆| okirakuhuhu在线观看| 亚洲一区二区三区av网站| 精品久久婷婷免费视频| 日噜噜噜夜夜噜噜噜天天噜噜噜| 1000部国产精品成人观看视频| 精品一区二区三区三区色爱| 黑人巨大精品欧美视频| 国产在线观看黄色视频| 传媒在线播放国产精品一区| 免费观看污视频网站| 日本一二三区不卡无| 狠狠操操操操操操操操操| 午夜激情久久不卡一区二区 | av手机在线观播放网站| 日韩精品中文字幕在线| 在线观看欧美黄片一区二区三区| 久草视频在线免播放| 亚洲一区二区三区五区| 国产精品成久久久久三级蜜臀av | 亚洲区欧美区另类最新章节| 亚洲欧美激情国产综合久久久| 亚洲欧美国产综合777| 爱爱免费在线观看视频| 成人福利视频免费在线| 中文亚洲欧美日韩无线码| 国产在线观看黄色视频| 黑人3p华裔熟女普通话| 不卡一不卡二不卡三| 国产亚洲天堂天天一区| 国产精品黄色的av| 亚洲狠狠婷婷综合久久app| 国产欧美日韩第三页| 日本少妇在线视频大香蕉在线观看| 黄色成年网站午夜在线观看| 不戴胸罩引我诱的隔壁的人妻| 夜夜骑夜夜操夜夜奸| 人妻自拍视频中国大陆| 97超碰最新免费在线观看| 大屁股熟女一区二区三区| 日韩欧美制服诱惑一区在线| 亚洲国产香蕉视频在线播放| 中文字幕之无码色多多| 五月天色婷婷在线观看视频免费| 在线视频精品你懂的| 91快播视频在线观看| 亚洲av成人免费网站| 日本美女性生活一级片| 中国熟女一区二区性xx| 成人性爱在线看四区| 亚洲熟妇x久久av久久| 丝袜肉丝一区二区三区四区在线| 欧美另类一区二区视频| 午夜91一区二区三区| 激情综合治理六月婷婷| 亚洲中文字幕国产日韩| 欧美性感尤物人妻在线免费看| 狠狠躁夜夜躁人人爽天天天天97| 黄片大全在线观看观看| 午夜av一区二区三区| 人人人妻人人澡人人| 91精品国产综合久久久蜜 | 成人蜜桃美臀九一一区二区三区| 天天操天天干天天艹| www日韩a级s片av| av天堂中文免费在线| 国产亚洲精品品视频在线| 91破解版永久免费| 狠狠操狠狠操免费视频| 日本欧美视频在线观看三区| 激情图片日韩欧美人妻| 日韩黄色片在线观看网站| 亚洲一区制服丝袜美腿| 欧美日韩一级黄片免费观看| 亚洲欧美一卡二卡三卡| 成年人该看的视频黄免费| 又色又爽又黄又刺激av网站| 精品高跟鞋丝袜一区二区| 北条麻妃高跟丝袜啪啪| 欧美一区二区三区在线资源 | 97人妻色免费视频| 人人爱人人妻人人澡39| 亚洲一区二区三区精品乱码| 亚洲超碰97人人做人人爱| 99热久久极品热亚洲| 成人资源在线观看免费官网| 亚洲av黄色在线网站| 亚洲乱码中文字幕在线| 免费观看理论片完整版| 在线国产中文字幕视频| 欧美中文字幕一区最新网址| 极品粉嫩小泬白浆20p主播 | 成人av中文字幕一区| 男女之间激情网午夜在线| 美女操逼免费短视频下载链接| 91久久精品色伊人6882| 久青青草视频手机在线免费观看| 一区二区三区美女毛片| 国产熟妇乱妇熟色T区| 91久久精品色伊人6882| 在线观看免费av网址大全| 亚洲av男人天堂久久| 人人超碰国字幕观看97| 精品视频中文字幕在线播放| 亚洲成人国产av在线| 成年人免费看在线视频| 亚洲福利精品福利精品福利| 日韩亚洲高清在线观看| 久草电影免费在线观看| 蜜桃视频在线欧美一区| 亚洲男人在线天堂网| 在线免费观看靠比视频的网站| 91精品视频在线观看免费| 亚洲一级 片内射视正片| 国产又大又黄免费观看| 国产精品人妻熟女毛片av久| 亚洲精品无码久久久久不卡 | 日韩国产乱码中文字幕| 日本少妇精品免费视频| 人人人妻人人澡人人| 精品av国产一区二区三区四区| 在线观看视频 你懂的| 超污视频在线观看污污污| 亚洲无线观看国产高清在线| 黄色的网站在线免费看| 熟女在线视频一区二区三区| 伊人情人综合成人久久网小说 | 国产亚洲视频在线二区| 老司机99精品视频在线观看| 激情国产小视频在线| 天堂av在线最新版在线| 黄片色呦呦视频免费看| 日韩熟女系列一区二区三区| 动色av一区二区三区| 欧美一区二区三区四区性视频| 国产大鸡巴大鸡巴操小骚逼小骚逼| 97精品综合久久在线| 免费一级特黄特色大片在线观看 | 国产一区二区神马久久| 日韩剧情片电影在线收看| 在线免费观看视频一二区| 亚洲一级av无码一级久久精品| 激情图片日韩欧美人妻| av久久精品北条麻妃av观看| 在线播放一区二区三区Av无码| 香蕉aⅴ一区二区三区| 成人免费做爰高潮视频| v888av在线观看视频| 国产一级精品综合av| 888欧美视频在线| 熟妇一区二区三区高清版| 亚洲天堂有码中文字幕视频| 亚洲人人妻一区二区三区| 伊人开心婷婷国产av| 骚货自慰被发现爆操| 成人精品在线观看视频| 日日夜夜大香蕉伊人| 偷偷玩弄新婚人妻h视频| 91国内视频在线观看| 日本人妻少妇18—xx| 美女福利写真在线观看视频| 视频在线亚洲一区二区| 国产日本欧美亚洲精品视| 18禁免费av网站| 国产av福利网址大全| 97少妇精品在线观看| 2020久久躁狠狠躁夜夜躁| 中文 成人 在线 视频| 精品成人午夜免费看| 99精品视频之69精品视频| 影音先锋女人av噜噜色| 日韩剧情片电影在线收看| 日韩在线视频观看有码在线| 免费观看理论片完整版| 国产之丝袜脚在线一区二区三区| 38av一区二区三区| 欧美亚洲自偷自拍 在线| 国产亚洲欧美45p| 中文字幕在线第一页成人| 国产av欧美精品高潮网站| 国产精品视频欧美一区二区 | 护士特殊服务久久久久久久| 人妻久久久精品69系列| 欧美日本国产自视大全| 狠狠躁狠狠爱网站视频| 一级黄色片夫妻性生活| 嫩草aⅴ一区二区三区| 蜜桃久久久久久久人妻| 91精品高清一区二区三区| 久久久久久久一区二区三| 人妻丝袜榨强中文字幕| 人妻丝袜精品中文字幕| 18禁美女无遮挡免费| 亚洲va国产va欧美va在线| 懂色av蜜桃a v| 天天日天天日天天擦| 99久久99一区二区三区| 欧美黑人性猛交xxxxⅹooo| 亚洲av日韩av第一区二区三区| aⅴ精产国品一二三产品| 日本人妻少妇18—xx| 女生被男生插的视频网站| 大香蕉伊人中文字幕| 日韩三级电影华丽的外出| 亚洲精品国产综合久久久久久久久| 亚洲中文字幕校园春色| 年轻的人妻被夫上司侵犯| 天码人妻一区二区三区在线看| 91一区精品在线观看| 免费成人av中文字幕| 做爰视频毛片下载蜜桃视频1| 可以在线观看的av中文字幕| 在线免费91激情四射 | 国产又粗又猛又爽又黄的视频美国| 无码中文字幕波多野不卡| 免费成人av中文字幕| 亚洲免费在线视频网站| 东京热男人的av天堂| 日韩二区视频一线天婷婷五| 午夜婷婷在线观看视频| 蜜桃视频在线欧美一区| 91老师蜜桃臀大屁股| 成人动漫大肉棒插进去视频| av无限看熟女人妻另类av| 欧洲黄页网免费观看| 国产乱子伦精品视频潮优女| 91老熟女连续高潮对白| 亚洲精品午夜久久久久| 亚洲人妻30pwc| av高潮迭起在线观看| 黄色成人在线中文字幕| 国产亚洲视频在线二区| free性日本少妇| 啊用力插好舒服视频| 成人免费公开视频无毒| 国产精品大陆在线2019不卡| 天天综合天天综合天天网| 欧美另类一区二区视频| 青青草在观免费国产精品| 天天干夜夜操天天舔| 91传媒一区二区三区| 换爱交换乱高清大片| 亚欧在线视频你懂的| 适合午夜一个人看的视频| 伊人综合aⅴ在线网| 中文字幕人妻三级在线观看| 少妇高潮无套内谢麻豆| 欧美精品激情在线最新观看视频| 1024久久国产精品| 国产极品美女久久久久久| 色狠狠av线不卡香蕉一区二区| 不卡一不卡二不卡三| 国产黄色高清资源在线免费观看| 免费观看污视频网站| 偷拍自拍 中文字幕| 国产精品黄片免费在线观看| 午夜免费体验区在线观看| 首之国产AV医生和护士小芳| 欧美在线精品一区二区三区视频| 久草视频 久草视频2| 精品老妇女久久9g国产| 自拍偷拍日韩欧美亚洲| 一区二区三区毛片国产一区| 98精产国品一二三产区区别| 亚洲中文字幕校园春色| 久久精品视频一区二区三区四区| 伊拉克及约旦宣布关闭领空| 在线不卡日韩视频播放| 中文字幕人妻三级在线观看| 国产男女视频在线播放| 在线免费观看靠比视频的网站 | 快点插进来操我逼啊视频| 狠狠嗨日韩综合久久| 五十路丰满人妻熟妇| 91精品国产黑色丝袜| 国产精品入口麻豆啊啊啊| 一区二区久久成人网| 国产刺激激情美女网站| 欧美精品资源在线观看| 扒开让我视频在线观看| 精品高跟鞋丝袜一区二区| 18禁污污污app下载| 啊啊好慢点插舔我逼啊啊啊视频| 丰满的继坶3中文在线观看| 欧美va亚洲va天堂va| 999九九久久久精品| 日韩熟女av天堂系列| 超黄超污网站在线观看| 国产亚洲精品欧洲在线观看| 大胸性感美女羞爽操逼毛片| 18禁美女羞羞免费网站| 精品人妻每日一部精品| 激情人妻校园春色亚洲欧美| 国产大学生援交正在播放| 成人免费毛片aaaa| 少妇人妻久久久久视频黄片| 蜜臀av久久久久久久| 国产极品美女久久久久久| 天天日天天做天天日天天做| 亚洲一区二区三区五区| 欧美日韩人妻久久精品高清国产 | 日本熟妇色熟妇在线观看| 久久久人妻一区二区| 日本av高清免费网站| 免费在线观看视频啪啪| 91片黄在线观看喷潮| 丝袜美腿欧美另类 中文字幕| av俺也去在线播放| 精品美女在线观看视频在线观看| 92福利视频午夜1000看| 福利视频广场一区二区| 五十路熟女av天堂| 夜鲁夜鲁狠鲁天天在线| 在线免费91激情四射| 免费在线黄色观看网站| 91国产在线免费播放| 91小伙伴中女熟女高潮| 97瑟瑟超碰在线香蕉| 大香蕉大香蕉大香蕉大香蕉大香蕉| 国产一区二区久久久裸臀| 婷婷午夜国产精品久久久| 国产超码片内射在线| av中文字幕国产在线观看| 亚洲1卡2卡三卡4卡在线观看| 激情内射在线免费观看| 一级黄片大鸡巴插入美女| 在线观看国产免费麻豆| 免费观看污视频网站| 亚洲自拍偷拍精品网| 国产97视频在线精品| 午夜在线观看一区视频| 人妻熟女中文字幕aⅴ在线| weyvv5国产成人精品的视频| 人妻少妇精品久久久久久| 性欧美日本大妈母与子| 国产乱子伦一二三区| 美女福利写真在线观看视频| 美女福利写真在线观看视频| 深夜男人福利在线观看| 极品粉嫩小泬白浆20p主播 | 国产黄色片在线收看| 亚洲av无码成人精品区辽| 国产白嫩美女一区二区| 亚洲一区二区三区精品视频在线| 在线免费观看国产精品黄色| 亚洲精品成人网久久久久久小说| 天天干狠狠干天天操| 亚洲欧洲一区二区在线观看| 午夜成午夜成年片在线观看| 91she九色精品国产| 亚洲少妇高潮免费观看| 国产一线二线三线的区别在哪| 偷拍自拍国产在线视频| 最新91九色国产在线观看| 国产午夜亚洲精品不卡在线观看| 中文字幕AV在线免费看 | 最近中文2019年在线看| 91福利在线视频免费观看| 阿v天堂2014 一区亚洲| 在线观看的a站 最新| 99热99re在线播放| 国产精品3p和黑人大战| 中文字幕午夜免费福利视频| 粉嫩小穴流水视频在线观看| 国产精品sm调教视频| 绝色少妇高潮3在线观看| 青青青青青操视频在线观看| 天天色天天舔天天射天天爽| 老司机深夜免费福利视频在线观看| 久久综合老鸭窝色综合久久| 亚洲欧美另类自拍偷拍色图| 又粗又硬又猛又爽又黄的| 美女 午夜 在线视频| 中文字幕国产专区欧美激情| 66久久久久久久久久久| 第一福利视频在线观看| sejizz在线视频| 久久久久久久久久久久久97| 91国内视频在线观看| 在线观看视频污一区| 日本熟女50视频免费| 国产精品久久久久久久久福交| 91免费放福利在线观看| 97国产在线av精品| 亚洲2021av天堂| 日本少妇高清视频xxxxx | 91麻豆精品91久久久久同性| 国产一区av澳门在线观看| 无码日韩人妻精品久久| 懂色av之国产精品| 久久99久久99精品影院| 天天干天天操天天爽天天摸| 天天色天天舔天天射天天爽| 亚国产成人精品久久久| 亚洲激情唯美亚洲激情图片| 日本免费午夜视频网站| 精品91自产拍在线观看一区| 日韩人妻xxxxx| 国产1区,2区,3区| 男人的天堂一区二区在线观看| 亚洲国产成人av在线一区| 国产露脸对白在线观看| 在线免费观看日本伦理| 黄色成年网站午夜在线观看 | 视频 一区二区在线观看| 亚洲成人熟妇一区二区三区 | yellow在线播放av啊啊啊| 大肉大捧一进一出好爽在线视频| 国产揄拍高清国内精品对白| 中文字幕在线欧美精品| 色97视频在线播放| 国产精品大陆在线2019不卡| 亚洲综合色在线免费观看| 久久免看30视频口爆视频| 女同性ⅹxx女同hd| 我想看操逼黄色大片| 777奇米久久精品一区| 性欧美激情久久久久久久| 农村胖女人操逼视频| 少妇高潮无套内谢麻豆| 韩国三级aaaaa高清视频| 欧美亚洲自偷自拍 在线| 在线视频精品你懂的| 精品国产午夜视频一区二区| 国产亚洲国产av网站在线| 香港一级特黄大片在线播放| 亚洲欧美一区二区三区爱爱动图| 一区二区三区日韩久久| 人妻久久无码中文成人| 在线观看视频污一区| 精内国产乱码久久久久久| yellow在线播放av啊啊啊| 天天躁日日躁狠狠躁躁欧美av| 护士特殊服务久久久久久久| 99一区二区在线观看| 五色婷婷综合狠狠爱| 亚洲伊人久久精品影院一美女洗澡| 日韩影片一区二区三区不卡免费| 成人资源在线观看免费官网| 亚洲精品ww久久久久久| 欧美成人小视频在线免费看| 深夜男人福利在线观看| 欧美另类重口味极品在线观看| 亚洲高清视频在线不卡| 亚洲粉嫩av一区二区三区| 少妇人妻二三区视频| 国产在线自在拍91国语自产精品| av视网站在线观看| 久久人人做人人妻人人玩精品vr| 任你操视频免费在线观看| 蜜桃视频17c在线一区二区| 青青草视频手机免费在线观看| 九九视频在线精品播放| 人妻自拍视频中国大陆| 亚洲国产精品久久久久蜜桃| 视频一区二区在线免费播放| 日本少妇的秘密免费视频| 欧洲欧美日韩国产在线| 精品一区二区三区午夜| 成人av电影免费版| 中文字幕—97超碰网| 亚洲免费在线视频网站| 国产激情av网站在线观看| 亚洲福利午夜久久久精品电影网| 适合午夜一个人看的视频| caoporm超碰国产| 亚洲欧美国产综合777| 国产普通话插插视频| 国产精品人妻66p| 91欧美在线免费观看| 91福利视频免费在线观看| 人妻少妇亚洲一区二区| 男人天堂最新地址av| 欧美精品亚洲精品日韩在线| 99精品国产自在现线观看| 男女之间激情网午夜在线| 亚洲欧美自拍另类图片| 久久丁香婷婷六月天| 天天爽夜夜爽人人爽QC| 美女福利视频导航网站 | 人妻熟女在线一区二区| 97年大学生大白天操逼| 亚洲在线免费h观看网站| 人妻av无码专区久久绿巨人 | 黄色资源视频网站日韩| 搡老熟女一区二区在线观看| 农村胖女人操逼视频| 伊人综合aⅴ在线网| 成年人的在线免费视频| 在线观看国产网站资源| 91成人精品亚洲国产| 中文字幕在线一区精品| 一区二区在线视频中文字幕| 最新国产精品拍在线观看| 蜜桃专区一区二区在线观看| 国产黄色高清资源在线免费观看| 国产日韩av一区二区在线| 特黄老太婆aa毛毛片| 在线视频国产欧美日韩| 免费观看国产综合视频| 亚洲2021av天堂| 国产麻豆剧传媒精品国产av蜜桃| 日韩剧情片电影在线收看| 日辽宁老肥女在线观看视频| 国产变态另类在线观看| 亚洲中文精品人人免费| av线天堂在线观看| 国产精品三级三级三级| 免费观看丰满少妇做受| 五十路熟女人妻一区二| 东京热男人的av天堂| 91麻豆精品久久久久| 亚洲 自拍 色综合图| 大鸡八强奸视频在线观看| 女同久久精品秋霞网| 久久h视频在线观看| 欧美国产亚洲中英文字幕| 绝顶痉挛大潮喷高潮无码| 中文字幕中文字幕 亚洲国产| 欧美中国日韩久久精品| 激情啪啪啪啪一区二区三区| 亚洲成人情色电影在线观看| 1000部国产精品成人观看视频| 在线观看视频网站麻豆| 40道精品招牌菜特色| japanese五十路熟女熟妇| 国产午夜激情福利小视频在线| 国产综合精品久久久久蜜臀| 欧美精产国品一二三产品区别大吗| 97色视频在线观看| 亚洲欧美一卡二卡三卡| 国产午夜无码福利在线看| 人妻久久久精品69系列| 天天日天天干天天舔天天射| 亚洲精品色在线观看视频| 亚洲成a人片777777| 欧洲国产成人精品91铁牛tv| 哥哥姐姐综合激情小说| 小穴多水久久精品免费看| 青娱乐在线免费视频盛宴| 人妻久久久精品69系列| 激情五月婷婷免费视频| 女生被男生插的视频网站| 丝袜美腿欧美另类 中文字幕| 四虎永久在线精品免费区二区| 天天日天天日天天擦| 日本熟妇一区二区x x| 视频啪啪啪免费观看| 亚洲图片欧美校园春色 | 丝袜肉丝一区二区三区四区在线| 男人操女人的逼免费视频| 日韩北条麻妃一区在线| 国产欧美精品免费观看视频| 视频在线免费观看你懂得| 超碰公开大香蕉97| av在线播放国产不卡| av日韩在线观看大全| 国产成人精品av网站| 久久香蕉国产免费天天| 中文字幕一区二区自拍| 丝袜美腿视频诱惑亚洲无| 国产亚洲天堂天天一区| 亚洲人妻国产精品综合| 亚洲一区av中文字幕在线观看| 中文字幕乱码av资源| 欧美激情电影免费在线| 老司机免费福利视频网| 在线观看一区二区三级| 亚洲熟女久久久36d| 免费黄页网站4188| av一本二本在线观看| 国产精品黄色的av| 97人人妻人人澡人人爽人人精品| 视频久久久久久久人妻| 好太好爽好想要免费| 久久久噜噜噜久久熟女av| 国产精品手机在线看片| 国产成人自拍视频播放| 中国把吊插入阴蒂的视频| 绝顶痉挛大潮喷高潮无码| 久久久久久性虐视频| 日韩黄色片在线观看网站| 亚洲一区二区三区久久午夜 | 小泽玛利亚视频在线观看| 国产一区二区在线欧美| 十八禁在线观看地址免费| 亚洲自拍偷拍精品网| 北条麻妃高跟丝袜啪啪| 制服丝袜在线人妻中文字幕| 经典亚洲伊人第一页| 国产亚洲精品欧洲在线观看| 午夜影院在线观看视频羞羞羞| huangse网站在线观看| 亚洲免费在线视频网站| 视频在线免费观看你懂得| 国产精品黄片免费在线观看| 日韩亚洲高清在线观看| 中文字幕在线免费第一页| 久久久精品999精品日本| 日本成人不卡一区二区| 丰满的继坶3中文在线观看| 动漫av网站18禁| 亚洲熟妇无码一区二区三区| 亚洲激情唯美亚洲激情图片| jiuse91九色视频| 欧美特级特黄a大片免费| 操操网操操伊剧情片中文字幕网| 亚洲精品欧美日韩在线播放| 天天躁夜夜躁日日躁a麻豆| 成人综合亚洲欧美一区 | 亚洲av男人天堂久久| 欧美日本在线观看一区二区| 日韩欧美一级精品在线观看| 日韩成人综艺在线播放| 2021年国产精品自拍| 99久久激情婷婷综合五月天| 久久永久免费精品人妻专区| 中文人妻AV久久人妻水| 岛国青草视频在线观看| 欧美 亚洲 另类综合| 日韩av免费观看一区| 蜜臀av久久久久蜜臀av麻豆| 97人人模人人爽人人喊| 99久久超碰人妻国产| 黑人变态深video特大巨大| 久久久久久久99精品| 在线播放 日韩 av| 成年美女黄网站18禁久久| 1区2区3区不卡视频| 喷水视频在线观看这里只有精品| 青青草成人福利电影| 欧洲欧美日韩国产在线| 欧美黑人巨大性xxxxx猛交| 97人妻色免费视频| 中文字幕最新久久久| 色婷婷综合激情五月免费观看| 亚洲 中文 自拍 无码| 国产又大又黄免费观看| 国产高清精品极品美女| 午夜福利资源综合激情午夜福利资| 成人av久久精品一区二区| 国产大鸡巴大鸡巴操小骚逼小骚逼| 国产又粗又黄又硬又爽| 美女大bxxxx内射| av天堂资源最新版在线看| 蜜桃色婷婷久久久福利在线| 青青青青青操视频在线观看| 40道精品招牌菜特色| 欧美精产国品一二三产品价格| 少妇一区二区三区久久久| 欧美aa一级一区三区四区| 免费在线看的黄网站| 国产又粗又硬又大视频| 中文字幕在线乱码一区二区 | 毛茸茸的大外阴中国视频| 日本熟女50视频免费| 四川乱子伦视频国产vip| 日本一道二三区视频久久| 色伦色伦777国产精品| 免费观看成年人视频在线观看| 青青尤物在线观看视频网站| 91高清成人在线视频| 涩涩的视频在线观看视频| 91人妻精品一区二区在线看| 人妻少妇av在线观看| 日韩美女精品视频在线观看网站| 亚洲精品无码久久久久不卡| 日本五十路熟新垣里子| 综合色区亚洲熟妇shxstz| 亚洲av无硬久久精品蜜桃| 精品av国产一区二区三区四区| 免费av岛国天堂网站| 人妻少妇一区二区三区蜜桃| 日韩激情文学在线视频| 色花堂在线av中文字幕九九 | 日本最新一二三区不卡在线 | 春色激情网欧美成人| av森泽佳奈在线观看| 日本丰满熟妇BBXBBXHD| 97国产在线观看高清| 沈阳熟妇28厘米大战黑人| 欧美在线一二三视频| 欧美成人一二三在线网| 啊啊啊想要被插进去视频| 97超碰最新免费在线观看| av天堂中文免费在线| 亚洲欧美福利在线观看| 久草视频首页在线观看| 操操网操操伊剧情片中文字幕网| 97国产在线观看高清| 视频在线亚洲一区二区| 97年大学生大白天操逼| 欧美美女人体视频一区| 91p0rny九色露脸熟女| 大香蕉大香蕉在线有码 av| 日本在线一区二区不卡视频| av天堂中文字幕最新| 中文字日产幕乱六区蜜桃| 97少妇精品在线观看| 97超碰免费在线视频| 亚洲av色图18p| 久久久久久国产精品| 中国视频一区二区三区| 日本一二三中文字幕| 久草福利电影在线观看| 人妻少妇av在线观看| av中文字幕福利网| aⅴ五十路av熟女中出| 在线观看视频污一区| 高潮视频在线快速观看国家快速 | 啪啪啪操人视频在线播放| 国产剧情演绎系列丝袜高跟| 九九视频在线精品播放| 午夜精品福利一区二区三区p| 日本黄在免费看视频| 国产亚洲国产av网站在线|