3atv精品不卡视频,97人人超碰国产精品最新,中文字幕av一区二区三区人妻少妇,久久久精品波多野结衣,日韩一区二区三区精品

歡迎訪問 生活随笔!

生活随笔

當前位置: 首頁 > 编程资源 > 编程问答 >内容正文

编程问答

通过transmittable-thread-local源码理解线程池线程本地变量传递的原理

發布時間:2025/3/16 编程问答 19 豆豆
生活随笔 收集整理的這篇文章主要介紹了 通过transmittable-thread-local源码理解线程池线程本地变量传递的原理 小編覺得挺不錯的,現在分享給大家,幫大家做個參考.

前提

最近一兩個月花了很大的功夫做UCloud服務和中間件遷移到阿里云的工作,沒什么空閑時間擼文。想起很早之前寫過ThreadLocal的源碼分析相關文章,里面提到了ThreadLocal存在一個不能向預先創建的線程中進行變量傳遞的局限性,剛好有一位HSBC的技術大牛前同事提到了團隊引入了transmittable-thread-local解決了此問題。

借著這個契機,順便clone了transmittable-thread-local源碼進行分析,這篇文章會把ThreadLocal和InheritableThreadLocal的局限性分析完畢,并且從一些基本原理以及設計模式的運用分析transmittable-thread-local(下文簡稱為TTL)整套框架的實現。

如果對線程池和ThreadLocal不熟悉的話,可以先參看一下前置文章:

  • 《JUC同步器框架AbstractQueuedSynchronizer源碼圖文分析》

  • 《JUC線程池ThreadPoolExecutor源碼分析》

  • 《ThreadLocal源碼分析-黃金分割數的使用》

這篇文章前后花了兩周時間編寫,行文比價干硬,文字比較多(接近5W字),希望帶著耐心閱讀。

父子線程的變量傳遞

在Java中沒有明確給出一個API可以基于子線程實例獲取其父線程實例,有一個相對可行的方案就是在創建子線程Thread實例的時候獲取當前線程的實例,用到的API是Thread#currentThread():

