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

Springboot整合Ehcache和Redis實現多級緩存實戰案例

開發 架構 Redis
本文通過springboot整合ehcache和redis實現多級緩存案例實戰,從源碼角度分析下多級緩存實現原理。

一、概述

在實際的工作中,我們通常會使用多級緩存機制,將本地緩存和分布式緩存結合起來,從而提高系統性能和響應速度。本文通過springboot整合ehcache和redis實現多級緩存案例實戰,從源碼角度分析下多級緩存實現原理。

二、實戰案例

pom依賴(注意引入cache和ehcache組件依賴)。

<?xml versinotallow="1.0" encoding="UTF-8"?>
<project xmlns="http://maven.apache.org/POM/4.0.0"
         xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
         xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
    <modelVersion>4.0.0</modelVersion>

    <groupId>org.example</groupId>
    <artifactId>cache-demo</artifactId>
    <version>1.0-SNAPSHOT</version>

    <properties>
        <maven.compiler.source>8</maven.compiler.source>
        <maven.compiler.target>8</maven.compiler.target>
    </properties>

    <parent>
        <groupId>org.springframework.boot</groupId>
        <artifactId>spring-boot-starter-parent</artifactId>
        <version>2.5.0</version>
    </parent>

    <dependencies>
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-web</artifactId>
        </dependency>
        <dependency>
            <groupId>junit</groupId>
            <artifactId>junit</artifactId>
            <version>4.12</version>
        </dependency>
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-test</artifactId>
            <scope>test</scope>
        </dependency>
        <dependency>
            <groupId>com.baomidou</groupId>
            <artifactId>mybatis-plus-boot-starter</artifactId>
            <version>3.4.3</version>
        </dependency>
        <dependency>
            <groupId>mysql</groupId>
            <artifactId>mysql-connector-java</artifactId>
        </dependency>
        <dependency>
            <groupId>com.alibaba</groupId>
            <artifactId>druid-spring-boot-starter</artifactId>
            <version>1.2.1</version>
        </dependency>
        <dependency>
            <groupId>org.projectlombok</groupId>
            <artifactId>lombok</artifactId>
        </dependency>
        <dependency>
            <groupId>com.alibaba</groupId>
            <artifactId>fastjson</artifactId>
            <version>1.2.76</version>
        </dependency>
        <dependency>
            <groupId>com.alibaba</groupId>
            <artifactId>druid</artifactId>
            <version>1.1.23</version>
        </dependency>
        <dependency>
            <groupId>com.google.guava</groupId>
            <artifactId>guava</artifactId>
            <version>23.0</version>
        </dependency>
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-data-redis</artifactId>
        </dependency>
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-cache</artifactId>
        </dependency>
        <dependency>
            <groupId>net.sf.ehcache</groupId>
            <artifactId>ehcache</artifactId>
            <version>2.10.8</version>
        </dependency>
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-aop</artifactId>
        </dependency>
    </dependencies>
</project>

application.properties(啟動類加上:@EnableCaching注解)。

server.port = 7001
spring.application.name = cache-demo

#log config
logging.config = classpath:log/logback.xml
debug = false

