久久福利_99r_国产日韩在线视频_直接看av的网站_中文欧美日韩_久久一

您的位置:首頁技術文章
文章詳情頁

Spring中異步注解@Async的使用、原理及使用時可能導致的問題及解決方法

瀏覽:57日期:2023-08-25 11:07:23

前言

其實最近都在研究事務相關的內容,之所以寫這么一篇文章是因為前面寫了一篇關于循環依賴的文章:

《Spring循環依賴的解決辦法,你真的懂了嗎》

然后,很多同學碰到了下面這個問題,添加了Spring提供的一個異步注解@Async循環依賴無法被解決了,下面是一些讀者的留言跟群里同學碰到的問題:

Spring中異步注解@Async的使用、原理及使用時可能導致的問題及解決方法

Spring中異步注解@Async的使用、原理及使用時可能導致的問題及解決方法

本著講一個知識點就要講明白、講透徹的原則,我決定單獨寫一篇這樣的文章對@Async這個注解做一下詳細的介紹,這個注解帶來的問題遠遠不止循環依賴這么簡單,如果對它不夠熟悉的話建議慎用。

文章要點

Spring中異步注解@Async的使用、原理及使用時可能導致的問題及解決方法

@Async的基本使用

這個注解的作用在于可以讓被標注的方法異步執行,但是有兩個前提條件

配置類上添加@EnableAsync注解需要異步執行的方法的所在類由Spring管理需要異步執行的方法上添加了@Async注解

我們通過一個Demo體會下這個注解的作用吧

第一步,配置類上開啟異步:

@EnableAsync@Configuration@ComponentScan('com.dmz.spring.async')public class Config {}

第二步,

[code]@Component // 這個類本身要被Spring管理public class DmzAsyncService { @Async // 添加注解表示這

@Component // 這個類本身要被Spring管理public class DmzAsyncService { @Async // 添加注解表示這個方法要異步執行public void testAsync(){try {TimeUnit.SECONDS.sleep(1);} catch (InterruptedException e) {e.printStackTrace();}System.out.println('testAsync invoked');}}

第三步,測試異步執行

public class Main {public static void main(String[] args) {AnnotationConfigApplicationContext ac = new AnnotationConfigApplicationContext(Config.class);DmzAsyncService bean = ac.getBean(DmzAsyncService.class);bean.testAsync();System.out.println('main函數執行完成');}}// 程序執行結果如下:// main函數執行完成// testAsync invoked

通過上面的例子我們可以發現,DmzAsyncService中的testAsync方法是異步執行的,那么這背后的原理是什么呢?我們接著分析

原理分析

我們在分析某一個技術的時候,最重要的事情是,一定一定要找到代碼的入口,像Spring這種都很明顯,入口必定是在@EnableAsync這個注解上面,我們來看看這個注解干了啥事(本文基于5.2.x版本)

@Target(ElementType.TYPE)@Retention(RetentionPolicy.RUNTIME)@Documented// 這里是重點,導入了一個ImportSelector@Import(AsyncConfigurationSelector.class)public @interface EnableAsync { // 這個配置可以讓程序員配置需要被檢查的注解,默認情況下檢查的就是@Async注解Class<? extends Annotation> annotation() default Annotation.class; // 默認使用jdk代理boolean proxyTargetClass() default false; // 默認使用Spring AOPAdviceMode mode() default AdviceMode.PROXY; // 在后續分析我們會發現,這個注解實際往容器中添加了一個 // AsyncAnnotationBeanPostProcessor,這個后置處理器實現了Ordered接口 // 這個配置主要代表了AsyncAnnotationBeanPostProcessor執行的順序int order() default Ordered.LOWEST_PRECEDENCE;}

上面這個注解做的最重要的事情就是導入了一個AsyncConfigurationSelector,這個類的源碼如下:

public class AsyncConfigurationSelector extends AdviceModeImportSelector<EnableAsync> {private static final String ASYNC_EXECUTION_ASPECT_CONFIGURATION_CLASS_NAME ='org.springframework.scheduling.aspectj.AspectJAsyncConfiguration';@Override@Nullablepublic String[] selectImports(AdviceMode adviceMode) {switch (adviceMode) { // 默認會使用SpringAOP進行代理case PROXY:return new String[] {ProxyAsyncConfiguration.class.getName()};case ASPECTJ:return new String[] {ASYNC_EXECUTION_ASPECT_CONFIGURATION_CLASS_NAME};default:return null;}}}

這個類的作用是像容器中注冊了一個ProxyAsyncConfiguration,這個類的繼承關系如下:

Spring中異步注解@Async的使用、原理及使用時可能導致的問題及解決方法

我們先看下它的父類AbstractAsyncConfiguration,其源碼如下:

@Configurationpublic abstract class AbstractAsyncConfiguration implements ImportAware {@Nullableprotected AnnotationAttributes enableAsync;@Nullableprotected Supplier<Executor> executor;@Nullableprotected Supplier<AsyncUncaughtExceptionHandler> exceptionHandler; // 這里主要就是檢查將其導入的類上是否有EnableAsync注解 // 如果沒有的話就報錯@Overridepublic void setImportMetadata(AnnotationMetadata importMetadata) {this.enableAsync = AnnotationAttributes.fromMap(importMetadata.getAnnotationAttributes(EnableAsync.class.getName(), false));if (this.enableAsync == null) {throw new IllegalArgumentException('@EnableAsync is not present on importing class ' + importMetadata.getClassName());}} // 將容器中配置的AsyncConfigurer注入 // 異步執行嘛,所以我們可以配置使用的線程池 // 另外也可以配置異常處理器@Autowired(required = false)void setConfigurers(Collection<AsyncConfigurer> configurers) {if (CollectionUtils.isEmpty(configurers)) {return;}if (configurers.size() > 1) {throw new IllegalStateException('Only one AsyncConfigurer may exist');}AsyncConfigurer configurer = configurers.iterator().next();this.executor = configurer::getAsyncExecutor;this.exceptionHandler = configurer::getAsyncUncaughtExceptionHandler;}}

再來看看ProxyAsyncConfiguration這個類的源碼

@Configuration@Role(BeanDefinition.ROLE_INFRASTRUCTURE)public class ProxyAsyncConfiguration extends AbstractAsyncConfiguration {@Bean(name = TaskManagementConfigUtils.ASYNC_ANNOTATION_PROCESSOR_BEAN_NAME)@Role(BeanDefinition.ROLE_INFRASTRUCTURE)public AsyncAnnotationBeanPostProcessor asyncAdvisor() {AsyncAnnotationBeanPostProcessor bpp = new AsyncAnnotationBeanPostProcessor(); // 將通過AsyncConfigurer配置好的線程池跟異常處理器設置到這個后置處理器中 bpp.configure(this.executor, this.exceptionHandler);Class<? extends Annotation> customAsyncAnnotation = this.enableAsync.getClass('annotation');if (customAsyncAnnotation != AnnotationUtils.getDefaultValue(EnableAsync.class, 'annotation')) {bpp.setAsyncAnnotationType(customAsyncAnnotation);}bpp.setProxyTargetClass(this.enableAsync.getBoolean('proxyTargetClass'));bpp.setOrder(this.enableAsync.<Integer>getNumber('order'));return bpp;}}

這個類本身是一個配置類,它的作用是向容器中添加一個AsyncAnnotationBeanPostProcessor。到這一步我們基本上就可以明白了,@Async注解的就是通過AsyncAnnotationBeanPostProcessor這個后置處理器生成一個代理對象來實現異步的,接下來我們就具體看看AsyncAnnotationBeanPostProcessor是如何生成代理對象的,我們主要關注一下幾點即可:

是在生命周期的哪一步完成的代理? 切點的邏輯是怎么樣的?它會對什么樣的類進行攔截? 通知的邏輯是怎么樣的?是如何實現異步的?

基于上面幾個問題,我們進行逐一分析

是在生命周期的哪一步完成的代理?

我們抓住重點,AsyncAnnotationBeanPostProcessor是一個后置處理器器,按照我們對Spring的了解,大概率是在這個后置處理器的postProcessAfterInitialization方法中完成了代理,直接定位到這個方法,這個方法位于父類AbstractAdvisingBeanPostProcessor中,具體代碼如下:

public Object postProcessAfterInitialization(Object bean, String beanName) { // 沒有通知,或者是AOP的基礎設施類,那么不進行代理 if (this.advisor == null || bean instanceof AopInfrastructureBean) { return bean; } // 對已經被代理的類,不再生成代理,只是將通知添加到代理類的邏輯中 // 這里通過beforeExistingAdvisors決定是將通知添加到所有通知之前還是添加到所有通知之后 // 在使用@Async注解的時候,beforeExistingAdvisors被設置成了true // 意味著整個方法及其攔截邏輯都會異步執行 if (bean instanceof Advised) { Advised advised = (Advised) bean; if (!advised.isFrozen() && isEligible(AopUtils.getTargetClass(bean))) { if (this.beforeExistingAdvisors) { advised.addAdvisor(0, this.advisor); } else { advised.addAdvisor(this.advisor); } return bean; } } // 判斷需要對哪些Bean進行來代理 if (isEligible(bean, beanName)) { ProxyFactory proxyFactory = prepareProxyFactory(bean, beanName); if (!proxyFactory.isProxyTargetClass()) { evaluateProxyInterfaces(bean.getClass(), proxyFactory); } proxyFactory.addAdvisor(this.advisor); customizeProxyFactory(proxyFactory); return proxyFactory.getProxy(getProxyClassLoader()); } return bean;}

果不其然,確實是在這個方法中完成的代理。接著我們就要思考,切點的過濾規則是什么呢?

切點的邏輯是怎么樣的?

其實也不難猜到肯定就是類上添加了@Async注解或者類中含有被@Async注解修飾的方法。基于此,我們看看這個isEligible這個方法的實現邏輯,這個方位位于AbstractBeanFactoryAwareAdvisingPostProcessor中,也是AsyncAnnotationBeanPostProcessor的父類,對應代碼如下:

// AbstractBeanFactoryAwareAdvisingPostProcessor的isEligible方法// 調用了父類protected boolean isEligible(Object bean, String beanName) { return (!AutoProxyUtils.isOriginalInstance(beanName, bean.getClass()) && super.isEligible(bean, beanName));}protected boolean isEligible(Object bean, String beanName) { return isEligible(bean.getClass());}protected boolean isEligible(Class<?> targetClass) { Boolean eligible = this.eligibleBeans.get(targetClass); if (eligible != null) { return eligible; } if (this.advisor == null) { return false; } // 這里完成的判斷 eligible = AopUtils.canApply(this.advisor, targetClass); this.eligibleBeans.put(targetClass, eligible); return eligible;}

實際上最后就是根據advisor來確定是否要進行代理,在Spring中基于xml的AOP的詳細步驟這篇文章中我們提到過,advisor實際就是一個綁定了切點的通知,那么AsyncAnnotationBeanPostProcessor這個advisor是什么時候被初始化的呢?我們直接定位到AsyncAnnotationBeanPostProcessor的setBeanFactory方法,其源碼如下:

public void setBeanFactory(BeanFactory beanFactory) { super.setBeanFactory(beanFactory); // 在這里new了一個AsyncAnnotationAdvisor AsyncAnnotationAdvisor advisor = new AsyncAnnotationAdvisor(this.executor, this.exceptionHandler); if (this.asyncAnnotationType != null) { advisor.setAsyncAnnotationType(this.asyncAnnotationType); } advisor.setBeanFactory(beanFactory); // 完成了初始化 this.advisor = advisor;}

我們來看看AsyncAnnotationAdvisor中的切點匹配規程是怎么樣的,直接定位到這個類的buildPointcut方法中,其源碼如下:

protected Pointcut buildPointcut(Set<Class<? extends Annotation>> asyncAnnotationTypes) { ComposablePointcut result = null; for (Class<? extends Annotation> asyncAnnotationType : asyncAnnotationTypes) { // 就是根據這兩個匹配器進行匹配的 Pointcut cpc = new AnnotationMatchingPointcut(asyncAnnotationType, true); Pointcut mpc = new AnnotationMatchingPointcut(null, asyncAnnotationType, true); if (result == null) { result = new ComposablePointcut(cpc); } else { result.union(cpc); } result = result.union(mpc); } return (result != null ? result : Pointcut.TRUE);}

代碼很簡單,就是根據cpc跟mpc兩個匹配器來進行匹配的,第一個是檢查類上是否有@Async注解,第二個是檢查方法是是否有@Async注解。

那么,到現在為止,我們已經知道了它在何時創建代理,會為什么對象創建代理,最后我們還需要解決一個問題,代理的邏輯是怎么樣的,異步到底是如何實現的?

通知的邏輯是怎么樣的?是如何實現異步的?

前面也提到了advisor是一個綁定了切點的通知,前面分析了它的切點,那么現在我們就來看看它的通知邏輯,直接定位到AsyncAnnotationAdvisor中的buildAdvice方法,源碼如下:

protected Advice buildAdvice( @Nullable Supplier<Executor> executor, @Nullable Supplier<AsyncUncaughtExceptionHandler> exceptionHandler) { AnnotationAsyncExecutionInterceptor interceptor = new AnnotationAsyncExecutionInterceptor(null); interceptor.configure(executor, exceptionHandler); return interceptor;}

簡單吧,加了一個攔截器而已,對于interceptor類型的對象,我們關注它的核心方法invoke就行了,代碼如下:

public Object invoke(final MethodInvocation invocation) throws Throwable { Class<?> targetClass = (invocation.getThis() != null ? AopUtils.getTargetClass(invocation.getThis()) : null); Method specificMethod = ClassUtils.getMostSpecificMethod(invocation.getMethod(), targetClass); final Method userDeclaredMethod = BridgeMethodResolver.findBridgedMethod(specificMethod); // 異步執行嘛,先獲取到一個線程池 AsyncTaskExecutor executor = determineAsyncExecutor(userDeclaredMethod); if (executor == null) { throw new IllegalStateException( 'No executor specified and no default executor set on AsyncExecutionInterceptor either'); } // 然后將這個方法封裝成一個 Callable對象傳入到線程池中執行 Callable<Object> task = () -> { try { Object result = invocation.proceed(); if (result instanceof Future) { return ((Future<?>) result).get(); } } catch (ExecutionException ex) { handleError(ex.getCause(), userDeclaredMethod, invocation.getArguments()); } catch (Throwable ex) { handleError(ex, userDeclaredMethod, invocation.getArguments()); } return null; };// 將任務提交到線程池 return doSubmit(task, executor, invocation.getMethod().getReturnType());}

導致的問題及解決方案

問題1:循環依賴報錯

就像在這張圖里這個讀者問的問題,

Spring中異步注解@Async的使用、原理及使用時可能導致的問題及解決方法

分為兩點回答:

第一:循環依賴為什么不能被解決?

這個問題其實很簡單,在《講一講Spring中的循環依賴》這篇文章中我從兩個方面分析了循環依賴的處理流程

簡單對象間的循環依賴處理AOP對象間的循環依賴處理

按照這種思路,@Async注解導致的循環依賴應該屬于AOP對象間的循環依賴,也應該能被處理。但是,重點來了,解決AOP對象間循環依賴的核心方法是三級緩存,如下:

Spring中異步注解@Async的使用、原理及使用時可能導致的問題及解決方法

在三級緩存緩存了一個工廠對象,這個工廠對象會調用getEarlyBeanReference方法來獲取一個早期的代理對象的引用,其源碼如下:

protected Object getEarlyBeanReference(String beanName, RootBeanDefinition mbd, Object bean) { Object exposedObject = bean; if (!mbd.isSynthetic() && hasInstantiationAwareBeanPostProcessors()) { for (BeanPostProcessor bp : getBeanPostProcessors()) { // 看到這個判斷了嗎,通過@EnableAsync導入的后置處理器 // AsyncAnnotationBeanPostProcessor根本就不是一個SmartInstantiationAwareBeanPostProcessor // 這就意味著即使我們通過AsyncAnnotationBeanPostProcessor創建了一個代理對象 // 但是早期暴露出去的用于給別的Bean進行注入的那個對象還是原始對象 if (bp instanceof SmartInstantiationAwareBeanPostProcessor) { SmartInstantiationAwareBeanPostProcessor ibp = (SmartInstantiationAwareBeanPostProcessor) bp; exposedObject = ibp.getEarlyBeanReference(exposedObject, beanName); } } } return exposedObject;}

看完上面的代碼循環依賴的問題就很明顯了,因為早期暴露的對象跟最終放入容器中的對象不是同一個,所以報錯了。報錯的具體位置我在談談我對Spring Bean 生命周期的理解 文章末尾已經分析過了,本文不再贅述

Spring中異步注解@Async的使用、原理及使用時可能導致的問題及解決方法

解決方案

就以上面讀者給出的Demo為例,只需要在為B注入A時添加一個@Lazy注解即可

@Componentpublic class B implements BService { @Autowired@Lazyprivate A a;public void doSomething() {}}

這個注解的作用在于,當為B注入A時,會為A生成一個代理對象注入到B中,當真正調用代理對象的方法時,底層會調用getBean(a)去創建A對象,然后調用方法,這個注解的處理時機是在org.springframework.beans.factory.support.DefaultListableBeanFactory#resolveDependency方法中,處理這個注解的代碼位于org.springframework.context.annotation.ContextAnnotationAutowireCandidateResolver#buildLazyResolutionProxy,這些代碼其實都在我之前的文章中分析過了

《Spring雜談 | Spring中的AutowireCandidateResolver》

《談談Spring中的對象跟Bean,你知道Spring怎么創建對象的嗎?》

所以本文不再做詳細分析

問題2:默認線程池不會復用線程

我覺得這是這個注解最坑的地方,沒有之一!我們來看看它默認使用的線程池是哪個,在前文的源碼分析中,我們可以看到決定要使用線程池的方法是org.springframework.aop.interceptor.AsyncExecutionAspectSupport#determineAsyncExecutor。其源碼如下:

protected AsyncTaskExecutor determineAsyncExecutor(Method method) { AsyncTaskExecutor executor = this.executors.get(method); if (executor == null) { Executor targetExecutor; // 可以在@Async注解中配置線程池的名字 String qualifier = getExecutorQualifier(method); if (StringUtils.hasLength(qualifier)) { targetExecutor = findQualifiedExecutor(this.beanFactory, qualifier); } else { // 獲取默認的線程池 targetExecutor = this.defaultExecutor.get(); } if (targetExecutor == null) { return null; } executor = (targetExecutor instanceof AsyncListenableTaskExecutor ? (AsyncListenableTaskExecutor) targetExecutor : new TaskExecutorAdapter(targetExecutor)); this.executors.put(method, executor); } return executor;}

最終會調用到org.springframework.aop.interceptor.AsyncExecutionInterceptor#getDefaultExecutor這個方法中

protected Executor getDefaultExecutor(@Nullable BeanFactory beanFactory) { Executor defaultExecutor = super.getDefaultExecutor(beanFactory); return (defaultExecutor != null ? defaultExecutor : new SimpleAsyncTaskExecutor());}

可以看到,它默認使用的線程池是SimpleAsyncTaskExecutor。我們不看這個類的源碼,只看它上面的文檔注釋,如下:

Spring中異步注解@Async的使用、原理及使用時可能導致的問題及解決方法

主要說了三點

為每個任務新起一個線程 默認線程數不做限制 不復用線程

就這三點,你還敢用嗎?只要你的任務耗時長一點,說不定服務器就給你來個OOM。

解決方案

最好的辦法就是使用自定義的線程池,主要有這么幾種配置方法

在之前的源碼分析中,我們可以知道,可以通過AsyncConfigurer來配置使用的線程池

如下:

public class DmzAsyncConfigurer implements AsyncConfigurer { @Override public Executor getAsyncExecutor() { // 創建自定義的線程池 }}

直接在@Async注解中配置要使用的線程池的名稱

如下:

public class A implements AService {private B b;@Autowiredpublic void setB(B b) {System.out.println(b);this.b = b;}@Async('dmzExecutor')public void doSomething() {}}

@EnableAsync@Configuration@ComponentScan('com.dmz.spring.async')@Aspectpublic class Config {@Bean('dmzExecutor')public Executor executor(){// 創建自定義的線程池return executor;}}

總結

本文主要介紹了Spring中異步注解的使用、原理及可能碰到的問題,針對每個問題文中也給出了方案。希望通過這篇文章能幫助你徹底掌握@Async注解的使用,知其然并知其所以然!

到此這篇關于Spring中異步注解@Async的使用、原理及使用時可能導致的問題及解決方法的文章就介紹到這了,更多相關Spring 異步注解@Async使用原理內容請搜索好吧啦網以前的文章或繼續瀏覽下面的相關文章希望大家以后多多支持好吧啦網!

標簽: Spring
相關文章:
主站蜘蛛池模板: 国产日韩中文字幕 | 视频一区久久 | 国产精品一区二区在线 | 最新日韩av | a级片网站 | 91精品国产一区二区三区免费 | 欧美视频三区 | 91,看片 | 精品亚洲国产成av人片传媒 | 亚洲一区二区高清视频 | 成人av一区二区三区 | 50人群体交乱视频 | 后人极品翘臀美女在线播放 | 亚洲国产婷婷 | av免费网 | 日韩中文字幕在线看 | 欧美成人激情视频 | 91亚洲国产精品 | 日日干日日操 | 天天干天天操 | 日本不卡一区 | 国产欧美日韩一区二区三区 | 精品国产一区二区三区久久 | 在线视频亚洲 | 国产精品美女久久久久久久久久久 | 欧美一区二区久久 | 国产激情网站 | 在线观看亚洲专区 | 超碰官网| 亚洲热在线视频 | 狠狠操综合网 | 日韩精品一区二区在线观看 | 欧美高清视频一区 | 91视频在线| 一区二区三区四区免费看 | 国产一区久久 | www国产亚洲 | 国产人妖一区二区 | 国产aⅴ| 性人久久精品 | 亚洲精品电影在线观看 | 一区二区亚洲视频 | 99视频免费在线观看 | 国产视频91在线 | 久久亚洲综合 | 中文字幕国产一区 | 国产成人精品亚洲777人妖 | 国产精品乱码一区二区三区 | 一级毛片免费 | 久久韩剧网 | 亚洲区在线 | 欧美激情精品久久久久久 | 欧美日韩三区 | 亚洲精品乱码久久久久久国产主播 | 国产综合精品一区二区三区 | 九色av | 日韩一区精品视频 | 久久精品免费视频播放 | 狠狠夜夜| 91中文在线观看 | 成人免费在线电影 | 国产艳妇av视国产精选av一区 | 羞羞视频免费观看入口 | 久久久久久久久久久久福利 | 国产成人免费网站 | 99视频精品 | 欧美日韩在线观看中文字幕 | 精品成人一区 | 免费毛片视频 | 精品一二区 | 久久男女 | 激情欧美一区二区三区中文字幕 | 国产精品亚洲一区二区三区在线 | www.亚洲精品| 国产精品久久在线观看 | 三区视频| 成人一级电影在线观看 | 伊人一区二区三区 | 精品久久久久久久久久久久包黑料 | 欧美国产日韩一区 | 可以在线观看的黄色 | 成人羞羞在线观看网站 | 欧美视频在线一区 | 91国产精品 | 国产无套丰满白嫩对白 | 在线观看亚洲大片短视频 | 日韩国产在线观看 | 国产精品永久 | 久久999免费视频 | 亚洲成人三区 | 一级毛片免费在线 | 操人网址 | 欧美a v在线播放 | 一区二区三区四区久久 | 91精品国产高清久久久久久久久 | 欧美综合久久 | 天天操夜夜操免费视频 | 99久久99热这里只有精品 | 国精日本亚洲欧州国产中文久久 | 国产精选一区二区三区不卡催乳 | 97人人超碰 | 午夜视频在线观看网址 | 一本色道精品久久一区二区三区 | 日韩激情综合 | 成人在线小视频 | 天天干人人干 | 国产免费av网站 | 日本a视频 | 欧美最猛性xxxxx亚洲精品 | 懂色一区二区三区免费观看 | 一区二区三区四区久久 | 午夜精品久久久久久久白皮肤 | 最新av在线网址 | 第一色视频 | 中文字幕亚洲一区二区三区 | 亚洲色图图片 | 黄网站涩免费蜜桃网站 | 国产高清精品在线 | 天堂一区二区三区四区 | 欧美日韩免费在线 | 男人的天堂在线视频 | 久久美女视频 | 99久精品 | 在线色网站 | 欧美日韩在线一 | 日韩城人免费 | 激情久久av一区av二区av三区 | 国产成人精品免费 | 三区在线 | 欧美午夜一区二区福利视频 | 在线一区二区免费 | 天天插天天干 | 香蕉视频成人在线观看 | 日韩综合网 | 国产精品久久久久久久天堂 | 中国妞videos高潮 | 欧美日韩中文 | 欧美国产日韩在线 | 日韩 国产 在线 | 国产成人一区二区三区影院在线 | 中文字幕av在线播放 | 精品九九 | 国产一区二区视频在线 | 欧美精品一区二区三区四区五区 | 亚洲精品久久久久一区二区三区 | 国产精品久久久久久久久小说 | 国产一区二区三区久久 | 亚洲高清视频一区 | 天天操,夜夜操 | 免费观看一级毛片 | 一级做a爰片性色毛片 | 一区二区激情 | 亚洲在线视频 | 精品日韩一区二区 | 黄色电影天堂 | 一区二区精品视频 | www久久国产| 成人片免费看 | 国产精品无码永久免费888 | 日韩一区二区三区在线观看 | 日韩在线一区二区 | 91精品电影 | 久久r免费视频 | 日本三级一区二区 | 在线看一区二区 | 色婷网 | www.色综合| 亚洲精品一区国语对白 | 蜜桃视频一区 | 欧美日视频 | 天天躁日日躁狠狠躁av麻豆 | 国产一区国产二区在线观看 | 丁香婷婷久久久综合精品国产 | 国产一级免费视频 | 亚洲精品一区二区另类图片 | 国产精品一区二区三区在线播放 | 午夜tv| 国产日韩欧美一区二区 | 久久精品99国产精品日本 | 日干夜操 | 日韩欧美一区二区视频 | 91久久综合 | 精品在线一区二区 | 姐姐在线观看动漫第二集免费 | 国产成人亚洲综合 | 国产成人在线视频 | 国产超碰人人模人人爽人人添 | caoporn国产精品免费公开 | 中文字幕黄色 | 国产精品一区二区不卡 | 久久综合久久综合久久综合 | 久久小视频 | 性欧美久久久 | 欧美日韩久久 | 久久久久无码国产精品一区 | 电影91久久久 | 天天操天天曰 | 欧美成人a| 精品无人乱码一区二区三区 | 97色综合 | 亚洲女人天堂成人av在线 | 99精品欧美一区二区三区综合在线 | 在线观看欧美成人 | 国产成人免费 | 亚洲欧美综合精品久久成人 | 男女国产网站 | 福利视频一区二区 | 久久久99久久久国产自输拍 | 午夜激情免费看 | 亚洲天堂av网 | 久久精品国产免费 | 91伦理片 | 久久久艹 | 99国产精品久久久久久久成人热 | 一级黄色录像毛片 | 亚洲免费a| 精品一区二区久久久久久久网站 | 国产aaa毛片 | 国产a久久精品一区二区三区 | 亚洲一区二区三区久久久 | 成人免费毛片高清视频 | 狠狠人人 | 国产精品99 | 三级日韩| 午夜激情视频 | 日本不卡一区二区 | 国产精品永久免费自在线观看 | 日韩城人免费 | 久草免费在线 | 欧美日韩国产高清 | 日本黄色三级网站 | 久久高清精品 | 久久精品国产v日韩v亚洲 | 91免费视频观看 | 国产无套丰满白嫩对白 | 天堂va| 精国产品一区二区三区四季综 | 久久99热精品免费观看牛牛 | 99pao成人国产永久免费视频 | 久久99国产精一区二区三区 | 久久亚洲一区二区三区四区 | 欧美精品一区二区三区在线四季 | 日本免费一区二区在线观看 | 在线激情视频 | 久久精选视频 | 日本免费三片免费观看 | 久久99国产精品久久99果冻传媒 | 日韩日韩日韩日韩日韩日韩日韩 | 中文一区 | 91成人短视频在线观看 | 综合网日韩 | 成人在线h| 国产乱视频 | 亚洲视频一区在线 | 中文字幕在线观看不卡视频 | 久久久久久久99精品免费观看 | 综合在线视频 | 天天综合网7799精品 | 色婷婷亚洲 | 海外中文字幕在线观看 | 成人免费视频网址 | 99精品一区二区三区 | 6080yy精品一区二区三区 | 狠狠狠色丁香婷婷综合久久五月 | 久久久久9999国产精品 | 精品久久久久一区二区国产 | 97视频精品 | av一道本 | 日本男人的天堂 | 伊人网亚洲 | 99久久夜色精品国产网站 | 国产精品免费一区二区三区四区 | 久久久精品一区 | 精品永久 | 成年人性视频 | 嫩呦国产一区二区三区av | 一级大片一级一大片 | 国产人妖在线 | 中文精品久久久 | 国产精品久久久久久久久免费桃花 | 国产欧美精品区一区二区三区 | 8x国产精品视频一区二区 | 97精品一区二区三区 | 亚州成人 | 日韩在线资源 | 亚洲乱码一区二区 | 君岛美绪一区二区三区在线视频 | 久草视频免费看 | 伊人精品视频在线观看 | 一区二区三区在线观看国产 | 日韩中文久久 | 色综合天天综合网国产成人网 | 91精品国产一区二区三区蜜臀 | 亚洲精品在线视频 | 久久久久中文字幕 | 欧美乱码久久久久久蜜桃 | 91精品国产人妻国产毛片在线 | 日韩福利在线 | 91亚洲成人 | 精品久久伊人 | 欧美久久久久久久久久 | 午夜黄色影院 | 欧美a v在线播放 | 欧美xxxx性 | 狠狠操av| 国产真实乱全部视频 | 日韩一二 | 亚洲国产精品99久久久久久久久 | 欧美日韩久久 | 成人中文视频 | 免费99精品国产自在在线 | 久久另类ts人妖一区二区 | 欧美一区二区三区久久精品 | 免费在线观看一级毛片 | 日韩国产免费观看 | 播放一区| 亚洲精品成人免费 | 久久久久久亚洲国产 | 久久香蕉国产 | 久草视频免费在线播放 | 成人久久久 | 91在线精品视频 | 欧美精品一区二区三区四区在线 | 免费黄色在线看 | 亚洲午夜精品 | www.天天操.com| 亚洲精品视频一区二区三区 | 国产一区二区三区在线看 | 99视频在线 | 亚洲成人首页 | 欧美一区永久视频免费观看 | 色婷婷综合久色 | 天天拍天天草 | 国产69久久 | av女人的天堂 | 精品国产成人 | 欧美精品久久 | 一区二区三区四区在线 | 电影91久久久 | 中文字幕在线播放不卡 | 国产最好的精华液网站 | 日韩在线视频第一页 | 日韩av免费在线观看 | 国产区视频在线观看 | 亚洲视频在线免费观看 | 日韩一区二区中文字幕 | 亚洲成人日韩 | av免费网站在线观看 | www.国产.com| 久久aⅴ乱码一区二区三区 91综合网 | 91观看在线视频 | 日韩视频在线观看 | 久久国产一区视频 | av毛片在线免费看 | 色综合99| 欧美一区不卡 | 亚洲精品久久久久久下一站 | 五月婷婷丁香婷婷 | 亚洲黄色一区二区三区 | 亚洲综合二区 | 久久99精品久久久久久琪琪 | 欧美伦理一区二区三区 | 一区二区三区在线免费观看 | 欧美一区二 | 欧美亚洲| aa级毛片毛片免费观看久 | 国产精品日韩欧美一区二区三区 | 亚洲精品成人av | 亚洲情网站 | 在线观看的av | 亚洲三级在线播放 | 国产一级一级国产 | 精品91| 国产精品久久免费视频 | 欧美在线播放一区 | 欧美亚洲专区 | 日本爽快片毛片 | av免费观看网页 | 日韩理伦片在线观看视频播放 | 国产欧美精品一区二区三区 | 欧美亚洲| 日本精品免费 | 色黄视频在线 | 久久久久久久一区 | 精品久久网 | 女男羞羞视频网站免费 | 久久久一区二区三区 | 丁香婷婷久久久综合精品国产 | 欧美一级全黄 | 日韩精品1区 | 日韩欧美国产精品综合嫩v 久久久久久国产精品高清 国产目拍亚洲精品99久久精品 | 香蕉成人啪国产精品视频综合网 | 久久精品欧美 | av在线影院 | 亚洲永久| 国产精品久久天天躁 | 色婷婷综合久久久中文字幕 | 久久久国产精品入口麻豆 | 一级a毛片 | 黄色免费网 | 欧美日韩国产一区二区三区不卡 | 亚洲第一se情网站 | 欧美日韩不卡合集视频 | 国产精品自在线 | 毛片免费观看 | 精品成人一区二区 | 国产精品成人在线观看 | 国产一二三四在线 | 久久久久一区二区 | 欧美国产日韩视频 | 国产精品国产三级国产aⅴ原创 | 久久一区二区三 | 色欧美日韩| 国内外成人在线视频 | 日韩欧美第一页 | 成人一区二区在线观看 | 不卡在线| 欧美a级成人淫片免费看 | 国产精品视频久久久 | 欧美激情自拍偷拍 | 色一级 | 自拍偷拍精品 | 日日日日日| 成年入口无限观看网站 | 久久精品麻豆 | 久久91精品| 青青草视频在线免费观看 | 成人情趣视频 | 亚洲综合色视频在线观看 | www.五月天婷婷 | 91视频18| 欧美成人一区二区 | 女人夜夜春 | 天堂av2020| 九九精品视频在线观看 | 亚洲欧美综合 | 国产成人一区二区三区 | 日韩成人在线网站 | 性毛片| 美女黄在线观看 | 天天天天天天天天干 | 综合伊人久久 | 久久免费精品视频 | 欧美xxxx做受欧美 | 国产精品777一区二区 | 天天av网| 欧美日本一区 | 亚洲精品视频一区 | 日韩视频网 | 精品无人乱码一区二区三区的优势 | 久久亚洲综合 | 久久97视频| 免费国产成人 | 国产精品婷婷久久久久 | 亚洲网站在线观看 | 黄色一级毛片在线观看 | 国产精品网站在线观看 | 亚洲国产精品成人 | 在线精品日韩 | 中文字幕av黄色 | 国产精品一区二区三区在线 | 91精品国产91综合久久蜜臀 | 国产剧情一区二区 | 亚洲精品专区 | 一级黄色影片在线观看 | 日韩精品免费在线视频 | 国产视频久久久 | 午夜精品久久久久久久久久久久 | 国产精品国产精品国产专区不卡 | 日韩城人网站 | 国产一区日韩欧美 | 欧美日韩视频在线第一区 | 亚洲国产精品久久久久秋霞蜜臀 | 国产一区二区三区免费观看 | 亚洲视频在线免费观看 | 亚洲日韩中文字幕 | 黄色小视频免费观看 | 国产欧美精品 | 免费国产视频 | 久久久久中文字幕 | 午夜久久| 在线色站| 日韩中文字幕在线观看 | 欧美午夜精品久久久久久蜜 | 欧美日韩精品一区二区三区在线观看 | 亚洲成人精品影视 | 国产欧美亚洲精品 | 亚洲永久免费 | 另类sb东北妇女av | 久久久网 | 欧美一级精品片在线看 | 国产精品一二三区 | 欧美另类久久 | 国产一区二区日韩 | 日韩中文字幕免费在线播放 | 欧美日韩一区二区三区在线观看 | www.中文字幕 | 亚洲成人一区二区 | 亚洲欧美一区二区三区在线 | 在线观看国产小视频 | 91精品国产高清一区二区三区 | 日韩欧美国产网站 | 夜本色| 中文字幕亚洲精品 | 国产高清精品一区二区三区 | 精品婷婷 | 欧美黄 片免费观看 | 一级片国产 | 亚洲伦理在线 | 综合久久综合 | 国产精品国产 | 久久久久久久久久国产精品 | 欧美大片在线看免费观看 | 久久一区二区三区四区 | 午夜影晥 | 日韩视频一区 | 国产一区二区在线看 | 日韩国产欧美精品 | 国产免费一区二区三区 | 九九资源站| 久久极品 | 国产亚洲精品v | 黄色国产一级视频 | 一区二区三区四区av | 天堂综合网 | 在线国产一区二区 | 一区二区免费播放 | 精品乱码一区二区 | 可以免费看黄的网站 | 成人欧美一区二区三区黑人孕妇 | 国产精品免费一区 | 国产精品国产三级国产aⅴ原创 | 国产一区二区电影 | 午夜成人免费视频 | 国产免费久久 | 日韩成人短视频 | 日韩在线观看 | 欧日韩在线视频 | 日韩电影免费在线观看中文字幕 | 亚洲综合视频 | 日韩和的一区二区 | 色毛片| 伊人久久国产 | 日韩精品一区二区三区在线 | 成人在线一区二区 | 影音先锋中文字幕在线 | 亚洲 欧美 精品 | 自拍偷拍欧美 | 手机看片369 | 亚洲网站视频 | 99久久99久久 | 亚洲日本乱码在线观看 | 99re热精品视频 | 国产精品免费观看 | 一区二区av| 久久久久久久久一区 | a级片网站 | 在线观看亚洲免费 | 国产aaaaav久久久一区二区 | 国产一级免费视频 | 国内精品视频 | 日韩在线视频中文字幕 | 亚洲精品一区在线观看 | 亚洲精品二区 | 国产成人精品一区二区三区视频 | 97伦理在线| 成人一级视频在线观看 | 亚洲一区在线日韩在线深爱 | 国产成人欧美一区二区三区的 | 国产成人在线视频 | 国产在线观看欧美 | 国产欧美精品一区aⅴ影院 毛片视频网站 | 欧美日韩中文 | 亚洲欧美日韩系列 | 欧美综合视频在线观看 | 成人亚洲网| 成人深夜福利在线观看 | 欧美第一页 | 欧美一级在线观看 | 99视频网站 | 日韩资源在线 | 99草在线视频 | 高清一区二区三区 | 夜添久久精品亚洲国产精品 | 亚洲九九 | 亚洲九九| 99国产精品99久久久久久 | 久久久久中文字幕 | 久久久久中文字幕 | 夜夜视频| 亚洲精品一区国语对白 | 欧美欧美欧美 | 久久成人精品一区二区三区 | 免费亚洲一区二区 | 欧美成人资源 | 青青草国产成人av片免费 | 天天色天天色 | 久久国产精品精品国产 | 男人天堂99 | 狠狠插天天干 | 久久99精品久久久久久琪琪 | 欧美亚洲日本 | 国产一区二区三区四区五区 | aⅴ色国产 欧美 | 欧美男人的天堂 | 欧美不卡 | 在线一级电影 | 欧美日韩成人在线观看 | 一区二区日韩 | 中文字幕av一区二区 | 久久久一区二区三区 | 大胆裸体gogo毛片免费看 | 久在线| 在线免费毛片 | 国产丝袜视频 |