1public?class?Thread?implements?Runnable?{ 2 3????//?省略其他代碼 4 5????@HotSpotIntrinsicCandidate 6????public?static?native?Thread?currentThread(); 7 8????//?省略其他代碼 9}

Thread#currentThread()方法是一個靜態本地方法,它是由JVM實現,這是在JDK中唯一可以獲取父線程實例的API。一般而言,如果想在子線程實例中得到它的父線程實例,那么需要像如下這樣操作:

1public?class?InheritableThread?{23????public?static?void?main(String[]?args)?throws?Exception{4????????//?父線程就是main線程5????????Thread?parentThread?=?Thread.currentThread();6????????Thread?childThread?=?new?Thread(()->?{7????????????System.out.println("Parent?thread?is:"?+?parentThread.getName());8????????},"childThread");9????????childThread.start(); 10????????TimeUnit.SECONDS.sleep(Long.MAX_VALUE); 11????} 12} 13//?輸出結果: 14Parent?thread?is:main

類似地,如果我們想把一個父子線程共享的變量實例傳遞,也可以這樣做:

1public?class?InheritableVars?{23????public?static?void?main(String[]?args)?throws?Exception?{4????????//?父線程就是main線程5????????Thread?parentThread?=?Thread.currentThread();6????????final?Var?var?=?new?Var();7????????var.setValue1("var1");8????????var.setValue2("var2");9????????Thread?childThread?=?new?Thread(()?->?{ 10????????????System.out.println("Parent?thread?is:"?+?parentThread.getName()); 11????????????methodFrame1(var); 12????????},?"childThread"); 13????????childThread.start(); 14????????TimeUnit.SECONDS.sleep(Long.MAX_VALUE); 15????} 16 17????private?static?void?methodFrame1(Var?var)?{ 18????????methodFrame2(var); 19????} 20 21????private?static?void?methodFrame2(Var?var)?{ 22 23????} 24 25????@Data 26????private?static?class?Var?{ 27 28????????private?Object?value1; 29????????private?Object?value2; 30????} 31}

這種做法其實是可行的,子線程調用的方法棧中的所有方法都必須顯示傳入需要從父線程傳遞過來的參數引用Var實例,這樣就會產生硬編碼問題,既不靈活也導致方法不能復用,所以才衍生出線程本地變量Thread Local,具體的實現有ThreadLocal和InheritableThreadLocal。它們兩者的基本原理是類似的,實際上所有的變量實例是緩存在線程實例的變量ThreadLocal.ThreadLocalMap中,線程本地變量實例都只是線程實例獲取ThreadLocal.ThreadLocalMap的一道橋梁:

1public?class?Thread?implements?Runnable?{23????//?省略其他代碼45????//?KEY為ThreadLocal實例,VALUE為具體的值6????ThreadLocal.ThreadLocalMap?threadLocals?=?null;78????//?KEY為InheritableThreadLocal實例,VALUE為具體的值9????ThreadLocal.ThreadLocalMap?inheritableThreadLocals?=?null; 10 11????//?省略其他代碼 12}

ThreadLocal和InheritableThreadLocal之間的區別可以結合源碼分析一下(見下一小節)。前面的分析聽起來如果覺得抽象的話,可以自己寫幾個類推敲一下,假如線程其實叫ThrowableThread,而線程本地變量叫ThrowableThreadLocal,那么它們之間的關系如下:

1public?class?Actor?{23????static?ThrowableThreadLocal?THREAD_LOCAL?=?new?ThrowableThreadLocal();45????public?static?void?main(String[]?args)?throws?Exception?{6????????ThrowableThread?throwableThread?=?new?ThrowableThread()?{78????????????@Override9????????????public?void?run()?{ 10????????????????methodFrame1(); 11????????????} 12????????}; 13????????throwableThread.start(); 14????} 15 16????private?static?void?methodFrame1()?{ 17????????THREAD_LOCAL.set("throwable"); 18????????methodFrame2(); 19????} 20 21????private?static?void?methodFrame2()?{ 22????????System.out.println(THREAD_LOCAL.get()); 23????} 24 25????/** 26?????*?這個類暫且認為是java.lang.Thread 27?????*/ 28????private?static?class?ThrowableThread?implements?Runnable?{ 29 30????????ThrowableThreadLocal.ThrowableThreadLocalMap?threadLocalMap; 31 32????????@Override 33????????public?void?run()?{ 34 35????????} 36 37????????//?這里模擬VM的實現,返回ThrowableThread自身,大家先認為不是返回NULL 38????????public?static?ThrowableThread?getCurrentThread()?{ 39//????????????return?new?ThrowableThread(); 40????????????return?null;???//?<---?假設這里在VM的實現里面返回的不是NULL而是當前的ThrowableThread 41????????} 42 43????????public?void?start()?{ 44????????????run(); 45????????} 46????} 47 48????private?static?class?ThrowableThreadLocal?{ 49 50????????public?ThrowableThreadLocal()?{ 51 52????????} 53 54????????public?void?set(Object?value)?{ 55????????????ThrowableThread?currentThread?=?ThrowableThread.getCurrentThread(); 56????????????assert?null?!=?currentThread; 57????????????ThrowableThreadLocalMap?threadLocalMap?=?currentThread.threadLocalMap; 58????????????if?(null?==?threadLocalMap)?{ 59????????????????threadLocalMap?=?currentThread.threadLocalMap?=?new?ThrowableThreadLocalMap(); 60????????????} 61????????????threadLocalMap.put(this,?value); 62????????} 63 64????????public?Object?get()?{ 65????????????ThrowableThread?currentThread?=?ThrowableThread.getCurrentThread(); 66????????????assert?null?!=?currentThread; 67????????????ThrowableThreadLocalMap?threadLocalMap?=?currentThread.threadLocalMap; 68????????????if?(null?==?threadLocalMap)?{ 69????????????????return?null; 70????????????} 71????????????return?threadLocalMap.get(this); 72????????} 73 74????????//?這里其實在ThreadLocal中用的是WeakHashMap 75????????public?static?class?ThrowableThreadLocalMap?extends?HashMap<ThrowableThreadLocal,?Object>?{ 76 77????????} 78????} 79}

上面的代碼不能運行,只是通過一個自定義的實現說明一下其中的原理和關系。

ThreadLocal和InheritableThreadLocal的局限性

InheritableThreadLocal是ThreadLocal的子類,它們之間的聯系是:兩者都是線程Thread實例獲取ThreadLocal.ThreadLocalMap的一個中間變量。區別是:兩者控制ThreadLocal.ThreadLocalMap創建的時機和通過Thread實例獲取ThreadLocal.ThreadLocalMap在Thread實例中對應的屬性并不一樣,導致兩者的功能有一點差別。通俗來說兩者的功能聯系和區別是:

  • ThreadLocal:單個線程生命周期強綁定,只能在某個線程的生命周期內對ThreadLocal進行存取,不能跨線程存取。

1public?class?ThreadLocalMain?{23????private?static?ThreadLocal<String>?TL?=?new?ThreadLocal<>();45????public?static?void?main(String[]?args)?throws?Exception?{6????????new?Thread(()?->?{7????????????methodFrame1();8????????},?"childThread").start();9????????TimeUnit.SECONDS.sleep(Long.MAX_VALUE); 10????} 11 12????private?static?void?methodFrame1()?{ 13????????TL.set("throwable"); 14????????methodFrame2(); 15????} 16 17????private?static?void?methodFrame2()?{ 18????????System.out.println(TL.get()); 19????} 20} 21//?輸出結果: 22throwable
  • InheritableThreadLocal:(1)可以無感知替代ThreadLocal的功能,當成ThreadLocal使用。(2)明確父-子線程關系的前提下,繼承(拷貝)父線程的線程本地變量緩存過的變量,而這個拷貝的時機是子線程Thread實例化時候進行的,也就是子線程實例化完畢后已經完成了InheritableThreadLocal變量的拷貝,這是一個變量傳遞的過程。

1public?class?InheritableThreadLocalMain?{23????//?此處可以嘗試替換為ThreadLocal,最后會輸出null4????static?InheritableThreadLocal<String>?ITL?=?new?InheritableThreadLocal<>();56????public?static?void?main(String[]?args)?throws?Exception?{7????????new?Thread(()?->?{8????????????//?在父線程中設置變量9????????????ITL.set("throwable"); 10????????????new?Thread(()?->?{ 11????????????????methodFrame1(); 12????????????},?"childThread").start(); 13????????},?"parentThread").start(); 14????????TimeUnit.SECONDS.sleep(Long.MAX_VALUE); 15????} 16 17????private?static?void?methodFrame1()?{ 18????????methodFrame2(); 19????} 20 21????private?static?void?methodFrame2()?{ 22????????System.out.println(ITL.get()); 23????} 24} 25//?輸出結果: 26throwable

上面提到的兩點可以具體參看ThreadLocal、InheritableThreadLocal和Thread三個類的源碼,這里筆者把一些必要的注釋和源碼段貼出:

1//?-->?java.lang.Thread類的源碼片段2public?class?Thread?implements?Runnable?{34????//?省略其他代碼?56????//?這是Thread最基本的構造函數7????private?Thread(ThreadGroup?g,?Runnable?target,?String?name,8???????????????????long?stackSize,?AccessControlContext?acc,9???????????????????boolean?inheritThreadLocals)?{1011????????//?省略其他代碼1213????????Thread?parent?=?currentThread();14????????this.group?=?g;15????????this.daemon?=?parent.isDaemon();16????????this.priority?=?parent.getPriority();17????????if?(security?==?null?||?isCCLOverridden(parent.getClass()))18????????????this.contextClassLoader?=?parent.getContextClassLoader();19????????else20????????????this.contextClassLoader?=?parent.contextClassLoader;21????????this.inheritedAccessControlContext?=22????????????????acc?!=?null???acc?:?AccessController.getContext();23????????this.target?=?target;24????????setPriority(priority);25????????//?inheritThreadLocals一般情況下為true26????????//?當前子線程實例拷貝父線程的inheritableThreadLocals屬性,創建一個新的ThreadLocal.ThreadLocalMap實例賦值到自身的inheritableThreadLocals屬性27????????if?(inheritThreadLocals?&&?parent.inheritableThreadLocals?!=?null)28????????????this.inheritableThreadLocals?=?ThreadLocal.createInheritedMap(parent.inheritableThreadLocals);29????????this.stackSize?=?stackSize;30????????this.tid?=?nextThreadID();31????}3233????//?省略其他代碼34}3536//?-->?java.lang.ThreadLocal源碼片段37public?class?ThreadLocal<T>?{3839????//?省略其他代碼?4041????public?void?set(T?value)?{42????????Thread?t?=?Thread.currentThread();43????????//?通過當前線程獲取線程實例中的threadLocals44????????ThreadLocalMap?map?=?getMap(t);45????????//?線程實例中的threadLocals為NULL,實例則創建一個ThreadLocal.ThreadLocalMap實例添加當前ThreadLocal->VALUE到ThreadLocalMap中,如果已經存在ThreadLocalMap則進行覆蓋對應的Entry46????????if?(map?!=?null)?{47????????????map.set(this,?value);48????????}?else?{49????????????createMap(t,?value);50????????}51????}5253????//?通過線程實例獲取該線程的threadLocals實例,其實是ThreadLocal.ThreadLocalMap類型的屬性54????ThreadLocalMap?getMap(Thread?t)?{55????????return?t.threadLocals;56????}575859????public?T?get()?{60????????Thread?t?=?Thread.currentThread();61????????//?通過當前線程獲取線程實例中的threadLocals,再獲取ThreadLocal.ThreadLocalMap中匹配上KEY為當前ThreadLocal實例的Entry對應的VALUE62????????ThreadLocalMap?map?=?getMap(t);63????????if?(map?!=?null)?{64????????????ThreadLocalMap.Entry?e?=?map.getEntry(this);65????????????if?(e?!=?null)?{66????????????????@SuppressWarnings("unchecked")67????????????????T?result?=?(T)e.value;68????????????????return?result;69????????????}70????????}71????????//?找不到則嘗試初始化ThreadLocal.ThreadLocalMap72????????return?setInitialValue();73????}7475????//?如果不存在ThreadLocal.ThreadLocalMap,則通過初始化initialValue()方法的返回值,構造一個ThreadLocal.ThreadLocalMap76????private?T?setInitialValue()?{77????????T?value?=?initialValue();78????????Thread?t?=?Thread.currentThread();79????????ThreadLocalMap?map?=?getMap(t);80????????if?(map?!=?null)81????????????map.set(this,?value);82????????else83????????????createMap(t,?value);84????????return?value;85????}8687????//?省略其他代碼?88}8990//?-->?java.lang.InheritableThreadLocal源碼?-?太簡單,全量貼出91public?class?InheritableThreadLocal<T>?extends?ThreadLocal<T>?{9293????//?這個方法使用在線程Thread的構造函數里面ThreadLocal.createInheritedMap(),基于父線程InheritableThreadLocal的屬性創建子線程的InheritableThreadLocal屬性,它的返回值決定了拷貝父線程的屬性時候傳入子線程的值94????protected?T?childValue(T?parentValue)?{95????????return?parentValue;96????}9798????//?覆蓋獲取線程實例中的綁定的ThreadLocalMap為Thread#inheritableThreadLocals,這個方法其實是覆蓋了ThreadLocal中對應的方法,應該加@Override注解99????ThreadLocalMap?getMap(Thread?t)?{ 100???????return?t.inheritableThreadLocals; 101????} 102 103????//?覆蓋創建ThreadLocalMap的邏輯,賦值到線程實例中的inheritableThreadLocals,而不是threadLocals,這個方法其實是覆蓋了ThreadLocal中對應的方法,應該加@Override注解 104????void?createMap(Thread?t,?T?firstValue)?{ 105????????t.inheritableThreadLocals?=?new?ThreadLocalMap(this,?firstValue); 106????} 107}

一定要注意,這里的setInitialValue()方法很重要,一個新的線程Thread實例在初始化(對于InheritableThreadLocal而言繼承父線程的線程本地變量)或者是首次調用ThreadLocal#set(),會通過此setInitialValue()方法去構造一個全新的ThreadLocal.ThreadLocalMap,會直接使用createMap()方法。

以前面提到的兩個例子,貼一個圖加深理解:

Example-1:

Example-2:

ThreadLocal、InheritableThreadLocal的最大局限性就是:無法為預先創建好(未投入使用)的線程實例傳遞變量(準確來說是首次傳遞某些場景是可行的,而后面由于線程池中的線程是復用的,無法進行更新或者修改變量的傳遞值),泛線程池Executor體系、TimerTask和ForkJoinPool等一般會預先創建(核心)線程,也就它們都是無法在線程池中由預創建的子線程執行的Runnable任務實例中使用。例如下面的方式會導致參數傳遞失敗:

1public?class?InheritableThreadForExecutor?{23????static?final?InheritableThreadLocal<String>?ITL?=?new?InheritableThreadLocal<>();4????static?final?Executor?EXECUTOR?=?Executors.newFixedThreadPool(1);56????public?static?void?main(String[]?args)?throws?Exception?{7????????ITL.set("throwable");8????????EXECUTOR.execute(()?->?{9????????????System.out.println(ITL.get()); 10????????}); 11????????ITL.set("doge"); 12????????EXECUTOR.execute(()?->?{ 13????????????System.out.println(ITL.get()); 14????????}); 15????????TimeUnit.SECONDS.sleep(Long.MAX_VALUE); 16????} 17} 18//?輸出結果: 19throwable 20throwable???#?<---?可見此處參數傳遞出現異常

首次變量傳遞成功是因為線程池中的所有子線程都是派生自main線程。

TTL的簡單使用

TTL的使用方式在它的項目README.md或者項目中的單元測試有十分詳細的介紹,先引入依賴com.alibaba:transmittable-thread-local:2.11.4,這里演示一個例子:

1//?父-子線程2public?class?TtlSample1?{34????static?TransmittableThreadLocal<String>?TTL?=?new?TransmittableThreadLocal<>();56????public?static?void?main(String[]?args)?throws?Exception?{7????????new?Thread(()?->?{8????????????//?在父線程中設置變量9????????????TTL.set("throwable"); 10????????????new?Thread(TtlRunnable.get(()?->?{ 11????????????????methodFrame1(); 12????????????}),?"childThread").start(); 13????????},?"parentThread").start(); 14????????TimeUnit.SECONDS.sleep(Long.MAX_VALUE); 15????} 16 17????private?static?void?methodFrame1()?{ 18????????methodFrame2(); 19????} 20 21????private?static?void?methodFrame2()?{ 22????????System.out.println(TTL.get()); 23????} 24} 25//?輸出: 26throwable 27 28//?線程池 29public?class?TtlSample2?{ 30 31????static?TransmittableThreadLocal<String>?TTL?=?new?TransmittableThreadLocal<>(); 32????static?final?Executor?EXECUTOR?=?Executors.newFixedThreadPool(1); 33 34????public?static?void?main(String[]?args)?throws?Exception?{ 35????????TTL.set("throwable"); 36????????EXECUTOR.execute(TtlRunnable.get(()?->?{ 37????????????System.out.println(TTL.get()); 38????????})); 39????????TTL.set("doge"); 40????????EXECUTOR.execute(TtlRunnable.get(()?->?{ 41????????????System.out.println(TTL.get()); 42????????})); 43????????TimeUnit.SECONDS.sleep(Long.MAX_VALUE); 44????} 45} 46//?輸出: 47throwable 48doge

TTL實現的基本原理

TTL設計上使用了大量的委托(Delegate),委托是C#里面的說法,對標Java的設計模式就是代理模式。舉個簡單的例子:

1@Slf4j2public?class?StaticDelegate?{34????public?static?void?main(String[]?args)?throws?Exception?{5????????new?RunnableDelegate(()?->?log.info("Hello?World!")).run();6????}78????@Slf4j9????@RequiredArgsConstructor 10????private?static?final?class?RunnableDelegate?implements?Runnable?{ 11 12????????private?final?Runnable?runnable; 13 14????????@Override 15????????public?void?run()?{ 16????????????try?{ 17????????????????log.info("Before?run..."); 18????????????????runnable.run(); 19????????????????log.info("After?run..."); 20????????????}?finally?{ 21????????????????log.info("Finally?run..."); 22????????????} 23????????} 24????} 25} 26//?輸出結果: 2723:45:27.763?[main]?INFO?club.throwable.juc.StaticDelegate$RunnableDelegate?-?Before?run... 2823:45:27.766?[main]?INFO?club.throwable.juc.StaticDelegate?-?Hello?World! 2923:45:27.766?[main]?INFO?club.throwable.juc.StaticDelegate$RunnableDelegate?-?After?run... 3023:45:27.766?[main]?INFO?club.throwable.juc.StaticDelegate$RunnableDelegate?-?Finally?run...

委托如果使用純熟的話,可以做出很多十分有用的功能,例如可以基于Micrometer去統計任務的執行時間,上報到Prometheus,然后用Grafana做監控和展示:

1//?需要引入io.micrometer:micrometer-core:${version}2@Slf4j3public?class?MeterDelegate?{45????public?static?void?main(String[]?args)?throws?Exception?{6????????Executor?executor?=?Executors.newFixedThreadPool(1);7????????Runnable?task?=?()?->?{8????????????try?{9????????????????//?模擬耗時 10????????????????Thread.sleep(1000); 11????????????}?catch?(Exception?ignore)?{ 12 13????????????} 14????????}; 15????????Map<String,?String>?tags?=?new?HashMap<>(8); 16????????tags.put("_class",?"MeterDelegate"); 17????????executor.execute(new?MicrometerDelegate(task,?"test-task",?tags)); 18????????TimeUnit.SECONDS.sleep(Long.MAX_VALUE); 19????} 20 21????@Slf4j 22????@RequiredArgsConstructor 23????private?static?final?class?MicrometerDelegate?implements?Runnable?{ 24 25????????private?final?Runnable?runnable; 26????????private?final?String?taskType; 27????????private?final?Map<String,?String>?tags; 28 29????????@Override 30????????public?void?run()?{ 31????????????long?start?=?System.currentTimeMillis(); 32????????????try?{ 33????????????????runnable.run(); 34????????????}?finally?{ 35????????????????long?end?=?System.currentTimeMillis(); 36????????????????List<Tag>?tagsList?=?Lists.newArrayList(); 37????????????????Optional.ofNullable(tags).ifPresent(x?->?x.forEach((k,?v)?->?{ 38????????????????????tagsList.add(Tag.of(k,?v)); 39????????????????})); 40????????????????Metrics.summary(taskType,?tagsList).record(end?-?start); 41????????????} 42????????} 43????} 44}

委托理論上只要不線程棧溢出,可以無限層級地包裝,有點像洋蔥的結構,原始的目標方法會被包裹在最里面并且最后執行:

1????public?static?void?main(String[]?args)?throws?Exception?{2????????Runnable?target?=?()?->?log.info("target");3????????Delegate?level1?=?new?Delegate(target);4????????Delegate?level2?=?new?Delegate(level1);5????????Delegate?level3?=?new?Delegate(level2);6????????//?......7????}89????@RequiredArgsConstructor 10????static?class?Delegate?implements?Runnable{ 11 12????????private?final?Runnable?runnable; 13 14????????@Override 15????????public?void?run()?{ 16????????????runnable.run(); 17????????} 18????}

當然,委托的層級越多,代碼結構就會越復雜,不利于理解和維護。多層級委托這個洋蔥結構,再配合Java反射API剝離對具體方法調用的依賴,就是Java中切面編程的普遍原理,spring-aop就是這樣實現的。委托如果再結合Agent和字節碼增強(使用ASM、Javassist等),可以實現類加載時期替換對應的Runnable、Callable或者一般接口的實現,這樣就能無感知完成了增強功能。此外,TTL中還使用了模板方法模式,如:

1@Slf4j2public?class?TemplateMethod?{34????public?static?void?main(String[]?args)?throws?Exception?{5????????Runnable?runnable?=?()?->?log.info("Hello?World!");6????????Template?template?=?new?Template(runnable)?{7????????????@Override8????????????protected?void?beforeExecute()?{9????????????????log.info("BeforeExecute..."); 10????????????} 11 12????????????@Override 13????????????protected?void?afterExecute()?{ 14????????????????log.info("AfterExecute..."); 15????????????} 16????????}; 17????????template.run(); 18????} 19 20????@RequiredArgsConstructor 21????static?abstract?class?Template?implements?Runnable?{ 22 23????????private?final?Runnable?runnable; 24 25????????protected?void?beforeExecute()?{ 26 27????????} 28 29????????@Override 30????????public?void?run()?{ 31????????????beforeExecute(); 32????????????runnable.run(); 33????????????afterExecute(); 34????????} 35 36????????protected?void?afterExecute()?{ 37 38????????} 39????} 40} 41//?輸出結果: 4200:25:32.862?[main]?INFO?club.throwable.juc.TemplateMethod?-?BeforeExecute... 4300:25:32.865?[main]?INFO?club.throwable.juc.TemplateMethod?-?Hello?World! 4400:25:32.865?[main]?INFO?club.throwable.juc.TemplateMethod?-?AfterExecute...

分析了兩種設計模式,下面簡單理解一下TTL實現的偽代碼:

1#?TTL?extends?InheritableThreadLocal 2#?Holder?of?TTL?->?InheritableThreadLocal<WeakHashMap<TransmittableThreadLocal<Object>,??>>?[??=>?NULL] 3(1)創建一個全局的Holder,用于保存父線程(或者明確了父線程的子線程)的TTL對象,這里注意,是TTL對象,Holder是當作Set使用 4(2)(父)線程A中使用了TTL,則所有設置的變量會被TTL捕獲 5(3)(子)線程B使用了TtlRunnable(Runnable的TTL實現,使用了前面提到的委托,像Callable的實現是TtlCallable),會重放所有存儲在TTL中的,來自于線程A的存儲變量 6(4)線程B重放完畢后,清理線程B獨立產生的ThreadLocal變量,歸還變TTL的變量

主要就是這幾步,里面的話術有點抽象,后面一節分析源碼的時候會詳細講解。

TTL的源碼分析

主要分析:

  • 框架的骨架。

  • 核心類TransmittableThreadLocal。

  • 發射器Transmitter。

  • 捕獲、重放和復原。

  • Agent模塊。


TTL框架骨架

TTL是一個十分精悍的框架,它依賴少量的類實現了比較強大的功能,除了提供給用戶使用的API,還提供了基于Agent和字節碼增強實現了無感知增強泛線程池對應類的功能,這一點是比較驚艷的。這里先分析編程式的API,再簡單分析Agent部分的實現。筆者閱讀TTL框架的時間是2020年五一勞動節前后,當前的最新發行版本為2.11.4。TTL的項目結構很簡單:

1-?transmittable-thread-local 2??-?com.alibaba.ttl 3???-?spi???SPI接口和一些實現 4???-?threadpool???線程池增強,包括ThreadFactory和線程池的Wrapper等 5?????-?agent???線程池的Agent實現相關 6???最外層的包有一些Wrapper的實現和TTL

先看spi包:

1-?spi 2??TtlAttachments 3??TtlAttachmentsDelegate 4??TtlEnhanced 5??TtlWrapper

TtlEnhanced是TTL的標識接口(空接口),標識具體的組件被TTL增強:

1public?interface?TtlEnhanced?{ 2 3}

通過instanceof關鍵字就可以判斷具體的實現是否TTL增強過的組件。TtlWrapper接口繼承自接口TtlEnhanced,用于標記實現類可以解包裝獲得原始實例:

1public?interface?TtlWrapper<T>?extends?TtlEnhanced?{ 2 3????//?返回解包裝實例,實際是就是原始實例 4????@NonNull 5????T?unwrap(); 6}

TtlAttachments接口也是繼承自接口TtlEnhanced,用于為TTL添加K-V結構的附件,TtlAttachmentsDelegate是其實現類,K-V的存儲實際上是委托給ConcurrentHashMap:

1public?interface?TtlAttachments?extends?TtlEnhanced?{23????//?添加K-V附件4????void?setTtlAttachment(@NonNull?String?key,?Object?value);56????//?通過KEY獲取值7????<T>?T?getTtlAttachment(@NonNull?String?key);89????//?標識自動包裝的KEY,Agent模式會使用自動包裝,這個時候會傳入一個附件的K-V,其中KEY就是KEY_IS_AUTO_WRAPPER 10????String?KEY_IS_AUTO_WRAPPER?=?"ttl.is.auto.wrapper"; 11} 12 13//?TtlAttachmentsDelegate 14public?class?TtlAttachmentsDelegate?implements?TtlAttachments?{ 15 16????private?final?ConcurrentMap<String,?Object>?attachments?=?new?ConcurrentHashMap<String,?Object>(); 17 18????@Override 19????public?void?setTtlAttachment(@NonNull?String?key,?Object?value)?{ 20????????attachments.put(key,?value); 21????} 22 23????@Override 24????@SuppressWarnings("unchecked") 25????public?<T>?T?getTtlAttachment(@NonNull?String?key)?{ 26????????return?(T)?attachments.get(key); 27????} 28}

因為TTL的實現覆蓋了泛線程池Executor、ExecutorService、ScheduledExecutorService、ForkJoinPool和TimerTask(在TTL中組件已經標記為過期,推薦使用ScheduledExecutorService),范圍比較廣,短篇幅無法分析所有的源碼,而且它們的實現思路是基本一致的,筆者下文只會挑選Executor的實現路線進行分析。

核心類TransmittableThreadLocal

TransmittableThreadLocal是TTL的核心類,TTL框架就是用這個類來命名的。先看它的構造函數和關鍵屬性:

1//?函數式接口,TTL拷貝器2@FunctionalInterface3public?interface?TtlCopier<T>?{45????//?拷貝父屬性6????T?copy(T?parentValue);7}89public?class?TransmittableThreadLocal<T>?extends?InheritableThreadLocal<T>?implements?TtlCopier<T>?{ 10 11????//?日志句柄,使用的不是SLF4J的接口,而是java.util.logging的實現 12????private?static?final?Logger?logger?=?Logger.getLogger(TransmittableThreadLocal.class.getName()); 13 14????//?是否禁用忽略NULL值的語義 15????private?final?boolean?disableIgnoreNullValueSemantics; 16 17????//?默認是false,也就是不禁用忽略NULL值的語義,也就是忽略NULL值,也就是默認的話,NULL值傳入不會覆蓋原來已經存在的值 18????public?TransmittableThreadLocal()?{ 19????????this(false); 20????} 21 22????//?可以通過手動設置,去覆蓋IgnoreNullValue的語義,如果設置為true,則是支持NULL值的設置,設置為true的時候,與ThreadLocal的語義一致 23????public?TransmittableThreadLocal(boolean?disableIgnoreNullValueSemantics)?{ 24????????this.disableIgnoreNullValueSemantics?=?disableIgnoreNullValueSemantics; 25????} 26 27????//?先忽略其他代碼 28}

disableIgnoreNullValueSemantics屬性相關可以查看Issue157,下文分析方法的時候也會說明具體的場景。TransmittableThreadLocal繼承自InheritableThreadLocal,本質就是ThreadLocal,那它到底怎么樣保證變量可以在線程池中的線程傳遞?接著分析其他所有方法:

1public?class?TransmittableThreadLocal<T>?extends?InheritableThreadLocal<T>?implements?TtlCopier<T>?{23????//?拷貝器的拷貝方法實現4????public?T?copy(T?parentValue)?{5????????return?parentValue;6????}78????//?模板方法,留給子類實現,在TtlRunnable或者TtlCallable執行前回調9????protected?void?beforeExecute()?{10????}1112????//?模板方法,留給子類實現,在TtlRunnable或者TtlCallable執行后回調13????protected?void?afterExecute()?{14????}1516????//?獲取值,直接從InheritableThreadLocal#get()獲取17????@Override18????public?final?T?get()?{19????????T?value?=?super.get();20????????//?如果值不為NULL?或者?禁用了忽略空值的語義(也就是和ThreadLocal語義一致),則重新添加TTL實例自身到存儲器21????????if?(disableIgnoreNullValueSemantics?||?null?!=?value)?addThisToHolder();22????????return?value;23????}2425????@Override26????public?final?void?set(T?value)?{27????????//?如果不禁用忽略空值的語義,也就是需要忽略空值,并且設置的入參值為空,則做一次徹底的移除,包括從存儲器移除TTL自身實例,TTL(ThrealLocalMap)中也移除對應的值28????????if?(!disableIgnoreNullValueSemantics?&&?null?==?value)?{29????????????//?may?set?null?to?remove?value30????????????remove();31????????}?else?{32????????????//?TTL(ThrealLocalMap)中設置對應的值33????????????super.set(value);34????????????//?添加TTL實例自身到存儲器35????????????addThisToHolder();36????????}37????}3839????//?從存儲器移除TTL自身實例,從TTL(ThrealLocalMap)中移除對應的值40????@Override41????public?final?void?remove()?{42????????removeThisFromHolder();43????????super.remove();44????}4546????//?從TTL(ThrealLocalMap)中移除對應的值47????private?void?superRemove()?{48????????super.remove();49????}5051????//?拷貝值,主要是拷貝get()的返回值52????private?T?copyValue()?{53????????return?copy(get());54????}5556????//?存儲器,本身就是一個InheritableThreadLocal(ThreadLocal)57????//?它的存放對象是WeakHashMap<TransmittableThreadLocal<Object>,??>類型,而WeakHashMap的VALUE總是為NULL,這里當做Set容器使用,WeakHashMap支持NULL值58????private?static?InheritableThreadLocal<WeakHashMap<TransmittableThreadLocal<Object>,??>>?holder?=59????????????new?InheritableThreadLocal<WeakHashMap<TransmittableThreadLocal<Object>,??>>()?{60????????????????@Override61????????????????protected?WeakHashMap<TransmittableThreadLocal<Object>,??>?initialValue()?{62????????????????????return?new?WeakHashMap<TransmittableThreadLocal<Object>,?Object>();63????????????????}6465????????????????@Override66????????????????protected?WeakHashMap<TransmittableThreadLocal<Object>,??>?childValue(WeakHashMap<TransmittableThreadLocal<Object>,??>?parentValue)?{67????????????????????//?注意這里的WeakHashMap總是拷貝父線程的值68????????????????????return?new?WeakHashMap<TransmittableThreadLocal<Object>,?Object>(parentValue);69????????????????}70????????????};7172????//?添加TTL自身實例到存儲器,不存在則添加策略73????@SuppressWarnings("unchecked")74????private?void?addThisToHolder()?{75????????if?(!holder.get().containsKey(this))?{76????????????holder.get().put((TransmittableThreadLocal<Object>)?this,?null);?//?WeakHashMap?supports?null?value.77????????}78????}7980????//?從存儲器移除TTL自身的實例81????private?void?removeThisFromHolder()?{82????????holder.get().remove(this);83????}8485????//?執行目標方法,isBefore決定回調beforeExecute還是afterExecute,注意此回調方法會吞掉所有的異常只打印日志86????private?static?void?doExecuteCallback(boolean?isBefore)?{87????????for?(TransmittableThreadLocal<Object>?threadLocal?:?holder.get().keySet())?{88????????????try?{89????????????????if?(isBefore)?threadLocal.beforeExecute();90????????????????else?threadLocal.afterExecute();91????????????}?catch?(Throwable?t)?{92????????????????if?(logger.isLoggable(Level.WARNING))?{93????????????????????logger.log(Level.WARNING,?"TTL?exception?when?"?+?(isBefore???"beforeExecute"?:?"afterExecute")?+?",?cause:?"?+?t.toString(),?t);94????????????????}95????????????}96????????}97????}9899????//?DEBUG模式下打印TTL里面的所有值 100????static?void?dump(@Nullable?String?title)?{ 101????????if?(title?!=?null?&&?title.length()?>?0)?{ 102????????????System.out.printf("Start?TransmittableThreadLocal[%s]?Dump...%n",?title); 103????????}?else?{ 104????????????System.out.println("Start?TransmittableThreadLocal?Dump..."); 105????????} 106 107????????for?(TransmittableThreadLocal<Object>?threadLocal?:?holder.get().keySet())?{ 108????????????System.out.println(threadLocal.get()); 109????????} 110????????System.out.println("TransmittableThreadLocal?Dump?end!"); 111????} 112 113????//?DEBUG模式下打印TTL里面的所有值 114????static?void?dump()?{ 115????????dump(null); 116????} 117 118????//?省略靜態類Transmitter的實現代碼 119}

這里一定要記住holder是全局靜態的,并且它自身也是一個InheritableThreadLocal(get()方法也是線程隔離的),它實際上就是線程管理所有TransmittableThreadLocal的橋梁。這里可以考慮一個單線程的例子來說明TransmittableThreadLocal的存儲架構:

1public?class?TtlSample3?{23????static?TransmittableThreadLocal<String>?TTL1?=?new?TransmittableThreadLocal<>();4????static?TransmittableThreadLocal<String>?TTL2?=?new?TransmittableThreadLocal<>();5????static?TransmittableThreadLocal<String>?TTL3?=?new?TransmittableThreadLocal<>();67????public?static?void?main(String[]?args)?throws?Exception?{8????????TTL1.set("VALUE-1");9????????TTL2.set("VALUE-2"); 10????????TTL3.set("VALUE-3"); 11????} 12}

這里簡化了例子,只演示了單線程的場景,圖中的一些對象的哈希碼有可能每次啟動JVM實例都不一樣,這里只是做示例:

注釋里面也提到,holder里面的WeakHashMap是當成Set容器使用,映射的值都是NULL,每次遍歷它的所有KEY就能獲取holder里面的所有的TransmittableThreadLocal實例,它是一個全局的存儲器,但是本身是一個InheritableThreadLocal,多線程共享后的映射關系會相對復雜:

再聊一下disableIgnoreNullValueSemantics的作用,默認情況下disableIgnoreNullValueSemantics=false,TTL如果設置NULL值,會直接從holder移除對應的TTL實例,在TTL#get()方法被調用的時候,如果原來持有的屬性不為NULL,該TTL實例會重新加到holder。如果設置disableIgnoreNullValueSemantics=true,則set(null)的語義和ThreadLocal一致。見下面的例子:

1public?class?TtlSample4?{23????static?TransmittableThreadLocal<Integer>?TL1?=?new?TransmittableThreadLocal<Integer>(false)?{4????????@Override5????????protected?Integer?initialValue()?{6????????????return?5;7????????}89????????@Override 10????????protected?Integer?childValue(Integer?parentValue)?{ 11????????????return?10; 12????????} 13????}; 14 15????static?TransmittableThreadLocal<Integer>?TL2?=?new?TransmittableThreadLocal<Integer>(true)?{ 16????????@Override 17????????protected?Integer?initialValue()?{ 18????????????return?5; 19????????} 20 21????????@Override 22????????protected?Integer?childValue(Integer?parentValue)?{ 23????????????return?10; 24????????} 25????}; 26 27????public?static?void?main(String[]?args)?throws?Exception?{ 28????????TL1.set(null); 29????????TL2.set(null); 30????????Thread?t1?=?new?Thread(TtlRunnable.get(()?->?{ 31????????????System.out.println(String.format("Thread:%s,value:%s",?Thread.currentThread().getName(),?TL1.get())); 32????????}),?"T1"); 33 34????????Thread?t2?=?new?Thread(TtlRunnable.get(()?->?{ 35????????????System.out.println(String.format("Thread:%s,value:%s",?Thread.currentThread().getName(),?TL2.get())); 36????????}),?"T2"); 37????????t1.start(); 38????????t2.start(); 39????????TimeUnit.SECONDS.sleep(Long.MAX_VALUE); 40????} 41} 42//?輸出結果: 43Thread:T2,value:null 44Thread:T1,value:5

這是因為框架的設計者不想把NULL作為有狀態的值,如果真的有需要保持和ThreadLocal一致的用法,可以在構造TransmittableThreadLocal實例的時候傳入true。

發射器Transmitter

發射器Transmitter是TransmittableThreadLocal的一個公有靜態類,它的核心功能是傳輸所有的TransmittableThreadLocal實例和提供靜態方法注冊當前線程的變量到其他線程。按照筆者閱讀源碼的習慣,先看構造函數和關鍵屬性:

1//?#?TransmittableThreadLocal#Transmitter2public?static?class?Transmitter?{34????//?保存手動注冊的ThreadLocal->TtlCopier映射,這里是因為部分API提供了TtlCopier給用戶實現5????private?static?volatile?WeakHashMap<ThreadLocal<Object>,?TtlCopier<Object>>?threadLocalHolder?=?new?WeakHashMap<ThreadLocal<Object>,?TtlCopier<Object>>();6????//?threadLocalHolder更變時候的監視器7????private?static?final?Object?threadLocalHolderUpdateLock?=?new?Object();8????//?標記WeakHashMap中的ThreadLocal的對應值為NULL的屬性,便于后面清理9????private?static?final?Object?threadLocalClearMark?=?new?Object(); 10 11????//?默認的拷貝器,影子拷貝,直接返回父值 12????private?static?final?TtlCopier<Object>?shadowCopier?=?new?TtlCopier<Object>()?{ 13????????@Override 14????????public?Object?copy(Object?parentValue)?{ 15????????????return?parentValue; 16????????} 17????}; 18 19????//?私有構造,說明只能通過靜態方法提供外部調用 20????private?Transmitter()?{ 21????????throw?new?InstantiationError("Must?not?instantiate?this?class"); 22????} 23 24????//?私有靜態類,快照,保存從holder中捕獲的所有TransmittableThreadLocal和外部手動注冊保存在threadLocalHolder的ThreadLocal的K-V映射快照 25????private?static?class?Snapshot?{ 26????????final?WeakHashMap<TransmittableThreadLocal<Object>,?Object>?ttl2Value; 27????????final?WeakHashMap<ThreadLocal<Object>,?Object>?threadLocal2Value; 28 29????????private?Snapshot(WeakHashMap<TransmittableThreadLocal<Object>,?Object>?ttl2Value,?WeakHashMap<ThreadLocal<Object>,?Object>?threadLocal2Value)?{ 30????????????this.ttl2Value?=?ttl2Value; 31????????????this.threadLocal2Value?=?threadLocal2Value; 32????????} 33????} 34}

Transmitter在設計上是一個典型的工具類,外部只能調用其公有靜態方法。接著看其他靜態方法:

1//?#?TransmittableThreadLocal#Transmitter2public?static?class?Transmitter?{34????//#########################################?捕獲?###########################################################56????//?捕獲當前線程綁定的所有的TransmittableThreadLocal和已經注冊的ThreadLocal的值?-?使用了用時拷貝快照的策略7????//?筆者注:它一般在構造任務實例的時候被調用,因此當前線程相對于子線程或者線程池的任務就是父線程,其實本質是捕獲父線程的所有線程本地變量的值8????@NonNull9????public?static?Object?capture()?{10????????return?new?Snapshot(captureTtlValues(),?captureThreadLocalValues());11????}1213????//?新建一個WeakHashMap,遍歷TransmittableThreadLocal#holder中的所有TransmittableThreadLocal的Entry,獲取K-V,存放到這個新的WeakHashMap返回14????private?static?WeakHashMap<TransmittableThreadLocal<Object>,?Object>?captureTtlValues()?{15????????WeakHashMap<TransmittableThreadLocal<Object>,?Object>?ttl2Value?=?new?WeakHashMap<TransmittableThreadLocal<Object>,?Object>();16????????for?(TransmittableThreadLocal<Object>?threadLocal?:?holder.get().keySet())?{17????????????ttl2Value.put(threadLocal,?threadLocal.copyValue());18????????}19????????return?ttl2Value;20????}2122????//?新建一個WeakHashMap,遍歷threadLocalHolder中的所有ThreadLocal的Entry,獲取K-V,存放到這個新的WeakHashMap返回23????private?static?WeakHashMap<ThreadLocal<Object>,?Object>?captureThreadLocalValues()?{24????????final?WeakHashMap<ThreadLocal<Object>,?Object>?threadLocal2Value?=?new?WeakHashMap<ThreadLocal<Object>,?Object>();25????????for?(Map.Entry<ThreadLocal<Object>,?TtlCopier<Object>>?entry?:?threadLocalHolder.entrySet())?{26????????????final?ThreadLocal<Object>?threadLocal?=?entry.getKey();27????????????final?TtlCopier<Object>?copier?=?entry.getValue();28????????????threadLocal2Value.put(threadLocal,?copier.copy(threadLocal.get()));29????????}30????????return?threadLocal2Value;31????}3233????//#########################################?重放?###########################################################3435????//?重放capture()方法中捕獲的TransmittableThreadLocal和手動注冊的ThreadLocal中的值,本質是重新拷貝holder中的所有變量,生成新的快照36????//?筆者注:重放操作一般會在子線程或者線程池中的線程的任務執行的時候調用,因此此時的holder#get()拿到的是子線程的原來就存在的本地線程變量,重放操作就是把這些子線程原有的本地線程變量備份37????@NonNull38????public?static?Object?replay(@NonNull?Object?captured)?{39????????final?Snapshot?capturedSnapshot?=?(Snapshot)?captured;40????????return?new?Snapshot(replayTtlValues(capturedSnapshot.ttl2Value),?replayThreadLocalValues(capturedSnapshot.threadLocal2Value));41????}4243????//?重放所有的TTL的值44????@NonNull45????private?static?WeakHashMap<TransmittableThreadLocal<Object>,?Object>?replayTtlValues(@NonNull?WeakHashMap<TransmittableThreadLocal<Object>,?Object>?captured)?{46????????//?新建一個新的備份WeakHashMap,其實也是一個快照47????????WeakHashMap<TransmittableThreadLocal<Object>,?Object>?backup?=?new?WeakHashMap<TransmittableThreadLocal<Object>,?Object>();48????????//?這里的循環針對的是子線程,用于獲取的是子線程的所有線程本地變量49????????for?(final?Iterator<TransmittableThreadLocal<Object>>?iterator?=?holder.get().keySet().iterator();?iterator.hasNext();?)?{50????????????TransmittableThreadLocal<Object>?threadLocal?=?iterator.next();5152????????????//?拷貝holder當前線程(子線程)綁定的所有TransmittableThreadLocal的K-V結構到備份中53????????????backup.put(threadLocal,?threadLocal.get());5455????????????//?清理所有的非捕獲快照中的TTL變量,以防有中間過程引入的額外的TTL變量(除了父線程的本地變量)影響了任務執行后的重放操作56????????????//?簡單來說就是:移除所有子線程的不包含在父線程捕獲的線程本地變量集合的中所有子線程本地變量和對應的值57????????????/**58?????????????*?這個問題可以舉個簡單的例子:59?????????????*?static?TransmittableThreadLocal<Integer>?TTL?=?new?TransmittableThreadLocal<>();60?????????????*?61?????????????*?線程池中的子線程C中原來初始化的時候,在線程C中綁定了TTL的值為10087,C線程是核心線程不會主動銷毀。62?????????????*?63?????????????*?父線程P在沒有設置TTL值的前提下,調用了線程C去執行任務,那么在C線程的Runnable包裝類中通過TTL#get()就會獲取到10087,顯然是不符合預期的64?????????????*65?????????????*?所以,在C線程的Runnable包裝類之前之前,要從C線程的線程本地變量,移除掉不包含在父線程P中的所有線程本地變量,確保Runnable包裝類執行期間只能拿到父線程中捕獲到的線程本地變量66?????????????*67?????????????*?下面這個判斷和移除做的就是這個工作68?????????????*/69????????????if?(!captured.containsKey(threadLocal))?{70????????????????iterator.remove();71????????????????threadLocal.superRemove();72????????????}73????????}7475????????//?重新設置TTL的值到捕獲的快照中76????????//?其實真實的意圖是:把從父線程中捕獲的所有線程本地變量重寫設置到TTL中,本質上,子線程holder里面的TTL綁定的值會被刷新77????????setTtlValuesTo(captured);7879????????//?回調模板方法beforeExecute80????????doExecuteCallback(true);8182????????return?backup;83????}8485????//?提取WeakHashMap中的KeySet,遍歷所有的TransmittableThreadLocal,重新設置VALUE86????private?static?void?setTtlValuesTo(@NonNull?WeakHashMap<TransmittableThreadLocal<Object>,?Object>?ttlValues)?{87????????for?(Map.Entry<TransmittableThreadLocal<Object>,?Object>?entry?:?ttlValues.entrySet())?{88????????????TransmittableThreadLocal<Object>?threadLocal?=?entry.getKey();89????????????//?重新設置TTL值,本質上,當前線程(子線程)holder里面的TTL綁定的值會被刷新90????????????threadLocal.set(entry.getValue());91????????}92????}9394????//?重放所有的手動注冊的ThreadLocal的值95????private?static?WeakHashMap<ThreadLocal<Object>,?Object>?replayThreadLocalValues(@NonNull?WeakHashMap<ThreadLocal<Object>,?Object>?captured)?{96????????//?新建備份97????????final?WeakHashMap<ThreadLocal<Object>,?Object>?backup?=?new?WeakHashMap<ThreadLocal<Object>,?Object>();98????????//?注意這里是遍歷捕獲的快照中的ThreadLocal99????????for?(Map.Entry<ThreadLocal<Object>,?Object>?entry?:?captured.entrySet())?{ 100????????????final?ThreadLocal<Object>?threadLocal?=?entry.getKey(); 101????????????//?添加到備份中 102????????????backup.put(threadLocal,?threadLocal.get()); 103????????????final?Object?value?=?entry.getValue(); 104????????????//?如果值為清除標記則綁定在當前線程的變量進行remove,否則設置值覆蓋 105????????????if?(value?==?threadLocalClearMark)?threadLocal.remove(); 106????????????else?threadLocal.set(value); 107????????} 108????????return?backup; 109????} 110 111????//?從relay()或者clear()方法中恢復TransmittableThreadLocal和手工注冊的ThreadLocal的值對應的備份 112????//?筆者注:恢復操作一般會在子線程或者線程池中的線程的任務執行的時候調用 113????public?static?void?restore(@NonNull?Object?backup)?{ 114????????final?Snapshot?backupSnapshot?=?(Snapshot)?backup; 115????????restoreTtlValues(backupSnapshot.ttl2Value); 116????????restoreThreadLocalValues(backupSnapshot.threadLocal2Value); 117????} 118 119????private?static?void?restoreTtlValues(@NonNull?WeakHashMap<TransmittableThreadLocal<Object>,?Object>?backup)?{ 120????????//?回調模板方法afterExecute 121????????doExecuteCallback(false); 122????????//?這里的循環針對的是子線程,用于獲取的是子線程的所有線程本地變量 123????????for?(final?Iterator<TransmittableThreadLocal<Object>>?iterator?=?holder.get().keySet().iterator();?iterator.hasNext();?)?{ 124????????????TransmittableThreadLocal<Object>?threadLocal?=?iterator.next(); 125????????????//?如果子線程原來就綁定的線程本地變量的值,如果不包含某個父線程傳來的對象,那么就刪除 126????????????//?這一步可以結合前面reply操作里面的方法段一起思考,如果不刪除的話,就相當于子線程的原來存在的線程本地變量綁定值被父線程對應的值污染了 127????????????if?(!backup.containsKey(threadLocal))?{ 128????????????????iterator.remove(); 129????????????????threadLocal.superRemove(); 130????????????} 131????????} 132 133????????//?重新設置TTL的值到捕獲的快照中 134????????//?其實真實的意圖是:把子線程的線程本地變量恢復到reply()的備份(前面的循環已經做了父線程捕獲變量的判斷),本質上,等于把holder中綁定于子線程本地變量的部分恢復到reply操作之前的狀態 135????????setTtlValuesTo(backup); 136????} 137 138????//?恢復所有的手動注冊的ThreadLocal的值 139????private?static?void?restoreThreadLocalValues(@NonNull?WeakHashMap<ThreadLocal<Object>,?Object>?backup)?{ 140????????for?(Map.Entry<ThreadLocal<Object>,?Object>?entry?:?backup.entrySet())?{ 141????????????final?ThreadLocal<Object>?threadLocal?=?entry.getKey(); 142????????????threadLocal.set(entry.getValue()); 143????????} 144????} 145}???

這里三個核心方法,看起來比較抽象,要結合多線程的場景和一些空間想象進行推敲才能比較容易地理解:

  • capture():捕獲操作,父線程原來就存在的線程本地變量映射和手動注冊的線程本地變量映射捕獲,得到捕獲的快照值captured。

  • reply():重放操作,子線程原來就存在的線程本地變量映射和手動注冊的線程本地變量生成備份backup,刷新captured的所有值到子線程在全局存儲器holder中綁定的值。

  • restore():復原操作,子線程原來就存在的線程本地變量映射和手動注冊的線程本地變量恢復成backup。

setTtlValuesTo()這個方法比較隱蔽,要特別要結合多線程和空間思維去思考,例如當入參是captured,本質是從父線程捕獲到的綁定在父線程的所有線程本地變量,調用的時機在reply()和restore(),這兩個方法只會在子線程中調用,setTtlValuesTo()里面拿到的TransmittableThreadLocal實例調用set()方法相當于把綁定在父線程的所有線程本地變量的值全部刷新到子線程當前綁定的TTL中的線程本地變量的值,更深層次地想,是基于外部的傳入值刷新了子線程綁定在全局存儲器holder里面綁定到該子線程的線程本地變量的值。

Transmitter還有不少靜態工具方法,這里不做展開,可以參考項目里面的測試demo和README.md進行調試。

捕獲、重放和復原

其實上面一節已經介紹了Transmitter提供的捕獲、重放和復原的API,這一節主要結合分析TtlRunnable中的相關邏輯。TtlRunnable的源碼如下:

1public?final?class?TtlRunnable?implements?Runnable,?TtlWrapper<Runnable>,?TtlEnhanced,?TtlAttachments?{23????//?存放從父線程捕獲得到的線程本地變量映射的備份4????private?final?AtomicReference<Object>?capturedRef;5????//?原始的Runable實例6????private?final?Runnable?runnable;7????//?執行之后是否釋放TTL值引用8????private?final?boolean?releaseTtlValueReferenceAfterRun;9 10????private?TtlRunnable(@NonNull?Runnable?runnable,?boolean?releaseTtlValueReferenceAfterRun)?{ 11????????//?這里關鍵點:TtlRunnable實例化的時候就已經進行了線程本地變量的捕獲,所以一定是針對父線程的,因為此時任務還沒提交到線程池 12????????this.capturedRef?=?new?AtomicReference<Object>(capture()); 13????????this.runnable?=?runnable; 14????????this.releaseTtlValueReferenceAfterRun?=?releaseTtlValueReferenceAfterRun; 15????} 16 17????@Override 18????public?void?run()?{ 19????????//?獲取父線程捕獲到的線程本地變量映射的備份,做一些前置判斷 20????????Object?captured?=?capturedRef.get(); 21????????if?(captured?==?null?||?releaseTtlValueReferenceAfterRun?&&?!capturedRef.compareAndSet(captured,?null))?{ 22????????????throw?new?IllegalStateException("TTL?value?reference?is?released?after?run!"); 23????????} 24????????//?重放操作 25????????Object?backup?=?replay(captured); 26????????try?{ 27????????????//?真正的Runnable調用 28????????????runnable.run(); 29????????}?finally?{ 30????????????//?復原操作 31????????????restore(backup); 32????????} 33????} 34 35????@Nullable 36????public?static?TtlRunnable?get(@Nullable?Runnable?runnable)?{ 37????????return?get(runnable,?false,?false); 38????} 39 40????@Nullable 41????public?static?TtlRunnable?get(@Nullable?Runnable?runnable,?boolean?releaseTtlValueReferenceAfterRun,?boolean?idempotent)?{ 42????????if?(null?==?runnable)?return?null; 43????????if?(runnable?instanceof?TtlEnhanced)?{ 44????????????//?avoid?redundant?decoration,?and?ensure?idempotency 45????????????if?(idempotent)?return?(TtlRunnable)?runnable; 46????????????else?throw?new?IllegalStateException("Already?TtlRunnable!"); 47????????} 48????????return?new?TtlRunnable(runnable,?releaseTtlValueReferenceAfterRun); 49????} 50 51????//?省略其他不太重要的方法 52}

其實關注點只需要放在構造函數、run()方法,其他都是基于此做修飾或者擴展。構造函數的源碼說明,capture()在TtlRunnable實例化的時候已經被調用,實例化它的一般就是父線程,所以整體的執行流程如下:

Agent模塊

啟用Agent功能,需要在Java的啟動參數添加:-javaagent:path/to/transmittable-thread-local-x.yzx.jar。原理是通過Instrumentation回調激發ClassFileTransformer實現目標類的字節碼增強,使用到javassist,被增強的類主要是泛線程池的類:

  • Executor體系:主要包括ThreadPoolExecutor和ScheduledThreadPoolExecutor,對應的字節碼增強類實現是TtlExecutorTransformlet。

  • ForkJoinPool:對應的字節碼增強類實現是TtlForkJoinTransformlet。

  • TimerTask:對應的字節碼增強類實現是TtlTimerTaskTransformlet。

Agent的入口類是TtlAgent,這里查看對應的源碼:

1public?final?class?TtlAgent?{23????public?static?void?premain(String?agentArgs,?@NonNull?Instrumentation?inst)?{4????????kvs?=?splitCommaColonStringToKV(agentArgs);56????????Logger.setLoggerImplType(getLogImplTypeFromAgentArgs(kvs));7????????final?Logger?logger?=?Logger.getLogger(TtlAgent.class);89????????try?{ 10????????????logger.info("[TtlAgent.premain]?begin,?agentArgs:?"?+?agentArgs?+?",?Instrumentation:?"?+?inst); 11????????????final?boolean?disableInheritableForThreadPool?=?isDisableInheritableForThreadPool(); 12????????????//?裝載所有的JavassistTransformlet 13????????????final?List<JavassistTransformlet>?transformletList?=?new?ArrayList<JavassistTransformlet>(); 14????????????transformletList.add(new?TtlExecutorTransformlet(disableInheritableForThreadPool)); 15????????????transformletList.add(new?TtlForkJoinTransformlet(disableInheritableForThreadPool)); 16????????????if?(isEnableTimerTask())?transformletList.add(new?TtlTimerTaskTransformlet()); 17????????????final?ClassFileTransformer?transformer?=?new?TtlTransformer(transformletList); 18????????????inst.addTransformer(transformer,?true); 19????????????logger.info("[TtlAgent.premain]?addTransformer?"?+?transformer.getClass()?+?"?success"); 20????????????logger.info("[TtlAgent.premain]?end"); 21????????????ttlAgentLoaded?=?true; 22????????}?catch?(Exception?e)?{ 23????????????String?msg?=?"Fail?to?load?TtlAgent?,?cause:?"?+?e.toString(); 24????????????logger.log(Level.SEVERE,?msg,?e); 25????????????throw?new?IllegalStateException(msg,?e); 26????????} 27????} 28}

List<JavassistTransformlet>作為參數傳入ClassFileTransformer的實現類TtlTransformer中,其中的轉換方法為:

1public?class?TtlTransformer?implements?ClassFileTransformer?{23????private?final?List<JavassistTransformlet>?transformletList?=?new?ArrayList<JavassistTransformlet>();45????TtlTransformer(List<??extends?JavassistTransformlet>?transformletList)?{6????????for?(JavassistTransformlet?transformlet?:?transformletList)?{7????????????this.transformletList.add(transformlet);8????????????logger.info("[TtlTransformer]?add?Transformlet?"?+?transformlet.getClass()?+?"?success");9????????} 10????} 11 12????@Override 13????public?final?byte[]?transform(@Nullable?final?ClassLoader?loader,?@Nullable?final?String?classFile,?final?Class<?>?classBeingRedefined, 14??????????????????????????????????final?ProtectionDomain?protectionDomain,?@NonNull?final?byte[]?classFileBuffer)?{ 15????????try?{ 16????????????//?Lambda?has?no?class?file,?no?need?to?transform,?just?return. 17????????????if?(classFile?==?null)?return?NO_TRANSFORM; 18????????????final?String?className?=?toClassName(classFile); 19????????????ClassInfo?classInfo?=?new?ClassInfo(className,?classFileBuffer,?loader); 20????????????//?這里做變量,如果字節碼被修改,則跳出循環返回 21????????????for?(JavassistTransformlet?transformlet?:?transformletList)?{ 22????????????????transformlet.doTransform(classInfo); 23????????????????if?(classInfo.isModified())?return?classInfo.getCtClass().toBytecode(); 24????????????} 25????????}?catch?(Throwable?t)?{ 26????????????String?msg?=?"Fail?to?transform?class?"?+?classFile?+?",?cause:?"?+?t.toString(); 27????????????logger.log(Level.SEVERE,?msg,?t); 28????????????throw?new?IllegalStateException(msg,?t); 29????????} 30????????return?NO_TRANSFORM; 31????} 32}

這里挑選TtlExecutorTransformlet的部分方法來看:

1????@Override2????public?void?doTransform(@NonNull?final?ClassInfo?classInfo)?throws?IOException,?NotFoundException,?CannotCompileException?{3????????//?如果當前加載的類包含java.util.concurrent.ThreadPoolExecutor或者java.util.concurrent.ScheduledThreadPoolExecutor4????????if?(EXECUTOR_CLASS_NAMES.contains(classInfo.getClassName()))?{5????????????final?CtClass?clazz?=?classInfo.getCtClass();6????????????//?遍歷所有的方法進行增強7????????????for?(CtMethod?method?:?clazz.getDeclaredMethods())?{8????????????????updateSubmitMethodsOfExecutorClass_decorateToTtlWrapperAndSetAutoWrapperAttachment(method);9????????????} 10????????????//?省略其他代碼 11????????}? 12????????//?省略其他代碼 13????} 14 15????private?void?updateSubmitMethodsOfExecutorClass_decorateToTtlWrapperAndSetAutoWrapperAttachment(@NonNull?final?CtMethod?method)?throws?NotFoundException,?CannotCompileException?{ 16????????final?int?modifiers?=?method.getModifiers(); 17????????if?(!Modifier.isPublic(modifiers)?||?Modifier.isStatic(modifiers))?return; 18????????//?這里主要在java.lang.Runnable構造時候調用com.alibaba.ttl.TtlRunnable#get()包裝為com.alibaba.ttl.TtlRunnable 19????????//?在java.util.concurrent.Callable構造時候調用com.alibaba.ttl.TtlCallable#get()包裝為com.alibaba.ttl.TtlCallable 20????????//?并且設置附件K-V為ttl.is.auto.wrapper=true 21????????CtClass[]?parameterTypes?=?method.getParameterTypes(); 22????????StringBuilder?insertCode?=?new?StringBuilder(); 23????????for?(int?i?=?0;?i?<?parameterTypes.length;?i++)?{ 24????????????final?String?paramTypeName?=?parameterTypes[i].getName(); 25????????????if?(PARAM_TYPE_NAME_TO_DECORATE_METHOD_CLASS.containsKey(paramTypeName))?{ 26????????????????String?code?=?String.format( 27????????????????????????//?decorate?to?TTL?wrapper, 28????????????????????????//?and?then?set?AutoWrapper?attachment/Tag 29????????????????????????"$%d?=?%s.get($%d,?false,?true);" 30????????????????????????????????+?"\ncom.alibaba.ttl.threadpool.agent.internal.transformlet.impl.Utils.setAutoWrapperAttachment($%<d);", 31????????????????????????i?+?1,?PARAM_TYPE_NAME_TO_DECORATE_METHOD_CLASS.get(paramTypeName),?i?+?1); 32????????????????logger.info("insert?code?before?method?"?+?signatureOfMethod(method)?+?"?of?class?"?+?method.getDeclaringClass().getName()?+?":?"?+?code); 33????????????????insertCode.append(code); 34????????????} 35????????} 36????????if?(insertCode.length()?>?0)?method.insertBefore(insertCode.toString()); 37????}

上面分析的方法的功能,就是讓java.util.concurrent.ThreadPoolExecutor和java.util.concurrent.ScheduledThreadPoolExecutor的字節碼被增強,提交的java.lang.Runnable類型的任務會被包裝為TtlRunnable,提交的java.util.concurrent.Callable類型的任務會被包裝為TtlCallable,實現了無入侵無感知地嵌入TTL的功能。

小結

TTL在使用線程池等會池化復用線程的執行組件情況下,提供ThreadLocal值的傳遞功能,解決異步執行時上下文傳遞的問題。它是一個Java標準庫,為框架/中間件設施開發提供的標配能力,項目代碼精悍,只依賴了javassist做字節碼增強,實現Agent模式下的近乎無入侵提供TTL功能的特性。

TTL能在業務代碼中實現透明/自動完成所有異步執行上下文的可定制、規范化的捕捉/傳遞,如果恰好碰到異步執行時上下文傳遞的問題,建議可以嘗試此庫。

參考資料:

  • JDK11相關源碼

  • TTL源碼

有道無術,術可成;有術無道,止于術

歡迎大家關注Java之道公眾號

好文章,我在看??

總結

以上是生活随笔為你收集整理的通过transmittable-thread-local源码理解线程池线程本地变量传递的原理的全部內容,希望文章能夠幫你解決所遇到的問題。

如果覺得生活随笔網站內容還不錯,歡迎將生活随笔推薦給好友。

久久精品女人的天堂av | 日本一卡2卡3卡4卡无卡免费网站 国产一区二区三区影院 | 亚洲欧洲日本综合aⅴ在线 | 青春草在线视频免费观看 | 丰满少妇高潮惨叫视频 | 国产麻豆精品精东影业av网站 | 中国大陆精品视频xxxx | 国精产品一品二品国精品69xx | 成人试看120秒体验区 | 理论片87福利理论电影 | 国产超级va在线观看视频 | 成人精品视频一区二区 | 一本加勒比波多野结衣 | 爆乳一区二区三区无码 | 欧美一区二区三区视频在线观看 | 国产一区二区三区影院 | 老熟妇仑乱视频一区二区 | 久久99国产综合精品 | 国产成人一区二区三区在线观看 | 中国女人内谢69xxxxxa片 | 国产精品久免费的黄网站 | 亚洲国产av精品一区二区蜜芽 | 国产熟妇高潮叫床视频播放 | 国产特级毛片aaaaaaa高清 | 亚洲色成人中文字幕网站 | 男人扒开女人内裤强吻桶进去 | 欧美日韩人成综合在线播放 | 亚洲国产av美女网站 | 国产三级精品三级男人的天堂 | 国产av一区二区三区最新精品 | 亚洲 激情 小说 另类 欧美 | 亚洲男人av天堂午夜在 | 亚洲高清偷拍一区二区三区 | 精品国产乱码久久久久乱码 | 成在人线av无码免观看麻豆 | 爆乳一区二区三区无码 | 久久国语露脸国产精品电影 | 日本熟妇浓毛 | 国产精品理论片在线观看 | 日日橹狠狠爱欧美视频 | 国精产品一区二区三区 | 亚洲国产精品久久人人爱 | 亚洲色在线无码国产精品不卡 | 免费人成在线观看网站 | 欧洲熟妇色 欧美 | 中文字幕乱妇无码av在线 | 国产亚洲精品久久久ai换 | 999久久久国产精品消防器材 | 中文字幕精品av一区二区五区 | 国产精品igao视频网 | 麻豆精品国产精华精华液好用吗 | 日日麻批免费40分钟无码 | 国产精品第一国产精品 | 精品厕所偷拍各类美女tp嘘嘘 | 国产精品国产自线拍免费软件 | 欧美日韩人成综合在线播放 | 红桃av一区二区三区在线无码av | 狂野欧美激情性xxxx | 国精产品一品二品国精品69xx | 成人无码视频在线观看网站 | 成在人线av无码免费 | 久久人人97超碰a片精品 | 国产香蕉尹人综合在线观看 | 欧美黑人乱大交 | 国产办公室秘书无码精品99 | 一本久久a久久精品亚洲 | 国产口爆吞精在线视频 | 少妇激情av一区二区 | 人人澡人摸人人添 | 天天综合网天天综合色 | 国产9 9在线 | 中文 | 久久亚洲国产成人精品性色 | 女人高潮内射99精品 | 亚洲 欧美 激情 小说 另类 | 国产激情无码一区二区 | 日日摸夜夜摸狠狠摸婷婷 | 在教室伦流澡到高潮hnp视频 | 波多野结衣aⅴ在线 | 久久久久99精品国产片 | 精品国产国产综合精品 | 狠狠色丁香久久婷婷综合五月 | 无码国内精品人妻少妇 | 婷婷丁香五月天综合东京热 | 内射后入在线观看一区 | 国产精品欧美成人 | 中文字幕无码免费久久9一区9 | 一本色道久久综合亚洲精品不卡 | 久久国内精品自在自线 | 国产sm调教视频在线观看 | 性做久久久久久久免费看 | 久久精品一区二区三区四区 | 亚洲 另类 在线 欧美 制服 | 精品一区二区三区波多野结衣 | 国产情侣作爱视频免费观看 | 美女毛片一区二区三区四区 | 日本饥渴人妻欲求不满 | 国产偷国产偷精品高清尤物 | 国产亚洲精品久久久久久久久动漫 | 国产午夜亚洲精品不卡下载 | 亚洲精品国产精品乱码不卡 | а√天堂www在线天堂小说 | 最近的中文字幕在线看视频 | 亚洲国产av精品一区二区蜜芽 | 一本大道伊人av久久综合 | 高清国产亚洲精品自在久久 | 国产一区二区三区日韩精品 | 丝袜美腿亚洲一区二区 | 伊人久久大香线蕉av一区二区 | 亚洲中文字幕久久无码 | 久久综合久久自在自线精品自 | 国产莉萝无码av在线播放 | 日日摸日日碰夜夜爽av | 久久99精品久久久久久动态图 | 国产精品永久免费视频 | 美女极度色诱视频国产 | 无码人妻精品一区二区三区下载 | 亚洲欧美国产精品专区久久 | 九九久久精品国产免费看小说 | 欧美精品国产综合久久 | 思思久久99热只有频精品66 | 免费中文字幕日韩欧美 | 色欲人妻aaaaaaa无码 | 精品少妇爆乳无码av无码专区 | 国产在线无码精品电影网 | 2020久久香蕉国产线看观看 | 内射老妇bbwx0c0ck | 在线 国产 欧美 亚洲 天堂 | 国产精品对白交换视频 | 亚洲成av人片在线观看无码不卡 | 久久天天躁夜夜躁狠狠 | 国产香蕉尹人综合在线观看 | 婷婷丁香五月天综合东京热 | 牲欲强的熟妇农村老妇女 | 日日摸天天摸爽爽狠狠97 | 爱做久久久久久 | 日日橹狠狠爱欧美视频 | 久久99久久99精品中文字幕 | 久久久久亚洲精品男人的天堂 | 日本一卡二卡不卡视频查询 | 无码av中文字幕免费放 | 国产农村妇女aaaaa视频 撕开奶罩揉吮奶头视频 | 精品国产福利一区二区 | 97久久国产亚洲精品超碰热 | 曰韩少妇内射免费播放 | 乌克兰少妇xxxx做受 | 免费看少妇作爱视频 | 国内精品一区二区三区不卡 | 97久久国产亚洲精品超碰热 | 日本护士毛茸茸高潮 | 丰满护士巨好爽好大乳 | 在线欧美精品一区二区三区 | 欧美老妇与禽交 | 一个人免费观看的www视频 | 四虎国产精品免费久久 | 色综合久久中文娱乐网 | av无码久久久久不卡免费网站 | 性生交大片免费看女人按摩摩 | 人人妻人人藻人人爽欧美一区 | 国产 浪潮av性色四虎 | 国产人妻久久精品二区三区老狼 | 国产亚洲人成a在线v网站 | 国产精品美女久久久 | 午夜成人1000部免费视频 | 无码人妻精品一区二区三区不卡 | 蜜臀av无码人妻精品 | 欧美性猛交xxxx富婆 | 国产办公室秘书无码精品99 | 亚洲欧洲日本综合aⅴ在线 | 久久久久久av无码免费看大片 | 六十路熟妇乱子伦 | 国产激情综合五月久久 | 好屌草这里只有精品 | 免费看少妇作爱视频 | 亚洲国产精品毛片av不卡在线 | 成在人线av无码免观看麻豆 | 在线欧美精品一区二区三区 | 精品欧美一区二区三区久久久 | 国产成人综合色在线观看网站 | 中文亚洲成a人片在线观看 | 久久综合色之久久综合 | 精品国产一区二区三区四区 | 国产午夜亚洲精品不卡下载 | 无码吃奶揉捏奶头高潮视频 | 国产无遮挡又黄又爽免费视频 | 日日天日日夜日日摸 | 国产农村妇女高潮大叫 | 国产亲子乱弄免费视频 | 国产亚洲精品久久久久久国模美 | 久久久久免费看成人影片 | 18禁止看的免费污网站 | 久久无码专区国产精品s | 天堂а√在线地址中文在线 | 香港三级日本三级妇三级 | 日韩av无码一区二区三区不卡 | 无码av免费一区二区三区试看 | 国产精品鲁鲁鲁 | 图片区 小说区 区 亚洲五月 | 免费人成在线视频无码 | 国产性生大片免费观看性 | 动漫av一区二区在线观看 | 国产精品无码一区二区桃花视频 | 欧美成人高清在线播放 | 成年美女黄网站色大免费全看 | 国产精品人人妻人人爽 | 亚洲国产精品无码一区二区三区 | 中文精品久久久久人妻不卡 | 狠狠综合久久久久综合网 | 男人扒开女人内裤强吻桶进去 | 狠狠躁日日躁夜夜躁2020 | 蜜桃av抽搐高潮一区二区 | √天堂中文官网8在线 | 99国产欧美久久久精品 | 国产精品沙发午睡系列 | 99视频精品全部免费免费观看 | 夜精品a片一区二区三区无码白浆 | 亚洲欧美国产精品专区久久 | 亚洲欧美色中文字幕在线 | 好爽又高潮了毛片免费下载 | 欧美 日韩 人妻 高清 中文 | 伊在人天堂亚洲香蕉精品区 | 亚洲一区二区三区四区 | 国产人妻人伦精品1国产丝袜 | 亚洲男女内射在线播放 | 亚洲成a人片在线观看日本 | 四十如虎的丰满熟妇啪啪 | 亚洲精品一区二区三区四区五区 | www成人国产高清内射 | 欧美丰满熟妇xxxx性ppx人交 | 亚洲天堂2017无码 | 成人免费视频在线观看 | aⅴ在线视频男人的天堂 | 亚洲成熟女人毛毛耸耸多 | 激情人妻另类人妻伦 | 色窝窝无码一区二区三区色欲 | 丰满少妇熟乱xxxxx视频 | 国产特级毛片aaaaaa高潮流水 | 色综合久久中文娱乐网 | 久久综合久久自在自线精品自 | 成人片黄网站色大片免费观看 | 中文字幕人妻无码一区二区三区 | 成人无码视频免费播放 | 无码av最新清无码专区吞精 | 99国产欧美久久久精品 | 国产午夜亚洲精品不卡 | 麻豆果冻传媒2021精品传媒一区下载 | 乱人伦中文视频在线观看 | 在线天堂新版最新版在线8 | 精品国产麻豆免费人成网站 | 久久精品国产精品国产精品污 | 精品国产aⅴ无码一区二区 | 精品一区二区不卡无码av | 日韩人妻无码一区二区三区久久99 | 国产精品第一区揄拍无码 | 一本大道久久东京热无码av | 精品国产一区二区三区av 性色 | 中文字幕人妻无码一区二区三区 | 大屁股大乳丰满人妻 | aa片在线观看视频在线播放 | 久久精品一区二区三区四区 | 国产舌乚八伦偷品w中 | 欧美一区二区三区视频在线观看 | 精品一区二区三区无码免费视频 | 亚洲欧美精品aaaaaa片 | 国产精品久久久久久无码 | 国产情侣作爱视频免费观看 | 性色av无码免费一区二区三区 | 少妇厨房愉情理9仑片视频 | 欧美人与禽zoz0性伦交 | 国产深夜福利视频在线 | 亚洲欧洲日本无在线码 | 在线播放无码字幕亚洲 | 国产在线无码精品电影网 | 天天燥日日燥 | 波多野结衣av在线观看 | 强伦人妻一区二区三区视频18 | a片免费视频在线观看 | 欧美老人巨大xxxx做受 | 日本饥渴人妻欲求不满 | 又色又爽又黄的美女裸体网站 | 国产亚洲tv在线观看 | 亚洲欧美国产精品专区久久 | 亚洲欧美精品伊人久久 | 久久人人97超碰a片精品 | 国产亚洲精品久久久久久 | 中文字幕无线码免费人妻 | 成人亚洲精品久久久久软件 | 99久久精品无码一区二区毛片 | 久久精品国产一区二区三区 | 国产午夜福利亚洲第一 | 又黄又爽又色的视频 | 亚洲一区av无码专区在线观看 | 成熟人妻av无码专区 | 久久精品国产99久久6动漫 | 欧美精品免费观看二区 | 国内精品人妻无码久久久影院 | 亚洲熟女一区二区三区 | 亚洲第一无码av无码专区 | 性色av无码免费一区二区三区 | 国产精品久久久久久亚洲毛片 | 天天拍夜夜添久久精品 | а√资源新版在线天堂 | 日本一区二区更新不卡 | 丝袜人妻一区二区三区 | 免费看男女做好爽好硬视频 | 99视频精品全部免费免费观看 | 亚洲一区二区三区含羞草 | 久久国内精品自在自线 | 奇米影视7777久久精品人人爽 | 无码国产乱人伦偷精品视频 | www国产亚洲精品久久网站 | 国产午夜福利亚洲第一 | 暴力强奷在线播放无码 | 激情内射日本一区二区三区 | 免费无码一区二区三区蜜桃大 | 久久婷婷五月综合色国产香蕉 | 国产在热线精品视频 | 色诱久久久久综合网ywww | 午夜性刺激在线视频免费 | 国产激情无码一区二区 | 亚洲日韩av片在线观看 | 久久久久久九九精品久 | 熟妇人妻中文av无码 | 永久免费观看美女裸体的网站 | 久久久久免费看成人影片 | 捆绑白丝粉色jk震动捧喷白浆 | 国产成人综合在线女婷五月99播放 | 天天躁夜夜躁狠狠是什么心态 | 九九久久精品国产免费看小说 | 亚洲精品一区二区三区大桥未久 | 国产精品人人爽人人做我的可爱 | 久久国产精品偷任你爽任你 | 色综合久久88色综合天天 | 成 人影片 免费观看 | 婷婷色婷婷开心五月四房播播 | 亚洲中文字幕久久无码 | www国产精品内射老师 | 精品人妻中文字幕有码在线 | 亚洲日韩中文字幕在线播放 | 无码人中文字幕 | 99精品视频在线观看免费 | 老司机亚洲精品影院 | 99久久婷婷国产综合精品青草免费 | 国产福利视频一区二区 | 1000部啪啪未满十八勿入下载 | 精品欧美一区二区三区久久久 | 国产亚洲精品久久久闺蜜 | 极品嫩模高潮叫床 | аⅴ资源天堂资源库在线 | 亚洲欧美综合区丁香五月小说 | 午夜精品久久久内射近拍高清 | 亚洲欧美日韩成人高清在线一区 | 国产性生交xxxxx无码 | 国产特级毛片aaaaaaa高清 | 99久久久国产精品无码免费 | 欧美老妇与禽交 | 18禁止看的免费污网站 | 亚洲小说图区综合在线 | 性欧美大战久久久久久久 | 亚洲日本va中文字幕 | 久久精品国产99精品亚洲 | 日本一卡二卡不卡视频查询 | 欧美日韩视频无码一区二区三 | 精品国产麻豆免费人成网站 | 国产高清av在线播放 | 国产精品亚洲lv粉色 | 国产国语老龄妇女a片 | 又湿又紧又大又爽a视频国产 | 无码一区二区三区在线观看 | 久久精品人人做人人综合 | 久久久久久久女国产乱让韩 | 中文字幕 人妻熟女 | 国产亚洲精品久久久久久久久动漫 | 装睡被陌生人摸出水好爽 | 97夜夜澡人人爽人人喊中国片 | 国产农村乱对白刺激视频 | 性欧美熟妇videofreesex | 3d动漫精品啪啪一区二区中 | 婷婷五月综合激情中文字幕 | 鲁鲁鲁爽爽爽在线视频观看 | 青草青草久热国产精品 | 丰满人妻被黑人猛烈进入 | 国产亚洲精品精品国产亚洲综合 | 少妇被黑人到高潮喷出白浆 | 国语自产偷拍精品视频偷 | 成人免费视频视频在线观看 免费 | 一本久久a久久精品亚洲 | 国产69精品久久久久app下载 | 国产乱人伦av在线无码 | 国内精品九九久久久精品 | 亚洲午夜无码久久 | 亚洲一区二区三区四区 | 亚洲中文字幕在线观看 | 国产成人av免费观看 | 日韩欧美中文字幕在线三区 | 亚洲日本一区二区三区在线 | 精品一区二区不卡无码av | 久久99精品国产麻豆蜜芽 | 亚洲精品一区二区三区在线 | 欧美精品在线观看 | 在线天堂新版最新版在线8 | 大乳丰满人妻中文字幕日本 | 国产av人人夜夜澡人人爽麻豆 | 噜噜噜亚洲色成人网站 | 日韩欧美群交p片內射中文 | 99久久婷婷国产综合精品青草免费 | 国产偷国产偷精品高清尤物 | 成人三级无码视频在线观看 | 精品国产乱码久久久久乱码 | 国产69精品久久久久app下载 | 成年女人永久免费看片 | 99精品国产综合久久久久五月天 | 日韩精品a片一区二区三区妖精 | 少妇人妻av毛片在线看 | 精品乱子伦一区二区三区 | 国产精品高潮呻吟av久久4虎 | 国产精品二区一区二区aⅴ污介绍 | 国产又爽又猛又粗的视频a片 | 激情综合激情五月俺也去 | 无码中文字幕色专区 | 国产亚洲精品久久久久久久 | 精品乱子伦一区二区三区 | 色婷婷香蕉在线一区二区 | 精品国产成人一区二区三区 | 国内少妇偷人精品视频免费 | 人妻少妇精品无码专区二区 | 午夜精品久久久内射近拍高清 | 国产美女精品一区二区三区 | 性做久久久久久久免费看 | 久久久久人妻一区精品色欧美 | 成人亚洲精品久久久久 | 人妻插b视频一区二区三区 | 亚洲欧美国产精品久久 | 午夜精品久久久内射近拍高清 | 野外少妇愉情中文字幕 | 国产精品无码一区二区桃花视频 | 中文字幕无码免费久久99 | 97久久国产亚洲精品超碰热 | 欧美阿v高清资源不卡在线播放 | 色偷偷人人澡人人爽人人模 | 欧美自拍另类欧美综合图片区 | 日日碰狠狠丁香久燥 | 久久久久国色av免费观看性色 | a在线亚洲男人的天堂 | 国内丰满熟女出轨videos | 日本饥渴人妻欲求不满 | 中文字幕无码视频专区 | 真人与拘做受免费视频一 | 玩弄人妻少妇500系列视频 | 狠狠色欧美亚洲狠狠色www | 波多野结衣av一区二区全免费观看 | 全球成人中文在线 | 亚洲中文字幕成人无码 | 久久久久久a亚洲欧洲av冫 | 少妇性l交大片欧洲热妇乱xxx | 国产成人综合色在线观看网站 | 国产亚洲精品久久久久久国模美 | 精品久久综合1区2区3区激情 | 人妻人人添人妻人人爱 | 国产97人人超碰caoprom | 麻豆国产人妻欲求不满谁演的 | 国产在线一区二区三区四区五区 | 日韩欧美中文字幕在线三区 | 欧美自拍另类欧美综合图片区 | 性欧美熟妇videofreesex | 国产午夜亚洲精品不卡下载 | 大胆欧美熟妇xx | 18精品久久久无码午夜福利 | 人妻少妇精品久久 | 欧美xxxx黑人又粗又长 | 在线精品亚洲一区二区 | 亚洲成a人片在线观看无码3d | 亚洲综合伊人久久大杳蕉 | 精品成人av一区二区三区 | 天堂а√在线中文在线 | 久久亚洲精品中文字幕无男同 | 激情亚洲一区国产精品 | 国产9 9在线 | 中文 | 亚洲国产精品无码久久久久高潮 | 国产精品资源一区二区 | 亚洲色大成网站www | 又黄又爽又色的视频 | 国产人成高清在线视频99最全资源 | 丰满人妻翻云覆雨呻吟视频 | 国产成人一区二区三区别 | 亚洲人亚洲人成电影网站色 | 日本又色又爽又黄的a片18禁 | 97se亚洲精品一区 | 人妻人人添人妻人人爱 | 色综合视频一区二区三区 | 日韩亚洲欧美精品综合 | 妺妺窝人体色www在线小说 | 3d动漫精品啪啪一区二区中 | 久久综合久久自在自线精品自 | 一本大道伊人av久久综合 | 亚洲娇小与黑人巨大交 | 在线精品国产一区二区三区 | 毛片内射-百度 | 人妻无码αv中文字幕久久琪琪布 | 欧美丰满熟妇xxxx | 久久久婷婷五月亚洲97号色 | 无码人妻丰满熟妇区五十路百度 | 2020最新国产自产精品 | 国产国产精品人在线视 | 精品人人妻人人澡人人爽人人 | 精品成人av一区二区三区 | 中文字幕无码免费久久9一区9 | 全黄性性激高免费视频 | 乱人伦中文视频在线观看 | 波多野结衣av一区二区全免费观看 | 欧洲美熟女乱又伦 | 三上悠亚人妻中文字幕在线 | 男人和女人高潮免费网站 | 成人片黄网站色大片免费观看 | 色一情一乱一伦一区二区三欧美 | 国产日产欧产精品精品app | 少妇太爽了在线观看 | 乌克兰少妇性做爰 | 2020久久香蕉国产线看观看 | 久久久久免费精品国产 | 成熟人妻av无码专区 | 男人的天堂2018无码 | 无码一区二区三区在线观看 | 国产精品人人爽人人做我的可爱 | 国产精品久久久一区二区三区 | 少妇太爽了在线观看 | 乌克兰少妇性做爰 | 麻豆国产人妻欲求不满 | 国产一区二区三区影院 | 中文字幕 人妻熟女 | 国产性猛交╳xxx乱大交 国产精品久久久久久无码 欧洲欧美人成视频在线 | 午夜精品一区二区三区在线观看 | 成人免费无码大片a毛片 | 无套内射视频囯产 | 国产黄在线观看免费观看不卡 | 人人妻人人澡人人爽人人精品 | 国产成人综合美国十次 | 国产精品18久久久久久麻辣 | 色欲综合久久中文字幕网 | 久久综合九色综合欧美狠狠 | 熟妇女人妻丰满少妇中文字幕 | 婷婷五月综合激情中文字幕 | 国内综合精品午夜久久资源 | 亚洲一区二区观看播放 | 夜夜夜高潮夜夜爽夜夜爰爰 | 亚洲爆乳大丰满无码专区 | 1000部啪啪未满十八勿入下载 | 无码国产乱人伦偷精品视频 | 少妇人妻偷人精品无码视频 | a国产一区二区免费入口 | 色一情一乱一伦 | 欧美自拍另类欧美综合图片区 | 少妇高潮喷潮久久久影院 | 成人无码视频在线观看网站 | 亚洲理论电影在线观看 | 日日碰狠狠躁久久躁蜜桃 | 国产性猛交╳xxx乱大交 国产精品久久久久久无码 欧洲欧美人成视频在线 | 亚洲成在人网站无码天堂 | 国产乱人偷精品人妻a片 | 国产亚洲欧美日韩亚洲中文色 | 中文字幕无码乱人伦 | 国产亚洲精品久久久久久国模美 | 999久久久国产精品消防器材 | 精品国产一区二区三区四区 | 亚洲男人av香蕉爽爽爽爽 | 亚洲日韩av一区二区三区四区 | 国精品人妻无码一区二区三区蜜柚 | 欧美肥老太牲交大战 | 亚洲精品成a人在线观看 | 国产亚洲美女精品久久久2020 | 国产日产欧产精品精品app | 极品尤物被啪到呻吟喷水 | 日韩精品一区二区av在线 | 国产人妻大战黑人第1集 | 97夜夜澡人人爽人人喊中国片 | 99麻豆久久久国产精品免费 | 国产人妻大战黑人第1集 | 国产精品久久国产精品99 | 一个人免费观看的www视频 | 国产精品对白交换视频 | 色婷婷综合中文久久一本 | 精品欧美一区二区三区久久久 | 强伦人妻一区二区三区视频18 | 精品国精品国产自在久国产87 | 中文字幕精品av一区二区五区 | 中文字幕无线码 | 欧美亚洲日韩国产人成在线播放 | 午夜福利试看120秒体验区 | 亚洲精品鲁一鲁一区二区三区 | 帮老师解开蕾丝奶罩吸乳网站 | 少妇被粗大的猛进出69影院 | 国产精品无套呻吟在线 | 精品久久综合1区2区3区激情 | 国产激情无码一区二区app | 亚洲欧美综合区丁香五月小说 | 国产凸凹视频一区二区 | 秋霞特色aa大片 | 人妻插b视频一区二区三区 | 人妻少妇精品无码专区动漫 | 天海翼激烈高潮到腰振不止 | 无人区乱码一区二区三区 | 精品国产乱码久久久久乱码 | 大肉大捧一进一出好爽视频 | 国产亚洲精品久久久久久 | 国产极品视觉盛宴 | 色欲人妻aaaaaaa无码 | 久久视频在线观看精品 | 人妻天天爽夜夜爽一区二区 | 亚洲色在线无码国产精品不卡 | 巨爆乳无码视频在线观看 | 永久免费观看美女裸体的网站 | 内射巨臀欧美在线视频 | 亚洲人成网站在线播放942 | 精品无码国产一区二区三区av | 好男人社区资源 | 精品久久久久久人妻无码中文字幕 | 小鲜肉自慰网站xnxx | 国产精品99久久精品爆乳 | 亚洲高清偷拍一区二区三区 | 一区二区传媒有限公司 | 丁香啪啪综合成人亚洲 | 成人av无码一区二区三区 | 日本精品高清一区二区 | 国产精品国产自线拍免费软件 | 狠狠色噜噜狠狠狠7777奇米 | 国产精品igao视频网 | 午夜理论片yy44880影院 | 久久久精品456亚洲影院 | 学生妹亚洲一区二区 | 丰满少妇弄高潮了www | 蜜桃臀无码内射一区二区三区 | 在线看片无码永久免费视频 | 亚洲一区二区观看播放 | 久久无码中文字幕免费影院蜜桃 | 麻豆人妻少妇精品无码专区 | 国产香蕉97碰碰久久人人 | 久久人人爽人人爽人人片av高清 | 夜精品a片一区二区三区无码白浆 | 久久久国产精品无码免费专区 | 天天摸天天透天天添 | 欧美兽交xxxx×视频 | 亚洲综合色区中文字幕 | 任你躁国产自任一区二区三区 | 亚洲国产精华液网站w | 一区二区三区高清视频一 | 亚洲中文无码av永久不收费 | 国产成人无码a区在线观看视频app | 国产手机在线αⅴ片无码观看 | 国精品人妻无码一区二区三区蜜柚 | 性色欲情网站iwww九文堂 | 国产人成高清在线视频99最全资源 | 中文字幕无码人妻少妇免费 | 成在人线av无码免观看麻豆 | 99re在线播放 | 久久97精品久久久久久久不卡 | 漂亮人妻洗澡被公强 日日躁 | 中文字幕av伊人av无码av | 奇米影视888欧美在线观看 | 国产乱人伦app精品久久 国产在线无码精品电影网 国产国产精品人在线视 | 精品人妻人人做人人爽 | 中文字幕无码日韩欧毛 | 在线观看欧美一区二区三区 | 欧美大屁股xxxxhd黑色 | 精品国产成人一区二区三区 | 波多野结衣av一区二区全免费观看 | 成人av无码一区二区三区 | 激情五月综合色婷婷一区二区 | 熟女少妇在线视频播放 | 国产av无码专区亚洲a∨毛片 | 国产精品久久久午夜夜伦鲁鲁 | 日本爽爽爽爽爽爽在线观看免 | 小sao货水好多真紧h无码视频 | 亚洲人成影院在线无码按摩店 | 377p欧洲日本亚洲大胆 | 色婷婷av一区二区三区之红樱桃 | 无套内射视频囯产 | 2020久久超碰国产精品最新 | 性史性农村dvd毛片 | 国产又粗又硬又大爽黄老大爷视 | 高清不卡一区二区三区 | 久久久久久九九精品久 | 久久精品无码一区二区三区 | 国産精品久久久久久久 | 丁香花在线影院观看在线播放 | 国产免费无码一区二区视频 | 色五月五月丁香亚洲综合网 | 日本乱人伦片中文三区 | 国产电影无码午夜在线播放 | 在线欧美精品一区二区三区 | 1000部夫妻午夜免费 | 久久精品国产一区二区三区肥胖 | 日本爽爽爽爽爽爽在线观看免 | 国产97人人超碰caoprom | 色一情一乱一伦一视频免费看 | 九九综合va免费看 | 久久精品国产99精品亚洲 | 国产乱人偷精品人妻a片 | 青草青草久热国产精品 | 亚洲 高清 成人 动漫 | 97精品国产97久久久久久免费 | 亚洲欧美日韩国产精品一区二区 | 国产日产欧产精品精品app | 欧美黑人巨大xxxxx | 风流少妇按摩来高潮 | 国产精品无码永久免费888 | 欧美人与禽猛交狂配 | 天天躁日日躁狠狠躁免费麻豆 | 97无码免费人妻超级碰碰夜夜 | 欧洲精品码一区二区三区免费看 | 成人影院yy111111在线观看 | 精品aⅴ一区二区三区 | 蜜臀av无码人妻精品 | 中文亚洲成a人片在线观看 | www成人国产高清内射 | 疯狂三人交性欧美 | 亚洲精品久久久久avwww潮水 | 成人无码视频免费播放 | 久久午夜夜伦鲁鲁片无码免费 | 风流少妇按摩来高潮 | 国产午夜亚洲精品不卡 | 亚洲区小说区激情区图片区 | 国产麻豆精品一区二区三区v视界 | 亚洲日韩av片在线观看 | 亚洲熟悉妇女xxx妇女av | 中文无码伦av中文字幕 | 无码吃奶揉捏奶头高潮视频 | 国产日产欧产精品精品app | 少妇性l交大片欧洲热妇乱xxx | 漂亮人妻洗澡被公强 日日躁 | 国产精品对白交换视频 | 日本熟妇浓毛 | 久久人人爽人人爽人人片ⅴ | 久久久久国色av免费观看性色 | 色综合视频一区二区三区 | 狠狠噜狠狠狠狠丁香五月 | 免费人成在线视频无码 | 国产激情无码一区二区app | 人妻少妇精品视频专区 | 日韩精品成人一区二区三区 | 国产成人无码av片在线观看不卡 | 在线播放亚洲第一字幕 | 99久久无码一区人妻 | 小鲜肉自慰网站xnxx | 久久成人a毛片免费观看网站 | 亚洲熟妇色xxxxx亚洲 | 亚洲熟悉妇女xxx妇女av | 亚洲一区二区三区偷拍女厕 | 性生交大片免费看女人按摩摩 | 久久久久av无码免费网 | 欧美熟妇另类久久久久久多毛 | 亚洲一区二区三区香蕉 | 国产成人精品久久亚洲高清不卡 | 精品国产一区av天美传媒 | 亚洲中文字幕无码中字 | 欧美zoozzooz性欧美 | 亚洲精品国产品国语在线观看 | 成人av无码一区二区三区 | www一区二区www免费 | 午夜成人1000部免费视频 | 亚洲中文字幕无码一久久区 | 久久亚洲中文字幕无码 | 精品国产一区二区三区av 性色 | 国内精品久久久久久中文字幕 | 思思久久99热只有频精品66 | 女人被男人爽到呻吟的视频 | 国产成人无码a区在线观看视频app | 久久无码中文字幕免费影院蜜桃 | 国产内射老熟女aaaa | 国产深夜福利视频在线 | 国产亚洲美女精品久久久2020 | 国产精品亚洲综合色区韩国 | 图片区 小说区 区 亚洲五月 | 国产亚洲精品久久久闺蜜 | 久久久成人毛片无码 | 中文字幕av无码一区二区三区电影 | 国产午夜视频在线观看 | 精品午夜福利在线观看 | 久久精品成人欧美大片 | 色综合久久中文娱乐网 | 男人扒开女人内裤强吻桶进去 | 国产成人亚洲综合无码 | 东京一本一道一二三区 | 少妇被黑人到高潮喷出白浆 | 67194成是人免费无码 | 麻豆国产丝袜白领秘书在线观看 | 日韩人妻无码一区二区三区久久99 | 亚洲欧美国产精品久久 | 精品一区二区三区波多野结衣 | 亚洲一区二区三区在线观看网站 | 亚洲色无码一区二区三区 | 国产精品毛片一区二区 | 国产成人无码午夜视频在线观看 | 国产一区二区三区日韩精品 | 亚洲の无码国产の无码影院 | 粗大的内捧猛烈进出视频 | 美女扒开屁股让男人桶 | 午夜男女很黄的视频 | 亚洲欧美日韩国产精品一区二区 | 国产午夜无码精品免费看 | 大地资源网第二页免费观看 | 18禁止看的免费污网站 | 亚洲精品中文字幕乱码 | 国产口爆吞精在线视频 | 国产亚洲欧美日韩亚洲中文色 | 亚洲狠狠色丁香婷婷综合 | 国产手机在线αⅴ片无码观看 | 99久久精品无码一区二区毛片 | 欧美性生交xxxxx久久久 | 欧洲精品码一区二区三区免费看 | av无码不卡在线观看免费 | 无码午夜成人1000部免费视频 | 99久久久无码国产aaa精品 | 在线精品国产一区二区三区 | 色综合久久久无码中文字幕 | 精品水蜜桃久久久久久久 | 成人无码精品1区2区3区免费看 | 亚洲欧美中文字幕5发布 | 俺去俺来也www色官网 | 中文字幕亚洲情99在线 | 少妇被粗大的猛进出69影院 | 欧美xxxxx精品 | 久久亚洲中文字幕精品一区 | 国产成人人人97超碰超爽8 | 日韩精品一区二区av在线 | 欧美性生交活xxxxxdddd | a在线观看免费网站大全 | 欧美老熟妇乱xxxxx | 亚洲中文字幕va福利 | 无码精品国产va在线观看dvd | 亚洲天堂2017无码 | 国产又爽又猛又粗的视频a片 | 丰满少妇弄高潮了www | 99er热精品视频 | 狠狠亚洲超碰狼人久久 | 欧美老妇交乱视频在线观看 | 国产综合色产在线精品 | 亚洲欧洲无卡二区视頻 | 欧洲熟妇精品视频 | 亚洲成在人网站无码天堂 | 夜夜影院未满十八勿进 | 亚洲自偷自偷在线制服 | 欧美熟妇另类久久久久久不卡 | 亚洲色无码一区二区三区 | 国产无遮挡又黄又爽又色 | 少妇激情av一区二区 | 国产熟妇高潮叫床视频播放 | 天天摸天天透天天添 | 国内精品久久久久久中文字幕 | 亚洲日本一区二区三区在线 | 国产性生交xxxxx无码 | 亚洲男人av香蕉爽爽爽爽 | 免费网站看v片在线18禁无码 | 又粗又大又硬毛片免费看 | 波多野结衣高清一区二区三区 | 久精品国产欧美亚洲色aⅴ大片 | 国内精品九九久久久精品 | 无码午夜成人1000部免费视频 | 377p欧洲日本亚洲大胆 | 国产又爽又黄又刺激的视频 | 麻花豆传媒剧国产免费mv在线 | 无码国内精品人妻少妇 | 男人的天堂av网站 | 天堂一区人妻无码 | 免费乱码人妻系列无码专区 | 国产香蕉97碰碰久久人人 | 永久黄网站色视频免费直播 | 少妇厨房愉情理9仑片视频 | 久久99精品国产.久久久久 | 亚洲中文字幕无码中字 | 国产精品亚洲五月天高清 | 国产亚洲人成在线播放 | 国内精品人妻无码久久久影院 | 成人性做爰aaa片免费看不忠 | 亚洲va欧美va天堂v国产综合 | 欧美日韩亚洲国产精品 | 精品无码一区二区三区的天堂 | 国产精品久久久午夜夜伦鲁鲁 | 亚洲精品中文字幕久久久久 | 在线天堂新版最新版在线8 | 国产色视频一区二区三区 | 国产精品久久精品三级 | 亚洲精品美女久久久久久久 | 亚洲伊人久久精品影院 | 中国女人内谢69xxxx | 婷婷五月综合缴情在线视频 | 十八禁真人啪啪免费网站 | 亚洲七七久久桃花影院 | 少妇高潮喷潮久久久影院 | 午夜精品久久久久久久久 | 4hu四虎永久在线观看 | 夜夜高潮次次欢爽av女 | 国产成人综合在线女婷五月99播放 | 国产三级精品三级男人的天堂 | 久久成人a毛片免费观看网站 | 国产又爽又猛又粗的视频a片 | 欧美国产亚洲日韩在线二区 | 中文字幕无码乱人伦 | 国产精品无码一区二区桃花视频 | 东京无码熟妇人妻av在线网址 | 99久久99久久免费精品蜜桃 | 国产精品无码久久av | 国产午夜视频在线观看 | 天天拍夜夜添久久精品大 | 大肉大捧一进一出好爽视频 | 国产精品国产自线拍免费软件 | 国产香蕉尹人综合在线观看 | 国产精品无套呻吟在线 | 亚洲人成影院在线无码按摩店 | 欧美35页视频在线观看 | 久久 国产 尿 小便 嘘嘘 | 久久精品国产大片免费观看 | 大色综合色综合网站 | 国产va免费精品观看 | 精品国偷自产在线视频 | 国产熟妇高潮叫床视频播放 | 亚洲日韩乱码中文无码蜜桃臀网站 | 日韩视频 中文字幕 视频一区 | 2020久久香蕉国产线看观看 | 免费无码一区二区三区蜜桃大 | 好屌草这里只有精品 | 无码精品国产va在线观看dvd | 无码一区二区三区在线 | 国产成人综合在线女婷五月99播放 | 国产午夜精品一区二区三区嫩草 | 午夜精品久久久久久久 | 在线精品国产一区二区三区 | 色综合久久88色综合天天 | 色婷婷香蕉在线一区二区 | 午夜精品一区二区三区在线观看 | 丰满人妻一区二区三区免费视频 | 大肉大捧一进一出好爽视频 | 成人精品视频一区二区 | 国产亚洲精品久久久ai换 | 人人妻人人澡人人爽欧美一区九九 | 天天拍夜夜添久久精品大 | 精品一区二区不卡无码av | 东北女人啪啪对白 | 无套内谢的新婚少妇国语播放 | 亚洲性无码av中文字幕 | 国产精华av午夜在线观看 | 久久久久久久女国产乱让韩 | 青草视频在线播放 | 国产疯狂伦交大片 | 国产精品久久久一区二区三区 | 久久午夜无码鲁丝片 | 天天av天天av天天透 | 国产手机在线αⅴ片无码观看 | ass日本丰满熟妇pics | 无码人妻久久一区二区三区不卡 | 国产av剧情md精品麻豆 | 麻豆国产丝袜白领秘书在线观看 | 国产 精品 自在自线 | 亚洲国产av精品一区二区蜜芽 | 捆绑白丝粉色jk震动捧喷白浆 | 国产精品久久久久影院嫩草 | 欧美刺激性大交 | 午夜熟女插插xx免费视频 | 麻豆人妻少妇精品无码专区 | 国产av人人夜夜澡人人爽麻豆 | 中文字幕 人妻熟女 | 国产在线精品一区二区三区直播 | 国产精品久久久久久久9999 | 亚洲日韩一区二区 | 亚洲精品国产精品乱码不卡 | 爽爽影院免费观看 | 免费无码午夜福利片69 | 丰满肥臀大屁股熟妇激情视频 | 国产一区二区三区日韩精品 | 青青草原综合久久大伊人精品 | 色婷婷久久一区二区三区麻豆 | 免费网站看v片在线18禁无码 | 青草青草久热国产精品 | 99久久精品国产一区二区蜜芽 | 丁香啪啪综合成人亚洲 | 久久久久久亚洲精品a片成人 | 精品成在人线av无码免费看 | 在线天堂新版最新版在线8 | 亚洲精品国偷拍自产在线观看蜜桃 | 国产高清av在线播放 | 激情人妻另类人妻伦 | 久久精品丝袜高跟鞋 | 久久久久亚洲精品中文字幕 | 亚洲天堂2017无码 | 中文字幕无码免费久久9一区9 | 亚洲男人av香蕉爽爽爽爽 | 日日摸夜夜摸狠狠摸婷婷 | 中文字幕无码人妻少妇免费 | 久久综合给合久久狠狠狠97色 | 久久精品国产99久久6动漫 | 久久久久成人精品免费播放动漫 | 国产成人无码av一区二区 | 精品人妻人人做人人爽夜夜爽 | 兔费看少妇性l交大片免费 | 曰韩少妇内射免费播放 | 亚洲a无码综合a国产av中文 | 3d动漫精品啪啪一区二区中 | 久久国产精品二国产精品 | 国产在线无码精品电影网 | 香港三级日本三级妇三级 | 一本久道高清无码视频 | 国产成人一区二区三区在线观看 | 麻豆国产人妻欲求不满 | 国产午夜无码精品免费看 | 亚洲小说图区综合在线 | 精品夜夜澡人妻无码av蜜桃 | 亚洲精品一区三区三区在线观看 | 国内揄拍国内精品少妇国语 | 国产精品久久久久9999小说 | 夜先锋av资源网站 | 精品国产aⅴ无码一区二区 | 国产成人无码av在线影院 | 久久午夜无码鲁丝片午夜精品 | 六月丁香婷婷色狠狠久久 | 人妻与老人中文字幕 | 国产精品久久久久久亚洲影视内衣 | 奇米综合四色77777久久 东京无码熟妇人妻av在线网址 | 奇米影视888欧美在线观看 | 亚洲色欲色欲天天天www | 国产熟妇另类久久久久 | 亚洲欧美国产精品专区久久 | 亚洲精品一区二区三区婷婷月 | 亚洲中文字幕在线观看 | 最近免费中文字幕中文高清百度 | av香港经典三级级 在线 | 强辱丰满人妻hd中文字幕 | 又大又硬又黄的免费视频 | 77777熟女视频在线观看 а天堂中文在线官网 | 最近的中文字幕在线看视频 | 久久综合色之久久综合 | 久久精品人妻少妇一区二区三区 | 欧美日韩一区二区三区自拍 | 成人片黄网站色大片免费观看 | 国产在热线精品视频 | 少妇人妻大乳在线视频 | 好爽又高潮了毛片免费下载 | 亚洲人成网站在线播放942 | 美女张开腿让人桶 | 久久精品女人天堂av免费观看 | 国产激情无码一区二区 | 沈阳熟女露脸对白视频 | 欧美日韩色另类综合 | 亚洲成av人综合在线观看 | 国产又粗又硬又大爽黄老大爷视 | 国产精品多人p群无码 | 国产成人综合色在线观看网站 | 欧美国产亚洲日韩在线二区 | 领导边摸边吃奶边做爽在线观看 | 亚洲中文字幕成人无码 | 日日鲁鲁鲁夜夜爽爽狠狠 | 久久久精品国产sm最大网站 | 18无码粉嫩小泬无套在线观看 | 97久久超碰中文字幕 | 亚洲精品国产a久久久久久 | a片在线免费观看 | 窝窝午夜理论片影院 | 无码乱肉视频免费大全合集 | 精品一区二区不卡无码av | 欧美人与禽zoz0性伦交 | 波多野42部无码喷潮在线 | 国产乱人伦av在线无码 | 免费看男女做好爽好硬视频 | 久久精品中文字幕一区 | 中文字幕色婷婷在线视频 | 亚洲另类伦春色综合小说 | 成人精品一区二区三区中文字幕 | 久久这里只有精品视频9 | av在线亚洲欧洲日产一区二区 | 老子影院午夜伦不卡 | 人妻插b视频一区二区三区 | 成人免费视频在线观看 | 成人欧美一区二区三区黑人免费 | 日本熟妇乱子伦xxxx | 久久精品99久久香蕉国产色戒 | 色婷婷香蕉在线一区二区 | 中文字幕无码免费久久9一区9 | 日产国产精品亚洲系列 | 成人精品视频一区二区 | 久久久久久国产精品无码下载 | 最新版天堂资源中文官网 | 欧美精品在线观看 | 中文字幕无码免费久久99 | av无码电影一区二区三区 | 色偷偷av老熟女 久久精品人妻少妇一区二区三区 | 亚洲成在人网站无码天堂 | 99久久精品午夜一区二区 | 中文毛片无遮挡高清免费 | 97精品国产97久久久久久免费 | 日本饥渴人妻欲求不满 | 天堂亚洲2017在线观看 | 亚洲综合精品香蕉久久网 | 亚洲男人av天堂午夜在 | 欧美三级不卡在线观看 | 麻豆国产丝袜白领秘书在线观看 | √天堂资源地址中文在线 | 久久精品国产一区二区三区 | 亚洲日韩av一区二区三区中文 | 亚洲综合在线一区二区三区 | 玩弄人妻少妇500系列视频 | 中文字幕人妻无码一区二区三区 | 国产色精品久久人妻 | 久久国产36精品色熟妇 | 国产三级精品三级男人的天堂 | 成人aaa片一区国产精品 | 国产内射爽爽大片视频社区在线 | 九九在线中文字幕无码 | 国产亚洲精品久久久久久久久动漫 | 久9re热视频这里只有精品 | 无码国产乱人伦偷精品视频 | 在线 国产 欧美 亚洲 天堂 | 亚洲经典千人经典日产 | 好爽又高潮了毛片免费下载 | 国内精品久久久久久中文字幕 | 国产熟妇高潮叫床视频播放 | 亚洲一区二区三区播放 | 精品久久综合1区2区3区激情 | 粉嫩少妇内射浓精videos | 国产午夜精品一区二区三区嫩草 | 亚洲无人区一区二区三区 | 精品偷拍一区二区三区在线看 | 小鲜肉自慰网站xnxx | 久久人人97超碰a片精品 | 高清不卡一区二区三区 | 亚洲乱码国产乱码精品精 | 国产精品久久久一区二区三区 | 久久午夜夜伦鲁鲁片无码免费 | 午夜精品久久久内射近拍高清 | 亚洲国产精品美女久久久久 | 中文字幕无码人妻少妇免费 | 欧美阿v高清资源不卡在线播放 | 搡女人真爽免费视频大全 | 国产农村妇女aaaaa视频 撕开奶罩揉吮奶头视频 | 麻豆国产丝袜白领秘书在线观看 | 高潮毛片无遮挡高清免费 | 久久久久久亚洲精品a片成人 | 男人扒开女人内裤强吻桶进去 | 国产黑色丝袜在线播放 | 国产亚洲人成a在线v网站 | 亚洲精品一区二区三区在线观看 | 国产成人无码av片在线观看不卡 | 无套内谢的新婚少妇国语播放 | 又大又黄又粗又爽的免费视频 | av无码久久久久不卡免费网站 | 国产成人精品久久亚洲高清不卡 | 男女下面进入的视频免费午夜 | 伊在人天堂亚洲香蕉精品区 | 少妇性l交大片欧洲热妇乱xxx | 中文精品久久久久人妻不卡 | 色综合视频一区二区三区 | 成人欧美一区二区三区 | 性生交大片免费看女人按摩摩 | 亚洲国产精品成人久久蜜臀 | 人妻互换免费中文字幕 | 国产又爽又猛又粗的视频a片 | 国产成人精品一区二区在线小狼 | 午夜嘿嘿嘿影院 | 强辱丰满人妻hd中文字幕 | 亚洲日韩av一区二区三区中文 | 日本一卡2卡3卡四卡精品网站 | 日韩精品一区二区av在线 | 99久久无码一区人妻 | 国产精品99爱免费视频 | 国产成人精品久久亚洲高清不卡 | 成人欧美一区二区三区黑人 | 亚洲日本一区二区三区在线 | 亚洲精品一区二区三区婷婷月 | 一区二区三区乱码在线 | 欧洲 | 欧美精品一区二区精品久久 | аⅴ资源天堂资源库在线 | 极品尤物被啪到呻吟喷水 | 麻豆md0077饥渴少妇 | 国精产品一区二区三区 | 日本高清一区免费中文视频 | 亚洲男人av香蕉爽爽爽爽 | 日日碰狠狠躁久久躁蜜桃 | 久久久久亚洲精品中文字幕 | 久久精品国产99精品亚洲 | 黄网在线观看免费网站 | 无码人妻精品一区二区三区不卡 | 东京热一精品无码av | 成人免费视频视频在线观看 免费 | 成人免费视频在线观看 | 亚洲日韩av一区二区三区四区 | 日韩人妻无码一区二区三区久久99 | 在线观看免费人成视频 | 对白脏话肉麻粗话av | 欧美老人巨大xxxx做受 | 免费人成在线视频无码 | 中文亚洲成a人片在线观看 | 露脸叫床粗话东北少妇 | 色五月五月丁香亚洲综合网 | 人妻aⅴ无码一区二区三区 | 中文字幕人成乱码熟女app | 奇米影视7777久久精品人人爽 | 色婷婷综合激情综在线播放 | 国产综合久久久久鬼色 | 97精品国产97久久久久久免费 | 领导边摸边吃奶边做爽在线观看 | 又色又爽又黄的美女裸体网站 | 欧美日韩一区二区三区自拍 | 无码纯肉视频在线观看 | 成年美女黄网站色大免费全看 | 丰满少妇人妻久久久久久 | 亚洲毛片av日韩av无码 | 好爽又高潮了毛片免费下载 | 4hu四虎永久在线观看 | 中文字幕日产无线码一区 | 又色又爽又黄的美女裸体网站 | 熟女少妇人妻中文字幕 | 丰满少妇熟乱xxxxx视频 | 台湾无码一区二区 | 国产精品人人爽人人做我的可爱 | 久久综合久久自在自线精品自 | 啦啦啦www在线观看免费视频 | 丰满少妇女裸体bbw | 国产肉丝袜在线观看 | 熟妇人妻激情偷爽文 | 澳门永久av免费网站 | 亚洲日韩一区二区 | 色偷偷av老熟女 久久精品人妻少妇一区二区三区 | 国产精品久久久久久久9999 | 国产深夜福利视频在线 | 人妻天天爽夜夜爽一区二区 | 亚洲成色www久久网站 | 欧美放荡的少妇 | 亚洲自偷自拍另类第1页 | 久久久中文字幕日本无吗 | 国产莉萝无码av在线播放 | 国产女主播喷水视频在线观看 | 日韩视频 中文字幕 视频一区 | 亚洲色无码一区二区三区 | 国产激情艳情在线看视频 | 亚洲 高清 成人 动漫 | 麻豆果冻传媒2021精品传媒一区下载 | 男女下面进入的视频免费午夜 | 国产乱子伦视频在线播放 | 亚洲精品中文字幕 | 国产97在线 | 亚洲 | 亚洲一区二区三区偷拍女厕 | 日本护士xxxxhd少妇 | 天堂а√在线地址中文在线 | 无套内射视频囯产 | 人妻无码久久精品人妻 | 中文字幕+乱码+中文字幕一区 | 国产激情艳情在线看视频 | 亚洲国产精品一区二区美利坚 | 精品无码国产自产拍在线观看蜜 | 67194成是人免费无码 | 一区二区传媒有限公司 | 国产精品自产拍在线观看 | 无码午夜成人1000部免费视频 | 亚洲一区二区三区国产精华液 | 动漫av网站免费观看 | 任你躁在线精品免费 | 午夜精品一区二区三区在线观看 | 欧洲美熟女乱又伦 | 无码人妻黑人中文字幕 | 国产真实乱对白精彩久久 | 国产精品人人爽人人做我的可爱 | 亚洲国产欧美国产综合一区 | 国产婷婷色一区二区三区在线 | 亚洲综合无码久久精品综合 | 国产三级精品三级男人的天堂 | 午夜无码区在线观看 | 国产成人一区二区三区别 | 无码一区二区三区在线 | 兔费看少妇性l交大片免费 | 精品一区二区不卡无码av | 国产免费无码一区二区视频 | 欧洲美熟女乱又伦 | 人妻少妇精品视频专区 | 精品无人区无码乱码毛片国产 | 黑人玩弄人妻中文在线 | 88国产精品欧美一区二区三区 | 欧美 日韩 人妻 高清 中文 | 婷婷综合久久中文字幕蜜桃三电影 | 精品aⅴ一区二区三区 | 好男人社区资源 | 久久精品国产亚洲精品 | 少妇人妻大乳在线视频 | 国产成人精品无码播放 | 国产精品久久久久9999小说 | 亚洲综合在线一区二区三区 | 亚洲人成网站免费播放 | 亚洲精品国产a久久久久久 | 亚洲精品欧美二区三区中文字幕 | 波多野结衣乳巨码无在线观看 | 免费播放一区二区三区 | 亚洲欧洲日本无在线码 | 97夜夜澡人人爽人人喊中国片 | 久久精品丝袜高跟鞋 | 精品偷拍一区二区三区在线看 | 乱中年女人伦av三区 | 国产精品va在线观看无码 | 亚洲精品一区二区三区在线 | 久久国产劲爆∧v内射 | 久久久精品欧美一区二区免费 | 久久97精品久久久久久久不卡 | 国产特级毛片aaaaaa高潮流水 | 无码人中文字幕 | 日产精品99久久久久久 | 久久天天躁夜夜躁狠狠 | 美女扒开屁股让男人桶 | 女人被男人爽到呻吟的视频 | 中文无码精品a∨在线观看不卡 | 免费播放一区二区三区 | 亚无码乱人伦一区二区 | 丰满肥臀大屁股熟妇激情视频 | 风流少妇按摩来高潮 | 久久久久免费看成人影片 | 人妻尝试又大又粗久久 | 沈阳熟女露脸对白视频 | 国产suv精品一区二区五 | 纯爱无遮挡h肉动漫在线播放 | 国产乱子伦视频在线播放 | 99久久久无码国产精品免费 | 成人试看120秒体验区 | 亚洲gv猛男gv无码男同 | 日本又色又爽又黄的a片18禁 | 国产免费观看黄av片 | 九九久久精品国产免费看小说 | 久久这里只有精品视频9 | 国产艳妇av在线观看果冻传媒 | 九九热爱视频精品 | 亚洲乱码中文字幕在线 | 国产麻豆精品精东影业av网站 | 亚洲国产成人a精品不卡在线 | 99精品无人区乱码1区2区3区 | 久久亚洲精品成人无码 | 爽爽影院免费观看 | 欧美性猛交xxxx富婆 | 国产精品无码永久免费888 | 欧美激情一区二区三区成人 | 国精产品一区二区三区 | 成人免费视频一区二区 | 亚洲男人av香蕉爽爽爽爽 | 久久午夜夜伦鲁鲁片无码免费 | 狠狠色噜噜狠狠狠狠7777米奇 | 无码av岛国片在线播放 | 色综合久久网 | 国产激情精品一区二区三区 | 性生交大片免费看l | 国产午夜无码精品免费看 | 久久99国产综合精品 | 欧美人与善在线com | 久久精品国产一区二区三区 | 久久久久99精品成人片 | 九月婷婷人人澡人人添人人爽 | 中文字幕人妻无码一区二区三区 | 国产精品va在线播放 | 亚洲精品国偷拍自产在线观看蜜桃 | 精品少妇爆乳无码av无码专区 | 日韩精品久久久肉伦网站 | 婷婷色婷婷开心五月四房播播 | 97色伦图片97综合影院 | 亚洲区欧美区综合区自拍区 | 亚洲色成人中文字幕网站 | 久久97精品久久久久久久不卡 | 女人色极品影院 | 97人妻精品一区二区三区 | 女高中生第一次破苞av | 中文字幕人妻无码一区二区三区 | 亚洲日韩av一区二区三区四区 | 亚洲大尺度无码无码专区 | 曰韩少妇内射免费播放 | 三上悠亚人妻中文字幕在线 | 亚洲精品午夜无码电影网 | 欧美日韩视频无码一区二区三 | 人妻熟女一区 | 久久久久人妻一区精品色欧美 | 国产美女精品一区二区三区 | 欧美性猛交xxxx富婆 | 欧美乱妇无乱码大黄a片 | 大屁股大乳丰满人妻 | 水蜜桃亚洲一二三四在线 | 理论片87福利理论电影 | 丁香啪啪综合成人亚洲 | 日本肉体xxxx裸交 | 无码精品国产va在线观看dvd | 激情五月综合色婷婷一区二区 | 国内精品一区二区三区不卡 | 精品国产成人一区二区三区 | 成年美女黄网站色大免费视频 | 欧美乱妇无乱码大黄a片 | 国产精品免费大片 | 免费视频欧美无人区码 | 欧美日韩综合一区二区三区 | 日本一卡2卡3卡四卡精品网站 | 国产精品无码成人午夜电影 | av无码久久久久不卡免费网站 | 亚洲色www成人永久网址 | 亚洲综合伊人久久大杳蕉 | 蜜臀av在线观看 在线欧美精品一区二区三区 | 真人与拘做受免费视频一 | 亚洲精品鲁一鲁一区二区三区 | 欧美午夜特黄aaaaaa片 | 久久国产36精品色熟妇 | 成熟女人特级毛片www免费 | 国产精品香蕉在线观看 | 97久久超碰中文字幕 | 人妻夜夜爽天天爽三区 | 98国产精品综合一区二区三区 | 又湿又紧又大又爽a视频国产 | 国精品人妻无码一区二区三区蜜柚 | 国产婷婷色一区二区三区在线 | 少妇无套内谢久久久久 | 亚洲日韩精品欧美一区二区 | 桃花色综合影院 | 人妻尝试又大又粗久久 | 久久久久久久人妻无码中文字幕爆 | 天天躁夜夜躁狠狠是什么心态 | 无码国产激情在线观看 | 亚洲色成人中文字幕网站 | 又色又爽又黄的美女裸体网站 | 欧美国产亚洲日韩在线二区 | 一本无码人妻在中文字幕免费 | 乱人伦人妻中文字幕无码久久网 | 久久久久久av无码免费看大片 | 超碰97人人射妻 | 久久精品人人做人人综合试看 | 四虎国产精品免费久久 | 国产乱人偷精品人妻a片 | 亚洲一区二区三区香蕉 | 曰本女人与公拘交酡免费视频 | 日本精品人妻无码77777 天堂一区人妻无码 | 熟妇人妻无乱码中文字幕 | 亚洲国产精品久久久天堂 | 无码人妻久久一区二区三区不卡 | 欧美猛少妇色xxxxx | 欧美自拍另类欧美综合图片区 | 亚洲成av人片天堂网无码】 | 久久无码中文字幕免费影院蜜桃 | 天堂亚洲免费视频 | 欧美freesex黑人又粗又大 | 国产性猛交╳xxx乱大交 国产精品久久久久久无码 欧洲欧美人成视频在线 | 亚洲人成影院在线观看 | 国产成人精品久久亚洲高清不卡 | 内射白嫩少妇超碰 | 好屌草这里只有精品 | 亚洲欧洲日本综合aⅴ在线 | 精品偷拍一区二区三区在线看 | 久久久久久av无码免费看大片 | 无码av岛国片在线播放 | 中国大陆精品视频xxxx | 国产成人无码av片在线观看不卡 | 国产乱人偷精品人妻a片 | 撕开奶罩揉吮奶头视频 | 午夜不卡av免费 一本久久a久久精品vr综合 | 日日碰狠狠丁香久燥 | 激情国产av做激情国产爱 | 久久精品中文字幕一区 | 在线播放亚洲第一字幕 | 亲嘴扒胸摸屁股激烈网站 | 精品乱码久久久久久久 | 无码人妻精品一区二区三区下载 | 午夜精品久久久内射近拍高清 | 日本一卡二卡不卡视频查询 | 无码中文字幕色专区 | 麻豆av传媒蜜桃天美传媒 | 中文无码伦av中文字幕 | 亚洲精品成a人在线观看 | 蜜臀av在线观看 在线欧美精品一区二区三区 | 午夜精品久久久久久久久 | 草草网站影院白丝内射 | 成人精品视频一区二区 | 国产偷自视频区视频 | 国产成人精品无码播放 | 中文字幕亚洲情99在线 | 国产热a欧美热a在线视频 | 国产免费观看黄av片 | 国产无遮挡又黄又爽免费视频 | 玩弄中年熟妇正在播放 | 男人的天堂av网站 | 精品国产乱码久久久久乱码 | 国产97人人超碰caoprom | 天堂а√在线中文在线 | 亚洲国产精品无码一区二区三区 | 无套内谢的新婚少妇国语播放 | 国产人妻人伦精品1国产丝袜 | 少妇被黑人到高潮喷出白浆 | 正在播放东北夫妻内射 | 乱人伦人妻中文字幕无码 | 帮老师解开蕾丝奶罩吸乳网站 | 狠狠色噜噜狠狠狠狠7777米奇 | 性生交大片免费看l | 欧美亚洲国产一区二区三区 | 日本精品人妻无码免费大全 | 少妇被黑人到高潮喷出白浆 | 蜜桃av蜜臀av色欲av麻 999久久久国产精品消防器材 | 欧美日本免费一区二区三区 | 亚洲一区二区三区含羞草 | 久久视频在线观看精品 | 在线观看欧美一区二区三区 | 午夜嘿嘿嘿影院 | 日韩亚洲欧美中文高清在线 | 亚洲精品综合一区二区三区在线 | 内射巨臀欧美在线视频 | 久久国语露脸国产精品电影 | 性色欲情网站iwww九文堂 | 人人妻人人澡人人爽欧美一区九九 | 永久免费观看美女裸体的网站 | 亚洲精品一区二区三区大桥未久 | 亚洲乱码国产乱码精品精 | 性做久久久久久久免费看 | 特级做a爰片毛片免费69 | 国产后入清纯学生妹 | 成人毛片一区二区 | 又大又硬又爽免费视频 | 人人澡人人妻人人爽人人蜜桃 | 欧美变态另类xxxx | 国产精品久久久av久久久 | 嫩b人妻精品一区二区三区 | 妺妺窝人体色www婷婷 | 大屁股大乳丰满人妻 | 激情内射亚州一区二区三区爱妻 | 中文字幕无码视频专区 | 精品久久久中文字幕人妻 | 久久亚洲中文字幕精品一区 | 狠狠色噜噜狠狠狠狠7777米奇 | 国产午夜无码视频在线观看 | 国语精品一区二区三区 | 免费网站看v片在线18禁无码 | 丰满护士巨好爽好大乳 | 日日噜噜噜噜夜夜爽亚洲精品 | 久久精品人妻少妇一区二区三区 | 色婷婷综合中文久久一本 | 国产精华av午夜在线观看 | 久久午夜夜伦鲁鲁片无码免费 | 蜜桃臀无码内射一区二区三区 | 日本丰满护士爆乳xxxx | 乱码av麻豆丝袜熟女系列 | 久久成人a毛片免费观看网站 | 午夜福利不卡在线视频 | 内射后入在线观看一区 | 99久久久无码国产aaa精品 | 国产亚洲精品精品国产亚洲综合 | 国产成人综合在线女婷五月99播放 | 亚洲人成影院在线无码按摩店 | 亚洲一区二区三区偷拍女厕 | 中文字幕无码av波多野吉衣 | 老太婆性杂交欧美肥老太 | 国产精品久久国产三级国 | 精品久久8x国产免费观看 | 国产精品久久久久久久影院 | 无码人妻丰满熟妇区五十路百度 | 亚洲精品国产品国语在线观看 | 精品国产av色一区二区深夜久久 | 国产 精品 自在自线 | 熟妇人妻中文av无码 | 国产亚洲精品久久久久久国模美 | 99国产精品白浆在线观看免费 | 1000部啪啪未满十八勿入下载 | 日韩欧美成人免费观看 | 国产97色在线 | 免 | 亚洲精品鲁一鲁一区二区三区 | 牲交欧美兽交欧美 | 7777奇米四色成人眼影 | 99精品久久毛片a片 | 中文字幕人妻丝袜二区 | 欧美 日韩 亚洲 在线 | 国产人妻精品午夜福利免费 | 亚洲一区二区三区在线观看网站 | 久久99精品久久久久久动态图 | 国产无遮挡吃胸膜奶免费看 | 男人扒开女人内裤强吻桶进去 | 日韩亚洲欧美精品综合 | 无码人妻出轨黑人中文字幕 | 男女爱爱好爽视频免费看 | 搡女人真爽免费视频大全 | 日本一卡2卡3卡4卡无卡免费网站 国产一区二区三区影院 | 精品国偷自产在线视频 | 午夜理论片yy44880影院 | 草草网站影院白丝内射 | 亚洲大尺度无码无码专区 | 中文字幕无码日韩欧毛 | 麻豆蜜桃av蜜臀av色欲av | 一本无码人妻在中文字幕免费 | 亚洲小说图区综合在线 | 国内精品九九久久久精品 | 日本xxxx色视频在线观看免费 | 国内精品人妻无码久久久影院 | 国产乡下妇女做爰 | 99久久久无码国产aaa精品 | 樱花草在线播放免费中文 | 精品国产成人一区二区三区 | 熟妇女人妻丰满少妇中文字幕 | 国产免费无码一区二区视频 | 精品国产一区二区三区av 性色 | 熟女少妇人妻中文字幕 | 色欲av亚洲一区无码少妇 | 国产女主播喷水视频在线观看 | 亚洲 欧美 激情 小说 另类 | 无码乱肉视频免费大全合集 | 国产av人人夜夜澡人人爽麻豆 | 女人高潮内射99精品 | 波多野结衣一区二区三区av免费 | 国产国语老龄妇女a片 | 国产福利视频一区二区 | 久激情内射婷内射蜜桃人妖 | 麻豆人妻少妇精品无码专区 | 少女韩国电视剧在线观看完整 | 又粗又大又硬又长又爽 | 少女韩国电视剧在线观看完整 | 国产真实乱对白精彩久久 | 精品国产青草久久久久福利 | 色欲综合久久中文字幕网 | 精品乱子伦一区二区三区 | 国产精品国产自线拍免费软件 | 夜夜高潮次次欢爽av女 | 精品人妻人人做人人爽 | 日产精品99久久久久久 | 蜜臀av无码人妻精品 | 娇妻被黑人粗大高潮白浆 | 免费人成在线观看网站 | 丰腴饱满的极品熟妇 | 国产97人人超碰caoprom | 亚洲日本va午夜在线电影 | 国内综合精品午夜久久资源 | 亚洲无人区一区二区三区 | 欧美日韩一区二区三区自拍 | 日日橹狠狠爱欧美视频 | 性色欲网站人妻丰满中文久久不卡 | 亚洲国产av美女网站 | 国产午夜无码精品免费看 | 国产成人人人97超碰超爽8 | 香蕉久久久久久av成人 | 国产精品资源一区二区 | 四虎国产精品免费久久 | 亚洲欧洲无卡二区视頻 | 亚洲精品成人福利网站 | 亚洲性无码av中文字幕 | 人人妻人人澡人人爽欧美一区九九 | 国产美女极度色诱视频www | 国产色xx群视频射精 | 日日橹狠狠爱欧美视频 | 精品国产福利一区二区 | 国产精品资源一区二区 | 三上悠亚人妻中文字幕在线 | 亚洲精品鲁一鲁一区二区三区 | 性欧美牲交在线视频 | 大肉大捧一进一出视频出来呀 | 日日鲁鲁鲁夜夜爽爽狠狠 | 国产三级精品三级男人的天堂 | 国产午夜福利100集发布 | 亚洲国产精品久久人人爱 | 国产精品美女久久久久av爽李琼 | 精品久久久久久亚洲精品 | 国产av剧情md精品麻豆 | 无码乱肉视频免费大全合集 | 四虎4hu永久免费 | 国产精品国产自线拍免费软件 | 亚洲精品一区二区三区四区五区 | 四虎国产精品一区二区 | 国产九九九九九九九a片 | 亚洲爆乳大丰满无码专区 | 无码av中文字幕免费放 | 久久婷婷五月综合色国产香蕉 | 久久久精品成人免费观看 | 131美女爱做视频 | 久久午夜无码鲁丝片午夜精品 | 欧美第一黄网免费网站 | 久久久婷婷五月亚洲97号色 | 亚洲精品美女久久久久久久 | 狠狠综合久久久久综合网 | 日韩精品成人一区二区三区 | 欧美 丝袜 自拍 制服 另类 | 成人一区二区免费视频 | 国产精品无码一区二区三区不卡 | 丰满岳乱妇在线观看中字无码 | 动漫av一区二区在线观看 | 丰满人妻翻云覆雨呻吟视频 | 人妻天天爽夜夜爽一区二区 | 蜜桃臀无码内射一区二区三区 | 国产成人无码av在线影院 | 一个人看的www免费视频在线观看 | 少妇被黑人到高潮喷出白浆 | 国产成人综合在线女婷五月99播放 | 亚洲自偷自拍另类第1页 |