#mp config
mybatis-plus.mapper-locations = classpath*:mapper/*.xml
mybatis-plus.configuration.log-impl = org.apache.ibatis.logging.stdout.StdOutImpl

spring.datasource.type = com.alibaba.druid.pool.DruidDataSource
spring.datasource.druid.driver-class-name = com.mysql.cj.jdbc.Driver
spring.datasource.url = jdbc:mysql://localhost:3306/數據庫?characterEncoding=utf-8
spring.datasource.username = 數據庫賬號
spring.datasource.password = 數據庫密碼

#redis config
spring.redis.host = redis主機
spring.redis.port = 6379
spring.redis.password=redis密碼,沒有就刪掉該配置

# ehcache config
spring.cache.type = ehcache
spring.cache.ehcache.config = classpath:ehcache.xml

ehcache.xml。

<?xml version="1.0" encoding="UTF-8"?>
<ehcache xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
         xsi:noNamespaceSchemaLocation="http://ehcache.org/ehcache.xsd"
         updateCheck="false">
    <diskStore path="D:\ehcache"/>

    <!--默認緩存策略 -->
    <!-- external:是否永久存在,設置為true則不會被清除,此時與timeout沖突,通常設置為false-->
    <!-- diskPersistent:是否啟用磁盤持久化-->
    <!-- maxElementsInMemory:最大緩存數量-->
    <!-- overflowToDisk:超過最大緩存數量是否持久化到磁盤-->
    <!-- timeToIdleSeconds:最大不活動間隔,設置過長緩存容易溢出,設置過短無效果,單位:秒-->
    <!-- timeToLiveSeconds:最大存活時間,單位:秒-->
    <!-- memoryStoreEvictionPolicy:緩存清除策略-->
    <defaultCache
            eternal="false"
            diskPersistent="false"
            maxElementsInMemory="1000"
            overflowToDisk="false"
            timeToIdleSeconds="60"
            timeToLiveSeconds="60"
            memoryStoreEvictionPolicy="LRU"/>

    <cache
            name="studentCache"
            eternal="false"
            diskPersistent="false"
            maxElementsInMemory="1000"
            overflowToDisk="false"
            timeToIdleSeconds="100"
            timeToLiveSeconds="100"
            memoryStoreEvictionPolicy="LRU"/>
</ehcache>

MybatisPlusConfig類(注意:@MapperScan注解,也可加在啟動類上)。

@Configuration
@MapperScan("com.cache.demo.mapper")
public class MybatisPlusConfig {

    @Bean
    public MybatisPlusInterceptor mybatisPlusInterceptor() {
        //分頁插件
        MybatisPlusInterceptor mybatisPlusInterceptor = new MybatisPlusInterceptor();
        mybatisPlusInterceptor.addInnerInterceptor(new PaginationInnerInterceptor());
        return mybatisPlusInterceptor;
    }
}

測試demo。

這里可以將一級緩存、二級緩存時效設置短一些,方便進行測試。

@Slf4j
@RestController
@RequestMapping("/cache")
public class CacheController {

    @Resource
    private StudentMapper studentMapper;
    @Autowired
    private StringRedisTemplate stringRedisTemplate;

  	// 添加緩存注解(一級緩存:ehcache)
    @Cacheable(value = "studentCache", key = "#id+'getStudentById'")
    @GetMapping("/getStudentById")
    public String getStudentById(Integer id) {
        String key = "student:" + id;
      	// 一級緩存中不存在,則從二級緩存:redis中查找
        String studentRedis = stringRedisTemplate.opsForValue().get(key);
        if (StringUtils.isNotBlank(studentRedis)) {
            return JSON.toJSONString(JSON.parseObject(studentRedis, Student.class));
        }
        // 二級緩存中不存在則查詢數據庫,并更新二級緩存、一級緩存
        Student student = studentMapper.selectStudentById(id);
        if (null != student) {
            stringRedisTemplate.opsForValue().set(key, JSON.toJSONString(student));
        }
        return JSON.toJSONString(student);
    }
}

啟動類上的:@EnableCaching注解。

@Target(ElementType.TYPE)
@Retention(RetentionPolicy.RUNTIME)
@Documented
@Import(CachingConfigurationSelector.class)
public @interface EnableCaching {
  
	boolean proxyTargetClass() default false;
  
	AdviceMode mode() default AdviceMode.PROXY;
  
  int order() default Ordered.LOWEST_PRECEDENCE;
}

導入的:CachingConfigurationSelector類:

public class CachingConfigurationSelector extends AdviceModeImportSelector<EnableCaching> {

	@Override
	public String[] selectImports(AdviceMode adviceMode) {
		switch (adviceMode) {
			case PROXY:
        // 此處走的是:PROXY
				return getProxyImports();
			case ASPECTJ:
				return getAspectJImports();
			default:
				return null;
		}
	}

	private String[] getProxyImports() {
		List<String> result = new ArrayList<>(3);
    // 導入了AutoProxyRegistrar類和ProxyCachingConfiguration類
		result.add(AutoProxyRegistrar.class.getName());
		result.add(ProxyCachingConfiguration.class.getName());
		if (jsr107Present && jcacheImplPresent) {
			result.add(PROXY_JCACHE_CONFIGURATION_CLASS);
		}
		return StringUtils.toStringArray(result);
	}
}

AutoProxyRegistrar類(代碼有所簡化):

public class AutoProxyRegistrar implements ImportBeanDefinitionRegistrar {

	private final Log logger = LogFactory.getLog(getClass());

	@Override
	public void registerBeanDefinitions(AnnotationMetadata importingClassMetadata, BeanDefinitionRegistry registry) {
				// 最終注冊了:InfrastructureAdvisorAutoProxyCreator(BeanPostProcessor接口實現類)
    		// 通過重寫postProcessAfterInitialization接口創建代理對象
      	AopConfigUtils.registerAutoProxyCreatorIfNecessary(registry);
	}
}

@Nullable
	public static BeanDefinition registerAutoProxyCreatorIfNecessary(BeanDefinitionRegistry registry, @Nullable Object source) {
		return registerOrEscalateApcAsRequired(InfrastructureAdvisorAutoProxyCreator.class, registry, source);
	}

導入的第一個類看完了,接著看導入的第二個類:ProxyCachingConfiguration。

@Configuration(proxyBeanMethods = false)
@Role(BeanDefinition.ROLE_INFRASTRUCTURE)
public class ProxyCachingConfiguration extends AbstractCachingConfiguration {

	@Bean(name = CacheManagementConfigUtils.CACHE_ADVISOR_BEAN_NAME)
	@Role(BeanDefinition.ROLE_INFRASTRUCTURE)
	public BeanFactoryCacheOperationSourceAdvisor cacheAdvisor(CacheOperationSource cacheOperationSource, CacheInterceptor cacheInterceptor) {
		//  構建BeanFactoryCacheOperationSourceAdvisor
    BeanFactoryCacheOperationSourceAdvisor advisor = new BeanFactoryCacheOperationSourceAdvisor();
		// 設置緩存注解解析器
    advisor.setCacheOperationSource(cacheOperationSource);
		// 設置緩存攔截器:cacheInterceptor
    advisor.setAdvice(cacheInterceptor);
		if (this.enableCaching != null) {
			advisor.setOrder(this.enableCaching.<Integer>getNumber("order"));
		}
		return advisor;
	}

	@Bean
	@Role(BeanDefinition.ROLE_INFRASTRUCTURE)
	public CacheOperationSource cacheOperationSource() {
    // 緩存注解解析器
		return new AnnotationCacheOperationSource();
	}

	@Bean
	@Role(BeanDefinition.ROLE_INFRASTRUCTURE)
	public CacheInterceptor cacheInterceptor(CacheOperationSource cacheOperationSource) {
		// 緩存攔截器
    CacheInterceptor interceptor = new CacheInterceptor();
		interceptor.configure(this.errorHandler, this.keyGenerator, this.cacheResolver, this.cacheManager);
		interceptor.setCacheOperationSource(cacheOperationSource);
		return interceptor;
	}
}

繼續看下CacheInterceptor類(重要):

public class CacheInterceptor extends CacheAspectSupport implements MethodInterceptor, Serializable {

	@Override
	@Nullable
	public Object invoke(final MethodInvocation invocation) throws Throwable {
		Method method = invocation.getMethod();
		CacheOperationInvoker aopAllianceInvoker = () -> {
			try {
				return invocation.proceed();
			}
			catch (Throwable ex) {
				throw new CacheOperationInvoker.ThrowableWrapper(ex);
			}
		};

		Object target = invocation.getThis();
		Assert.state(target != null, "Target must not be null");
		try {
      // 緩存執行邏輯
			return execute(aopAllianceInvoker, target, method, invocation.getArguments());
		}
		catch (CacheOperationInvoker.ThrowableWrapper th) {
			throw th.getOriginal();
		}
	}
}

@Nullable
	protected Object execute(CacheOperationInvoker invoker, Object target, Method method, Object[] args) {
		if (this.initialized) {
			Class<?> targetClass = getTargetClass(target);
			CacheOperationSource cacheOperationSource = getCacheOperationSource();
			if (cacheOperationSource != null) {
        // 解析緩存相關注解,返回CacheOperation
        // 每個緩存注解對應一種不同的解析處理操作
        // CacheEvictOperation、CachePutOperation、CacheableOperation等
				Collection<CacheOperation> operations = cacheOperationSource.getCacheOperations(method, targetClass);
				if (!CollectionUtils.isEmpty(operations)) {
          // 執行緩存邏輯
					return execute(invoker, method,
							new CacheOperationContexts(operations, method, args, target, targetClass));
				}
			}
		}
		return invoker.invoke();
	}

private Object execute(final CacheOperationInvoker invoker, Method method, CacheOperationContexts contexts) {
		// 解析處理@CacheEvict注解
    processCacheEvicts(contexts.get(CacheEvictOperation.class), true,	CacheOperationExpressionEvaluator.NO_RESULT);

		// 解析處理@Cacheable注解
		Cache.ValueWrapper cacheHit = findCachedItem(contexts.get(CacheableOperation.class));

		List<CachePutRequest> cachePutRequests = new ArrayList<>();
		if (cacheHit == null) {
			collectPutRequests(contexts.get(CacheableOperation.class),	CacheOperationExpressionEvaluator.NO_RESULT, cachePutRequests);
		}

		Object cacheValue;
		Object returnValue;

		if (cacheHit != null && !hasCachePut(contexts)) {
			// 命中緩存,則從緩存中獲取數據
			cacheValue = cacheHit.get();
			returnValue = wrapCacheValue(method, cacheValue);
		} else {
			// 未命中緩存,則通過反射執行目標方法
			returnValue = invokeOperation(invoker);
			cacheValue = unwrapReturnValue(returnValue);
		}

		// 解析處理@CachePut注解
		collectPutRequests(contexts.get(CachePutOperation.class), cacheValue, cachePutRequests);

		// 未命中緩存時,會封裝一個cachePutRequests
  	// 然后通過反射執行目標方法后,執行該方法,最終調用EhCacheCache.put方法將數據寫入緩存中
		for (CachePutRequest cachePutRequest : cachePutRequests) {
			cachePutRequest.apply(cacheValue);
		}
		// 解析處理@CacheEvict注解,和上面的方法相同,只不過第二個參數不同
		processCacheEvicts(contexts.get(CacheEvictOperation.class), false, cacheValue);
		return returnValue;
	}

接著看下findCachedItem方法。

private Cache.ValueWrapper findCachedItem(Collection<CacheOperationContext> contexts) {
		Object result = CacheOperationExpressionEvaluator.NO_RESULT;
		for (CacheOperationContext context : contexts) {
			if (isConditionPassing(context, result)) {
        // 生成key策略:解析@Cacheable注解中的key屬性
        // 若未配置則默認使用SimpleKeyGenerator#generateKey方法生成key
				Object key = generateKey(context, result);
				Cache.ValueWrapper cached = findInCaches(context, key);
				if (cached != null) {
					return cached;
				}	else {
					if (logger.isTraceEnabled()) {
						logger.trace("No cache entry for key '" + key + "' in cache(s) " + context.getCacheNames());
					}
				}
			}
		}
		return null;
	}

// SimpleKeyGenerator#generateKey
public static Object generateKey(Object... params) {
  	// 方法沒有參數,則返回空的SimpleKey
		if (params.length == 0) {
			return SimpleKey.EMPTY;
		}
  	// 方法參數只有一個,則返回該參數
		if (params.length == 1) {
			Object param = params[0];
			if (param != null && !param.getClass().isArray()) {
				return param;
			}
		}
  	// 否則將方法參數進行封裝,返回SimpleKey
		return new SimpleKey(params);
	}

private Cache.ValueWrapper findInCaches(CacheOperationContext context, Object key) {
		for (Cache cache : context.getCaches()) {
      // 從一級緩存中獲取數據
			Cache.ValueWrapper wrapper = doGet(cache, key);
			if (wrapper != null) {
				if (logger.isTraceEnabled()) {
					logger.trace("Cache entry for key '" + key + "' found in cache '" + cache.getName() + "'");
				}
				return wrapper;
			}
		}
		return null;
	}

protected Cache.ValueWrapper doGet(Cache cache, Object key) {
		try {
      // 這里我們使用的是:EhCacheCache,所以最終會調用EhCacheCache.get方法獲取緩存中的數據
			return cache.get(key);
		}
		catch (RuntimeException ex) {
			getErrorHandler().handleCacheGetError(ex, cache, key);
			return null;
		}
	}

三、總結

@EnableCaching和@Transactional等實現邏輯大體相同,看的多了,則一通百通。

責任編輯:姜華 來源: 今日頭條
相關推薦

2023-02-14 07:47:20

SpringBootEhcache

2020-06-29 07:43:12

緩存RedisSpringBoot

2020-01-10 15:42:13

SpringBootRedis數據庫

2022-03-15 08:22:31

Ehcachespring緩存

2023-01-13 07:39:07

2017-04-17 10:35:40

Spring BooRedis 操作

2019-04-23 08:42:42

EhcacheMemcacheRedis

2024-11-01 16:18:52

2024-11-04 08:02:23

SpringRabbitMQ中間件

2024-12-24 08:44:55

ActiveMQRabbitMQ交換機

2021-09-26 05:02:00

緩存Ehcache用法

2024-02-20 14:10:55

系統緩存冗余

2023-05-05 18:38:33

多級緩存Caffeine開發

2023-05-05 06:13:51

分布式多級緩存系統

2021-03-26 08:16:32

SpringbootWebsocket前端

2017-05-09 10:07:34

SpringbootDubboZooKeeper

2022-06-13 10:23:34

Helios緩存服務端

2025-04-21 03:00:00

2023-01-11 15:11:36

SpringEhcache

2015-12-28 10:48:44

RedisSpring緩存實例
點贊
收藏

51CTO技術棧公眾號

翔田千里一区二区| 韩日精品中文字幕| 8x8x成人免费视频| 国产99re66在线视频| 99久久国产综合色|国产精品| 国产激情久久久| 久草网站在线观看| 在线成人动漫av| 日韩一区和二区| 黄色影院一级片| 免费观看在线午夜影视| www.久久久久久久久| 国产精品入口日韩视频大尺度| 欧美黄色免费看| 精品一区三区| 欧美成人艳星乳罩| 91高清国产视频| 亚洲天堂av在线| 亚洲精品免费看| 日本午夜精品一区二区| 精品久久无码中文字幕| 日韩福利电影在线观看| 91精品国产一区| 黄色一级视频免费| 天天天综合网| 在线视频一区二区| 欧美日韩一区二区三区四区五区六区| 国产精品麻豆成人av电影艾秋| av免费在线不卡| 国产成人77亚洲精品www| 亚洲一区免费在线观看| 欧美裸体网站| 天天操天天爱天天干| 国产毛片一区二区| 国产日韩av在线播放| 91在线视频在线观看| 欧美啪啪一区| 久久国产精品免费视频| a一级免费视频| 欧美一级淫片| 亚洲图片在区色| 国产女主播喷水高潮网红在线| 国内精品麻豆美女在线播放视频| 日韩欧美国产wwwww| 北条麻妃亚洲一区| 9999精品| 欧美电影免费提供在线观看| 国产xxxxhd| 日韩精品视频在线看| 555www色欧美视频| 国产三级三级三级看三级| 国模冰冰炮一区二区| 色综合天天综合网国产成人综合天 | 中文亚洲视频在线| 91狠狠综合久久久久久| 第一会所亚洲原创| 这里只有精品丝袜| 天天操天天摸天天舔| 97视频热人人精品免费| 久久精品国产2020观看福利| 免费看特级毛片| 国产精品www.| 国内精品伊人久久| 91精品国产高清一区二区三密臀| 久久综合五月| 国产精品揄拍500视频| 国产一区二区三区三州| 国产一区二区三区日韩| 成人91视频| 天堂视频中文在线| 国产无人区一区二区三区| 亚洲v欧美v另类v综合v日韩v| 米奇精品一区二区三区| 一区二区三区在线观看国产 | 日日夜夜免费精品视频| 日本成人黄色片| 又污又黄的网站| 国产精品一级在线| 久久婷婷开心| 日本在线视频站| 夜夜亚洲天天久久| 日韩中文字幕免费在线| 国产aa精品| 亚洲精品91美女久久久久久久| 鲁丝一区二区三区| 亚洲国产精品91| 国产91精品不卡视频| 中文字幕91爱爱| 高清国产一区二区| 日本一区二区不卡高清更新| 最爽无遮挡行房视频在线| 福利视频导航一区| 国内av一区二区| 羞羞答答一区二区| 久久久99免费视频| 97超碰人人干| 国产精品白丝jk白祙喷水网站 | 可以免费看污视频的网站在线| 国产精品国产馆在线真实露脸| 欧洲精品一区二区三区久久| 精品久久毛片| 精品视频www| 成人免费精品动漫网站| 亚洲综合不卡| 3d精品h动漫啪啪一区二区| 久草在现在线| 亚洲成av人片一区二区梦乃| 黄大色黄女片18第一次| 免费看久久久| 欧美成人免费播放| 中文字幕男人天堂| 99久久婷婷国产| 美女黄色片网站| aaaa欧美| 亚洲美女黄色片| 国产稀缺真实呦乱在线| 国产在线精品视频| 亚洲国产一区二区在线 | av电影高清在线观看| 99久久免费视频.com| 一本二本三本亚洲码| 欧美电影免费观看高清完整| 精品久久国产老人久久综合| 午夜激情视频在线播放| 久久久久99| 久久青青草综合| 国产伦久视频在线观看| 日韩欧美久久久| 男人的午夜天堂| 男女激情视频一区| 狠狠综合久久av| eeuss鲁一区二区三区| 91精品婷婷国产综合久久性色| av男人的天堂av| 久久精品动漫| 蜜桃av色综合| 伊人久久国产| 亚洲图片欧美午夜| 日本免费精品视频| 久久久亚洲高清| 爱福利视频一区二区| 一区二区小说| 国产91免费看片| 久久综合九色综合久| 一本久久精品一区二区| 中文字幕免费在线播放| 亚洲美女视频在线免费观看| 精品国产乱码久久久久软件| 久久久日本电影| 自拍视频一区二区| 精品99视频| 国产区一区二区三区| 黄页网站大全在线免费观看| 欧美r级电影在线观看| 国产精品suv一区二区69| 91精品久久久久久久久久| 蜜桃久久精品乱码一区二区| 19禁羞羞电影院在线观看| 日韩欧美国产小视频| 久久亚洲成人av| 不卡视频在线观看| 日韩网址在线观看| 国产99久久精品一区二区300| 国产成人精品一区二区在线| av午夜在线| 7777精品伊人久久久大香线蕉完整版| 乱h高h女3p含苞待放| 国产成人精品免费| 国产黄色一级网站| 精品国产乱码久久久久久蜜坠欲下 | 国产成人久久精品| 午夜在线播放| 精品女同一区二区| 毛片基地在线观看| 国产精品无码永久免费888| 网站在线你懂的| 亚洲精品1234| 天天人人精品| 超碰97久久国产精品牛牛| 欧美又大又粗又长| 免费a级人成a大片在线观看| 精品久久久久久亚洲综合网| av图片在线观看| 综合网在线视频| 亚洲av成人精品一区二区三区| 丝袜美腿亚洲一区| 只有这里有精品| 欧美日韩精品一区二区三区在线观看| 国产精品第二页| 日韩av毛片| 亚洲欧美在线看| www国产一区| 在线观看国产精品网站| 久久天天躁日日躁| 亚洲视频一区二区三区四区| 亚洲综合自拍偷拍| 99精彩视频| 欧美黄色免费看| 国产欧美日韩在线| 国产精品熟妇一区二区三区四区 | av无码精品一区二区三区| 欧美激情 亚洲a∨综合| 日本一区二区高清视频| 超碰在线一区| 成人午夜一级二级三级| 日本成人三级电影| 久久久久久中文| 国产一区久久精品| 亚洲深夜福利视频| 日本成人动漫在线观看| 欧美电影一区二区| 午夜一区二区三区四区| 亚洲不卡一区二区三区| 搜索黄色一级片| 国产精品网站在线观看| jizz日本免费| 成人一区二区三区中文字幕| 午夜国产福利在线观看| 久久九九电影| 免费看国产曰批40分钟| 国内激情久久| 免费成人深夜夜行网站视频| 欧美综合另类| 欧美性天天影院| 色橹橹欧美在线观看视频高清| 91嫩草国产在线观看| 久久av日韩| 国产精品高清在线观看| 成人免费网站视频| 91精品国产高清久久久久久久久| 色综合999| 欧美精品免费在线| 成人免费视屏| 久久成人亚洲精品| 国产黄色小视频在线| 日韩在线欧美在线| av在线电影免费观看| 国产午夜精品一区二区三区| 亚洲人成色777777精品音频| 亚洲国产精彩中文乱码av在线播放| 精品久久在线观看| 精品国产欧美一区二区| 亚洲av无码片一区二区三区| 日韩写真欧美这视频| 国产精品伦理一区| 欧美人伦禁忌dvd放荡欲情| 又色又爽又黄无遮挡的免费视频| 精品视频色一区| 亚洲系列在线观看| 欧美片在线播放| 国产美女明星三级做爰| 欧美一级精品在线| 亚洲精品免费在线观看视频| 日韩一区二区在线播放| 国精产品乱码一区一区三区四区| 亚洲精品一线二线三线| 亚洲欧美综合一区二区| 亚洲欧洲国产精品| www在线播放| 精品国内产的精品视频在线观看| caopeng在线| 久久人人爽人人爽人人片av高请| 欧产日产国产精品视频| 国产激情999| 国产在线不卡一区二区三区| 成人自拍爱视频| 蜜桃成人av| 在线视频不卡一区二区| 欧美日韩一区二区三区四区在线观看| 国产精品12345| 欧美bbbbb| 国产5g成人5g天天爽| 成人精品一区二区三区中文字幕| 五月婷婷综合在线观看| 国产精品国产自产拍在线| 欧美黑人一级片| 欧美日韩亚洲一区二区| 在线视频1卡二卡三卡| 欧美va日韩va| 精彩国产在线| 欧美刺激性大交免费视频| 国产免费拔擦拔擦8x高清在线人 | 欧美成人在线免费| 亚洲精品福利电影| 成人黄色免费片| 天堂99x99es久久精品免费| 亚洲欧洲日韩综合二区| 精品动漫3d一区二区三区免费| 精品久久久久久久无码| 国产91在线|亚洲| 亚洲天堂av中文字幕| 午夜视频一区二区三区| 91久久国语露脸精品国产高跟| 亚洲а∨天堂久久精品9966| 成人av毛片| 91国产美女在线观看| 97国产精品videossex| 亚洲理论片在线观看| 亚洲欧美日韩精品久久久久| 91看片在线播放| 欧美精品tushy高清| 手机看片国产1024| 日韩手机在线视频| 欧美一区二区三区精品| 天堂av在线7| 久久综合电影一区| 女生影院久久| av日韩免费电影| 久久在线免费| 北条麻妃在线观看| 高清不卡一二三区| 成人做爰视频网站| 色综合久久久久| 亚洲精品视频网| 最近日韩中文字幕中文| 超级碰碰久久| 国产视色精品亚洲一区二区| 91精品国产自产拍在线观看蜜| 久草青青在线观看| 懂色av一区二区三区免费观看| 国产探花在线视频| 在线观看av一区二区| 青青久草在线| 91精品国产91| baoyu135国产精品免费| 国产精品久久成人免费观看| 捆绑调教一区二区三区| www.av天天| 色综合中文字幕| 午夜在线视频免费| 91国产美女在线观看| 久久99偷拍| 精品视频在线观看一区| 国产91高潮流白浆在线麻豆 | 国产成人精品一区二区免费看京 | 午夜久久中文| 九色91在线视频| 国产欧美另类| 久久丫精品国产亚洲av不卡| 欧美日韩国产专区| 五月天婷婷在线播放| 91av视频在线免费观看| 国产一区二区在线视频你懂的| 成年人看的毛片| www.成人网.com| 日韩美女视频网站| 亚洲电影在线观看| 欧美伦理91| 青青成人在线| 日本欧美一区二区| 国产精品一区二区亚洲| 欧美精品色综合| 成人在线免费看黄| 97中文在线观看| 亚洲成人原创| 日韩av一二区| 在线观看亚洲专区| 日本美女在线中文版| 92看片淫黄大片看国产片| 欧美日韩伊人| 91精品人妻一区二区| 在线精品视频一区二区三四| 日本高清中文字幕在线| 亚洲综合精品伊人久久| 在线观看视频免费一区二区三区| 亚洲精品乱码久久久久久不卡| 一本大道久久精品懂色aⅴ | 国产精品嫩草99a| 国产特级黄色片| 久久久免费电影| 国内成人自拍| 日本网站在线看| 性感美女极品91精品| 国产片在线观看| 亚洲综合色av| 性8sex亚洲区入口| 亚洲欧美综合7777色婷婷| 欧美tickling网站挠脚心| 国产精欧美一区二区三区蓝颜男同| 亚洲在线视频一区二区| 成人一区二区三区中文字幕| 最新中文字幕在线观看视频| 久久成年人视频| 亚洲动漫精品| wwwxxxx在线观看| 一本大道久久a久久综合| www在线免费观看视频| 精品久久蜜桃| 国产在线视视频有精品| 五月天婷婷久久| 久久综合伊人77777尤物| 欧美性生活一级片| 一级黄色高清视频| 91久久人澡人人添人人爽欧美| 成年人黄视频在线观看| 茄子视频成人在线观看| 成熟亚洲日本毛茸茸凸凹| 中文字幕男人天堂| 欧美亚洲在线播放| 欧美区国产区|