日韩av黄I国产麻豆传媒I国产91av视频在线观看I日韩一区二区三区在线看I美女国产在线I麻豆视频国产在线观看I成人黄色短片

歡迎訪問 生活随笔!

生活随笔

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

编程问答

aspectj 获取方法入参_深入探索编译插桩技术(二、AspectJ)

發布時間:2024/9/19 编程问答 32 豆豆
生活随笔 收集整理的這篇文章主要介紹了 aspectj 获取方法入参_深入探索编译插桩技术(二、AspectJ) 小編覺得挺不錯的,現在分享給大家,幫大家做個參考.

本文來自jsonchao的投稿,個人微信:bcce5360? ? ?? ? ? ?

現如今,編譯插樁技術已經深入 Android 開發中的各個領域,而 AOP 技術正是一種高效實現插樁的模式,它的出現正好給處于黑暗中的我們帶來了光明,極大地解決了傳統開發過程中的一些痛點,而 AspectJ 作為一套基于 Java 語言面向切面的擴展設計規范,能夠賦予我們新的能力。在這篇文章我們將來學習如何使用 AspectJ 來進行插樁。本篇內容如下所示:

  • 1)、編譯插樁技術的分類與應用場景。

  • 2)、AspectJ 的優勢與局限性。

  • 3)、AspectJ 核心語法簡介

  • 4)、AspectJX 實戰。

  • 5)、使用 AspectJX 打造自己的性能監控框架。

  • 6)、總結。

面向切面的程序設計 (aspect-oriented programming (AOP)) 吸引了很多開發者的目光, 但是如何在編碼中有效地實現這一套設計概念卻并不簡單,幸運的是,早在 2003 年,一套基于 Java 語言面向切面的擴展設計:AspectJ 誕生了。

不同與傳統的 OOP 編程,AspectJ (即 AOP) 的獨特之處在于 發現那些使用傳統編程方法無法處理得很好的問題。例如一個要在某些應用中實施安全策略的問題。安全性是貫穿于系統所有模塊間的問題,而且每一個模塊都必須要添加安全性才能保證整個應用的安全性,并且安全性模塊自身也需要安全性,很明顯這里的 安全策略的實施問題就是一個橫切關注點,使用傳統的編程解決此問題非常的困難而且容易產生差錯,這正是 AOP 發揮作用的時候了。傳統的面向對象編程中,每個單元就是一個類,而 類似于安全性這方面的問題,它們通 常不能集中在一個類中處理,因為它們橫跨多個類,這就導致了代碼無法重用,它們是不可靠和不可繼承的,這樣的編程方式使得可維護性差而且產生了大量的代碼冗余,這是我們所不愿意看到的

而面向切面編程的出現正好給處于黑暗中的我們帶來了光明,它針對于這些橫切關注點進行處理,就似面向對象編程處理一般的關注點一樣

在我繼續講解 AOP 編程之前,我們有必要先來看看當前編譯插樁技術的分類與應用場景。這樣能讓我們 從更高的緯度上去理解各個技術點之間的關聯與作用

一、編譯插樁技術的分類與應用場景

編譯插樁技術具體可以分為兩類,如下所示:

  • 1)、APT(Annotation Process Tools) :用于生成 Java 代碼。

  • 2)、AOP(Aspect Oriented Programming):用于操作字節碼

下面?,我們分別來詳細介紹下它們的作用。

1、APT(Annotation Process Tools)

總所周知,ButterKnife、Dagger、GreenDao、Protocol Buffers 這些常用的注解生成框架都會在編譯過程中生成代碼。而 這種使用 AndroidAnnotation 結合 APT 技術 來生成代碼的時機,是在編譯最開始的時候介入的。而 AOP 是在編譯完成后生成 dex 文件之前的時候,直接通過修改 .class 文件的方式,來直接添加或者修改代碼邏輯的。

使用 APT 技術生成 Java 代碼的方式具有如下 兩方面 的優勢:

  • 1)、隔離了框架復雜的內部實現,使得開發更加地簡單高效。

  • 2)、大大減少了手工重復的工作量,降低了開發時出錯的機率。

2、AOP(Aspect Oriented Programming)

而對于操作字節碼的方式來說,一般都在 代碼監控、代碼修改、代碼分析 這三個場景有著很廣泛的應用。

相對于 Java 代碼生成的方式,操作字節碼的方式有如下 特點

  • 1)、應用場景更廣。

  • 2)、功能更加強大

  • 3)、使用復雜度較高。

此外,我們不僅可以操作 .class 文件的 Java 字節碼,也可以操作 .dex 文件的 Dalvik 字節碼。下面我們就來大致了解下在以上三類場景中編譯插樁技術具體是如何應用的。

1、代碼監控

編譯插樁技術除了 不能夠實現耗電監控,它能夠實現各式各樣的性能監控,例如:網絡數據監控、耗時方法監控、大圖監控、線程監控 等等。

譬如 網絡數據監控 的實現,就是在 網絡層通過 hook 網絡庫方法 和 自動化注入攔截器的形式,實現網絡請求的全過程監控,包括獲取握手時長,首包時間,DNS 耗時,網絡耗時等各個網絡階段的信息。

實現了對網絡請求過程的監控之后,我們便可以 對整個網絡過程的數據表現進行詳細地分析,找到網絡層面性能的問題點,并做出針對性地優化措施。例如針對于 網絡錯誤率偏高 的問題,我們可以采取以下幾方面的措施,如下所示:

  • 1)、使用 HttpDNS

  • 2)、將錯誤數據同步 CDN。

  • 3)、CDN 調度鏈路優化。

2、代碼修改

用編譯插樁技術來實現代碼修改的場景非常之多,而使用最為頻繁的場景具體可細分為為如下四種:

  • 1)、實現無痕埋點:如網易HubbleData之Android無埋點實踐、51 信用卡 Android 自動埋點實踐。

  • 2)、統一處理點擊抖動:編譯階段統一 hook ?android.view.View.OnClickListener#onClick() 方法,來實現一個快速點擊無效的防抖動效果,這樣便能高效、無侵入性地統一解決客戶端快速點擊多次導致頻繁響應的問題

  • 3)、第三方 SDK 的容災處理:我們可以在上線前臨時修改或者 hook 第三方 SDK 的方法,做到快速容災上線

  • 4)、實現熱修復框架:我們可以在 Gradle 進行自動化構建的時候,即在 Java 源碼編譯完成之后,生成 dex 文件之前進行插樁,而插樁的作用是在每個方法執行時先去根據自己方法的簽名尋找是否有自己對應的 patch 方法,如果有,執行 patch 方法;如果沒有,則執行自己原有的邏輯。

3、代碼分析

例如 Findbugs 等三方的代碼檢查工具里面的 自定義代碼檢查 也使用了編譯插樁技術,利用它我們可以找出 不合理的 Hanlder 使用、new Thread 調用、敏感權限調用 等等一系列編碼問題。

二、AspectJ 的優勢與局限性

最常用的字節碼處理框架有 AspectJ、ASM 等等,它們的相同之處在于輸入輸出都是 Class 文件。并且,它們都是 在 Java 文件編譯成 .class 文件之后,生成 Dalvik 字節碼之前執行

AspectJ 作為 Java 中流行的 AOP(aspect-oriented programming) 編程擴展框架,其內部使用的是 BCEL框架 來完成其功能。下面,我們就來了解下 AspectJ 具備哪些優勢。

1、AspectJ 的優勢

它的優勢有兩點:成熟穩定、使用非常簡單。

1、成熟穩定

字節碼的處理并不簡單,特別是 針對于字節碼的格式和各種指令規則,如果處理出錯,就會導致程序編譯或者運行過程中出現問題。而 AspectJ 作為從 2001 年發展至今的框架,它已經發展地非常成熟,通常不用考慮插入的字節碼發生正確性相關的問題。

2、使用非常簡單

AspectJ 的使用非常簡單,并且它的功能非常強大,我們完全不需要理解任何 Java 字節碼相關的知識,就可以在很多情況下對字節碼進行操控。例如,它可以在如下五個位置插入自定義的代碼:

  • 1)、在方法(包括構造方法)被調用的位置。

  • 2)、在方法體(包括構造方法)的內部。

  • 3)、在讀寫變量的位置。

  • 4)、在靜態代碼塊內部。

  • 5)、在異常處理的位置的前后。

此外,它也可以 直接將原位置的代碼替換為自定義的代碼。

2、AspectJ 的缺陷

而 AspectJ 的缺點可以歸結為如下 三點

1、切入點固定

AspectJ 只能在一些固定的切入點來進行操作,如果想要進行更細致的操作則很難實現,它無法針對一些特定規則的字節碼序列做操作。

2、正則表達式的局限性

AspectJ 的匹配規則采用了類似正則表達式的規則,比如 匹配 Activity 生命周期的 onXXX 方法,如果有自定義的其他以 on 開頭的方法也會匹配到,這樣匹配的正確性就無法滿足

3、性能較低

AspectJ 在實現時會包裝自己一些特定的類,它并不會直接把 Trace 函數直接插入到代碼中,而是經過一系列自己的封裝。這樣不僅生成的字節碼比較大,而且對原函數的性能會有不小的影響。如果想對 App 中所有的函數都進行插樁,性能影響肯定會比較大。如果你只插樁一小部分函數,那么 AspectJ 帶來的性能損耗幾乎可以忽略不計。

三、AspectJ 核心語法簡介

AspectJ 其實就是一種 AOP 框架,AOP 是實現程序功能統一維護的一種技術。利用 AOP 可以對業務邏輯的各個部分進行隔離,從而使得業務邏輯各部分之間的耦合性降低,提高程序的可重用性,同時大大提高了開發效率。因此 AOP 的優勢可總結為如下 兩點

  • 1)、無侵入性。

  • 2)、修改方便。

此外,AOP 不同于 OOP 將問題劃分到單個模塊之中,它把 涉及到眾多模塊的同一類問題進行了統一處理。比如我們可以設計兩個切面,一個是用于處理 App 中所有模塊的日志輸出功能,另外一個則是用于處理 App 中一些特殊函數調用的權限檢查。

下面?,我們就來看看要掌握 AspectJ 的使用,我們需要了解的一些 核心概念

1、橫切關注點

對哪些方法進行攔截,攔截后怎么處理。

2、切面(Aspect)

類是對物體特征的抽象,切面就是對橫切關注點的抽象。

3、連接點(JoinPoint)

JPoint 是一個程序的關鍵執行點,也是我們關注的重點。它就是指被攔截到的點(如方法、字段、構造器等等)。

4、切入點(PointCut)

對 JoinPoint 進行攔截的定義。PointCut 的目的就是提供一種方法使得開發者能夠選擇自己感興趣的 JoinPoint。

5、通知(Advice)

切入點僅用于捕捉連接點集合,但是,除了捕捉連接點集合以外什么事情都沒有做。事實上實現橫切行為我們要使用通知。它 一般指攔截到 JoinPoint 后要執行的代碼,分為 前置、后置、環繞 三種類型。這里,我們需要注意 Advice Precedence(優先權) 的情況,比如我們對同一個切面方法同時使用了 @Before 和 @Around 時就會報錯,此時會提示需要設置 Advice 的優先級。

AspectJ 作為一種基于 Java 語言實現的一套面向切面程序設計規范。它向 Java 中加入了 連接點(Join Point) 這個新概念,其實它也只是現存的一個 Java 概 念的名稱而已。它向 Java 語言中加入了少許新結構,譬如 切入點(pointcut)、通知(Advice)、類型間聲明(Inter-type declaration) 和 切面(Aspect)。切入點和通知動態地影響程序流程,類型間聲明則是 靜態的影響程序的類等級結構,而切面則是對所有這些新結構的封裝。

對于 AsepctJ 中的各個核心概念來說,其 連接點就恰如程序流中適當的一點。而切入點收集特定的連接點集合和在這些點中的值。一個通知則是當一個連接點到達時執行的代碼,這些都是 AspectJ 的動態部分。其實連接點就好比是 程序中那一條一條的語句,而切入點就是特定一條語句處設置的一個斷點,它收集了斷點處程序棧的信息,而通知就是在這個斷點前后想要加入的程序代碼。

此外,AspectJ 中也有許多不同種類的類型間聲明,這就允許程序員修改程序的靜態結構、名稱、類的成員以及類之間的關系。AspectJ 中的切面是橫切關注點的模塊單元。它們的行為與 Java 語言中的類很象,但是切面 還封裝了切入點、通知以及類型間聲明。

在 Android 平臺上要使用 AspectJ 還是有點麻煩的,這里我們可以直接使用滬江的 AspectJX 框架。下面,我們就來使用 AspectJX 進行 AOP 切面編程。

四、AspectJX 實戰

首先,為了在 Android 使用 AOP 埋點需要引入 AspectJX,在項目根目錄的 build.gradle 下加入:

classpath 'com.hujiang.aspectjx:gradle-android-plugin- aspectjx:2.0.0'

然后,在 app 目錄下的 build.gradle 下加入:

apply plugin: 'android-aspectjx'implement 'org.aspectj:aspectjrt:1.8.+'

JoinPoint 一般定位在如下位置:

  • 1)、函數調用。

  • 2)、獲取、設置變量。

  • 3)、類初始化

使用 PointCut 對我們指定的連接點進行攔截,通過 Advice,就可以攔截到 JoinPoint 后要執行的代碼。Advice 通常有以下 三種類型

  • 1)、Before:PointCut 之前執行。

  • 2)、After:PointCut 之后執行。

  • 3)、Around:PointCut 之前、之后分別執行。

1、最簡單的 AspectJ 示例

首先,我們舉一個 小栗子?:

@Before("execution(* android.app.Activity.on**(..))")public void onActivityCalled(JoinPoint joinPoint) throws Throwable { Log.d(...)}

其中,在 execution 中的是一個匹配規則,第一個 * 代表匹配任意的方法返回值,后面的語法代碼匹配所有 Activity 中以 on 開頭的方法。這樣,我們就可以 在 App 中所有 Activity 中以 on 開頭的方法中輸出一句 log

上面的 execution 就是處理 Join Point 的類型,通常有如下兩種類型:

  • 1)、call:代表調用方法的位置,插入在函數體外面。

  • 2)、execution:代表方法執行的位置,插入在函數體內部。

2、統計 Application 中所有方法的耗時

那么,我們如何利用它統計?Application?中的所有方法耗時呢?

@Aspectpublic class ApplicationAop { @Around("call (* com.json.chao.application.BaseApplication.**(..))") public void getTime(ProceedingJoinPoint joinPoint) { Signature signature = joinPoint.getSignature(); String name = signature.toShortString(); long time = System.currentTimeMillis(); try { joinPoint.proceed(); } catch (Throwable throwable) { throwable.printStackTrace(); } Log.i(TAG, name + " cost" + (System.currentTimeMillis() - time)); }}

需要注意的是,當 Action 為 Before、After 時,方法入參為 JoinPoint。當 Action 為 Around 時,方法入參為 ProceedingPoint。

而 Around 和 Before、After 的最大區別就是 ProceedingPoint 不同于 JoinPoint,其提供了 proceed 方法執行目標方法。

3、對 App 中所有的方法進行 Systrace 函數插樁

@Aspectpublic class SystraceTraceAspectj { private static final String TAG = "SystraceTraceAspectj"; @Before("execution(* **(..))") public void before(JoinPoint joinPoint) { TraceCompat.beginSection(joinPoint.getSignature().toString()); } @After("execution(* **(..))") public void after() { TraceCompat.endSection(); }}

了解了 AspectJX 的基本使用之后,接下來我們就會使用它和 AspectJ 去打造一個簡易版的 APM(性能監控框架)。

五、使用 AspectJ 打造自己的性能監控框架

現在,我們將以奇虎360的 ArgusAPM 性能監控框架來全面分析下 AOP 技術在性能監控方面的應用。主要分為 三個部分

  • 1)、監控應用冷熱啟動耗時與生命周期耗時。

  • 2)、監控 OKHttp3 的每一次網絡請求。

  • 3)、監控 HttpConnection 的每一次網絡請求

1、監控應用冷熱啟動耗時與生命周期耗時

在 ArgusAPM 中,實現了 Activity 切面文件 TraceActivity, 它被用來監控應用冷熱啟動耗時與生命周期耗時,TraceActivity 的實現代碼如下所示:

@Aspectpublic class TraceActivity { // 1、定義一個切入點方法 baseCondition,用于排除 argusapm 中相應的類。 @Pointcut("!within(com.argusapm.android.aop.*) && !within(com.argusapm.android.core.job.activity.*)") public void baseCondition() { } // 2、定義一個切入點 applicationOnCreate,用于執行 Application 的 onCreate方法。 @Pointcut("execution(* android.app.Application.onCreate(android.content.Context)) && args(context)") public void applicationOnCreate(Context context) { } // 3、定義一個后置通知 applicationOnCreateAdvice,用于在 application 的 onCreate 方法執行完之后插入 AH.applicationOnCreate(context) 這行代碼。 @After("applicationOnCreate(context)") public void applicationOnCreateAdvice(Context context) { AH.applicationOnCreate(context); } // 4、定義一個切入點,用于執行 Application 的 attachBaseContext 方法。 @Pointcut("execution(* android.app.Application.attachBaseContext(android.content.Context)) && args(context)") public void applicationAttachBaseContext(Context context) { } // 5、定義一個前置通知,用于在 application 的 onAttachBaseContext 方法之前插入 AH.applicationAttachBaseContext(context) 這行代碼。 @Before("applicationAttachBaseContext(context)") public void applicationAttachBaseContextAdvice(Context context) { AH.applicationAttachBaseContext(context); } // 6、定義一個切入點,用于執行所有 Activity 中以 on 開頭的方法,后面的 ”&& baseCondition()“ 是為了排除 ArgusAPM 中的類。 @Pointcut("execution(* android.app.Activity.on**(..)) && baseCondition()") public void activityOnXXX() { } // 7、定義一個環繞通知,用于在所有 Activity 的 on 開頭的方法中的開始和結束處插入相應的代碼。(排除了 ArgusAPM 中的類) @Around("activityOnXXX()") public Object activityOnXXXAdvice(ProceedingJoinPoint proceedingJoinPoint) { Object result = null; try { Activity activity = (Activity) proceedingJoinPoint.getTarget(); // Log.d("AJAOP", "Aop Info" + activity.getClass().getCanonicalName() + // "\r\nkind : " + thisJoinPoint.getKind() + // "\r\nargs : " + thisJoinPoint.getArgs() + // "\r\nClass : " + thisJoinPoint.getClass() + // "\r\nsign : " + thisJoinPoint.getSignature() + // "\r\nsource : " + thisJoinPoint.getSourceLocation() + // "\r\nthis : " + thisJoinPoint.getThis() // ); long startTime = System.currentTimeMillis(); result = proceedingJoinPoint.proceed(); String activityName = activity.getClass().getCanonicalName(); Signature signature = proceedingJoinPoint.getSignature(); String sign = ""; String methodName = ""; if (signature != null) { sign = signature.toString(); methodName = signature.getName(); } if (!TextUtils.isEmpty(activityName) && !TextUtils.isEmpty(sign) && sign.contains(activityName)) { invoke(activity, startTime, methodName, sign); } } catch (Exception e) { e.printStackTrace(); } catch (Throwable throwable) { throwable.printStackTrace(); } return result; } public void invoke(Activity activity, long startTime, String methodName, String sign) { AH.invoke(activity, startTime, methodName, sign); }}

我們注意到,在注釋4、5這兩處代碼是用于 在 application 的 onAttachBaseContext 方法之前插入 AH.applicationAttachBaseContext(context) 這行代碼。此外,注釋2、3兩處的代碼是用于 在 application 的 onCreate 方法執行完之后插入 AH.applicationOnCreate(context) 這行代碼。下面,我們再看看 AH 類中這兩個方法的實現,代碼如下所示:

public static void applicationAttachBaseContext(Context context) { ActivityCore.appAttachTime = System.currentTimeMillis(); if (Env.DEBUG) { LogX.d(Env.TAG, SUB_TAG, "applicationAttachBaseContext time : " + ActivityCore.appAttachTime); }}public static void applicationOnCreate(Context context) { if (Env.DEBUG) { LogX.d(Env.TAG, SUB_TAG, "applicationOnCreate"); }}

可以看到,在 AH 類的 applicationAttachBaseContext 方法中將啟動時間 appAttachTime 記錄到了 ActivityCore 實例中。而 applicationOnCreate 基本上什么也沒有實現。

然后,我們再回到切面文件 TraceActivity 中,看到注釋6、7處的代碼,這里用于 在所有 Activity 的 on 開頭的方法中的開始和結束處插入相應的代碼。需要注意的是,這里 排除了 ArgusAPM 中的類

下面,我們來分析下 activityOnXXXAdvice 方法中的操作。首先,在目標方法執行前獲取了 startTime。然后,調用了 proceedingJoinPoint.proceed() 用于執行目標方法;最后,調用了 AH 類的 invoke 方法。我們看看 invoke 方法的處理,代碼如下所示:

public static void invoke(Activity activity, long startTime, String lifeCycle, Object... extars) { // 1 boolean isRunning = isActivityTaskRunning(); if (Env.DEBUG) { LogX.d(Env.TAG, SUB_TAG, lifeCycle + " isRunning : " + isRunning); } if (!isRunning) { return; } // 2 if (TextUtils.equals(lifeCycle, ActivityInfo.TYPE_STR_ONCREATE)) { ActivityCore.onCreateInfo(activity, startTime); } else { // 3 int lc = ActivityInfo.ofLifeCycleString(lifeCycle); if (lc <= ActivityInfo.TYPE_UNKNOWN || lc > ActivityInfo.TYPE_DESTROY) { return; } ActivityCore.saveActivityInfo(activity, ActivityInfo.HOT_START, System.currentTimeMillis() - startTime, lc); }}

首先,在注釋1處,我們會先去查看當前應用的 Activity 耗時統計任務是否打開了。如果打開了,然后就會走到注釋2處,這里 會先判斷目標方法名稱是否是 “onCreate”,如果是 onCreate 方法,就會執行 ActivityCore 的 onCreateInfo 方法,代碼如下所示:

// 是否是第一次啟動public static boolean isFirst = true;public static long appAttachTime = 0;// 啟動類型public static int startType;public static void onCreateInfo(Activity activity, long startTime) { // 1 startType = isFirst ? ActivityInfo.COLD_START : ActivityInfo.HOT_START; // 2 activity.getWindow().getDecorView().post(new FirstFrameRunnable(activity, startType, startTime)); //onCreate 時間 long curTime = System.currentTimeMillis(); // 3 saveActivityInfo(activity, startType, curTime - startTime, ActivityInfo.TYPE_CREATE);}

首先,在注釋1處,會 記錄此時的啟動類型,第一次默認是冷啟動。然后在注釋2處,當第一幀顯示時會 post 一個 Runnable。最后,在注釋3處,會 調用 saveActivityInfo 將目標方法相關的信息保存起來。這里我們先看看這個 FirstFrameRunnable 的 run 方法的實現代碼,如下所示:

@Override public void run() { if (DEBUG) { LogX.d(TAG, SUB_TAG, "FirstFrameRunnable time:" + (System.currentTimeMillis() - startTime)); } // 1 if ((System.currentTimeMillis() - startTime) >= ArgusApmConfigManager.getInstance().getArgusApmConfigData().funcControl.activityFirstMinTime) { saveActivityInfo(activity, startType, System.currentTimeMillis() - startTime, ActivityInfo.TYPE_FIRST_FRAME); } if (DEBUG) { LogX.d(TAG, SUB_TAG, "FirstFrameRunnable time:" + String.format("[%s, %s]", ActivityCore.isFirst, ActivityCore.appAttachTime)); } if (ActivityCore.isFirst) { ActivityCore.isFirst = false; if (ActivityCore.appAttachTime <= 0) { return; } // 2 int t = (int) (System.currentTimeMillis() - ActivityCore.appAttachTime); AppStartInfo info = new AppStartInfo(t); ITask task = Manager.getInstance().getTaskManager().getTask(ApmTask.TASK_APP_START); if (task != null) { // 3 task.save(info); if (AnalyzeManager.getInstance().isDebugMode()) { // 4 AnalyzeManager.getInstance().getParseTask(ApmTask.TASK_APP_START).parse(info); } } else { if (DEBUG) { LogX.d(TAG, SUB_TAG, "AppStartInfo task == null"); } } } }}

首先,在注釋1處,會計算出當前的 第一幀的時間,即 當前 Activity 的冷啟動時間,將它與 activityFirstMinTime 這個值作比較(activityFirstMinTime 的值默認為300ms),如果 Activity 的冷啟動時間大于300ms的話,就會將冷啟動時間調用 saveActivityInfo 方法保存起來

然后,在注釋2處,我們會 記錄 App 的啟動時間 并在注釋3處將它 保存到 AppStartTask 這個任務實例中。最后,在注釋4處,如果是 debug 模式,則會調用 AnalyzeManager 這個數據分析管理單例類的 getParseTask 方法獲取 AppStartParseTask 這個實例,關鍵代碼如下所示:

private Map mParsers;private AnalyzeManager() { mParsers = new HashMap(3); mParsers.put(ApmTask.TASK_ACTIVITY, new ActivityParseTask()); mParsers.put(ApmTask.TASK_NET, new NetParseTask()); mParsers.put(ApmTask.TASK_FPS, new FpsParseTask()); mParsers.put(ApmTask.TASK_APP_START, new AppStartParseTask()); mParsers.put(ApmTask.TASK_MEM, new MemoryParseTask()); this.isUiProcess = Manager.getContext().getPackageName().equals(ProcessUtils.getCurrentProcessName());}public IParser getParseTask(String name) { if (TextUtils.isEmpty(name)) { return null; } return mParsers.get(name);}

接著,就會調用 AppStartParseTask 類的 parse 方法,可以看出,它是一個 專門用于在 Debug 模式下的應用啟動時間分析類。parse 方法的代碼如下所示:

/** * app啟動 * * @param info */@Overridepublic boolean parse(IInfo info) { if (info instanceof AppStartInfo) { AppStartInfo aInfo = (AppStartInfo) info; if (aInfo == null) { return false; } try { JSONObject obj = aInfo.toJson(); obj.put("taskName", ApmTask.TASK_APP_START); // 1 OutputProxy.output("啟動時間:" + aInfo.getStartTime(), obj.toString()); } catch (JSONException e) { e.printStackTrace(); } DebugFloatWindowUtls.sendBroadcast(aInfo); } return true;}

在注釋1處,parse 方法中僅僅是繼續調用了 OutputProxy 的 output 方法 將啟動時間和記錄啟動信息的字符串傳入。我們再看看 OutputProxy 的 output 方法,如下所示:

/** * 警報信息輸出 * * @param showMsg */public static void output(String showMsg) { if (!AnalyzeManager.getInstance().isDebugMode()) { return; } if (TextUtils.isEmpty(showMsg)) { return; } // 1、存儲在本地 StorageManager.saveToFile(showMsg);}

注釋1處,在 output 方法中又繼續調用了 StorageManager 的 saveToFile 方法 將啟動信息存儲在本地,saveToFile 的實現代碼如下所示:

/** * 按行保存到文本文件 * * @param line */public static void saveToFile(String line) { TraceWriter.log(Env.TAG, line);}

這里又調用了 TraceWriter 的 log 方法 將啟動信息按行保存到文本文件中,關鍵代碼如下所示:

public static void log(String tagName, String content) { log(tagName, content, true);}private synchronized static void log(String tagName, String content, boolean forceFlush) { if (Env.DEBUG) { LogX.d(Env.TAG, SUB_TAG, "tagName = " + tagName + " content = " + content); } if (sWriteThread == null) { // 1 sWriteThread = new WriteFileRun(); Thread t = new Thread(sWriteThread); t.setName("ApmTrace.Thread"); t.setDaemon(true); t.setPriority(Thread.MIN_PRIORITY); t.start(); String initContent = "---- Phone=" + Build.BRAND + "/" + Build.MODEL + "/verName:" + " ----"; // 2 sQueuePool.offer(new Object[]{tagName, initContent, Boolean.valueOf(forceFlush)}); if (Env.DEBUG) { LogX.d(Env.TAG, SUB_TAG, "init offer content = " + content); } } if (Env.DEBUG) { LogX.d(Env.TAG, SUB_TAG, "offer content = " + content); } // 3 sQueuePool.offer(new Object[]{tagName, content, Boolean.valueOf(forceFlush)}); synchronized (LOCKER_WRITE_THREAD) { LOCKER_WRITE_THREAD.notify(); }}

在注釋1處,如果 sWriteThread 這個負責寫入 log 信息的 Runnable 不存在,就會新建并啟動這個寫入 log 信息的低優先級守護線程

然后,會在注釋2處,調用 sQueuePool 的 offer 方法將相關的信息保存,它的類型為 ConcurrentLinkedQueue,說明它是一個專用于并發環境下的隊列。如果 Runnable 已經存在了的話,就直接會在注釋3處將 log 信息入隊。最終,會在 sWriteThread 的 run 方法中調用 sQueuePool 的 poll() 方法將 log 信息拿出并通過 BufferWriter 封裝的 FileWriter 將信息保存在本地。

到此,我們就分析完了 onCreate 方法的處理,接著我們再回到 invoke 方法的注釋3處來分析不是 onCreate 方法的情況。如果方法名不是 onCreate 方法的話,就會調用 ActivityInfo 的 ofLifeCycleString 方法,我們看看它的實現,如下所示:

/** * 生命周期字符串轉換成數值 * * @param lcStr * @return */public static int ofLifeCycleString(String lcStr) { int lc = 0; if (TextUtils.equals(lcStr, TYPE_STR_FIRSTFRAME)) { lc = TYPE_FIRST_FRAME; } else if (TextUtils.equals(lcStr, TYPE_STR_ONCREATE)) { lc = TYPE_CREATE; } else if (TextUtils.equals(lcStr, TYPE_STR_ONSTART)) { lc = TYPE_START; } else if (TextUtils.equals(lcStr, TYPE_STR_ONRESUME)) { lc = TYPE_RESUME; } else if (TextUtils.equals(lcStr, TYPE_STR_ONPAUSE)) { lc = TYPE_PAUSE; } else if (TextUtils.equals(lcStr, TYPE_STR_ONSTOP)) { lc = TYPE_STOP; } else if (TextUtils.equals(lcStr, TYPE_STR_ONDESTROY)) { lc = TYPE_DESTROY; } return lc;}

可以看到,ofLifeCycleString 的作用就是將生命周期字符串轉換成相應的數值,下面是它們的定義代碼:

/** * Activity 生命周期類型枚舉 */public static final int TYPE_UNKNOWN = 0;public static final int TYPE_FIRST_FRAME = 1;public static final int TYPE_CREATE = 2;public static final int TYPE_START = 3;public static final int TYPE_RESUME = 4;public static final int TYPE_PAUSE = 5;public static final int TYPE_STOP = 6;public static final int TYPE_DESTROY = 7;/** * Activity 生命周期類型值對應的名稱 */public static final String TYPE_STR_FIRSTFRAME = "firstFrame";public static final String TYPE_STR_ONCREATE = "onCreate";public static final String TYPE_STR_ONSTART = "onStart";public static final String TYPE_STR_ONRESUME = "onResume";public static final String TYPE_STR_ONPAUSE = "onPause";public static final String TYPE_STR_ONSTOP = "onStop";public static final String TYPE_STR_ONDESTROY = "onDestroy";public static final String TYPE_STR_UNKNOWN = "unKnown";

然后,我們再回到 AH 類的 invoke 方法的注釋3處,僅僅當方法名是上述定義的方法,也就是 Acitivity 的生命周期方法或第一幀的方法時,才會調用 ActivityCore 的 saveActivityInfo 方法。該方法的實現代碼如下所示:

public static void saveActivityInfo(Activity activity, int startType, long time, int lifeCycle) { if (activity == null) { if (DEBUG) { LogX.d(TAG, SUB_TAG, "saveActivityInfo activity == null"); } return; } if (time < ArgusApmConfigManager.getInstance().getArgusApmConfigData().funcControl.activityLifecycleMinTime) { return; } String pluginName = ExtraInfoHelper.getPluginName(activity); String activityName = activity.getClass().getCanonicalName(); activityInfo.resetData(); activityInfo.activityName = activityName; activityInfo.startType = startType; activityInfo.time = time; activityInfo.lifeCycle = lifeCycle; activityInfo.pluginName = pluginName; activityInfo.pluginVer = ExtraInfoHelper.getPluginVersion(pluginName); if (DEBUG) { LogX.d(TAG, SUB_TAG, "apmins saveActivityInfo activity:" + activity.getClass().getCanonicalName() + " | lifecycle : " + activityInfo.getLifeCycleString() + " | time : " + time); } ITask task = Manager.getInstance().getTaskManager().getTask(ApmTask.TASK_ACTIVITY); boolean result = false; if (task != null) { result = task.save(activityInfo); } else { if (DEBUG) { LogX.d(TAG, SUB_TAG, "saveActivityInfo task == null"); } } if (DEBUG) { LogX.d(TAG, SUB_TAG, "activity info:" + activityInfo.toString()); } if (AnalyzeManager.getInstance().isDebugMode()) { AnalyzeManager.getInstance().getActivityTask().parse(activityInfo); } if (Env.DEBUG) { LogX.d(TAG, SUB_TAG, "saveActivityInfo result:" + result); }}

可以看到,這里的邏輯很簡單,僅僅是 將 log 信息保存在 ActivityInfo 這個實例中,并將 ActivityInfo 實例保存在了 ActivityTask 中,需要注意的是,在調用 ArgusAPM.init() 這句初始化代碼時就已經將 ActivityTask 實例保存在了 taskMap 這個 HashMap 對象中 了,關鍵代碼如下所示:

/** * 注冊 task:每添加一個task都要進行注冊,也就是把 * 相應的 xxxTask 實例放入 taskMap 集合中。 */public void registerTask() { if (Env.DEBUG) { LogX.d(Env.TAG, "TaskManager", "registerTask " + getClass().getClassLoader()); } if (Build.VERSION.SDK_INT >= 16) { taskMap.put(ApmTask.TASK_FPS, new FpsTask()); } taskMap.put(ApmTask.TASK_MEM, new MemoryTask()); taskMap.put(ApmTask.TASK_ACTIVITY, new ActivityTask()); taskMap.put(ApmTask.TASK_NET, new NetTask()); taskMap.put(ApmTask.TASK_APP_START, new AppStartTask()); taskMap.put(ApmTask.TASK_ANR, new AnrLoopTask(Manager.getContext())); taskMap.put(ApmTask.TASK_FILE_INFO, new FileInfoTask()); taskMap.put(ApmTask.TASK_PROCESS_INFO, new ProcessInfoTask()); taskMap.put(ApmTask.TASK_BLOCK, new BlockTask()); taskMap.put(ApmTask.TASK_WATCHDOG, new WatchDogTask());}

接著,我們再看看 ActivityTask 類的實現,如下所示:

public class ActivityTask extends BaseTask { @Override protected IStorage getStorage() { return new ActivityStorage(); } @Override public String getTaskName() { return ApmTask.TASK_ACTIVITY; } @Override public void start() { super.start(); if (Manager.getInstance().getConfig().isEnabled(ApmTask.FLAG_COLLECT_ACTIVITY_INSTRUMENTATION) && !InstrumentationHooker.isHookSucceed()) {//hook失敗 if (DEBUG) { LogX.d(TAG, "ActivityTask", "canWork hook : hook失敗"); } mIsCanWork = false; } } @Override public boolean isCanWork() { return mIsCanWork; }}

可以看到,這里并沒有看到 save 方法,說明是在基類 BaseTask 類中,繼續看到 BaseTask 類的實現代碼:

/*** ArgusAPM任務基類** @author ArgusAPM Team*/public abstract class BaseTask implements ITask { ... @Override public boolean save(IInfo info) { if (DEBUG) { LogX.d(TAG, SUB_TAG, "save task :" + getTaskName()); } // 1 return info != null && mStorage != null && mStorage.save(info); } ...}

在注釋1處,繼續調用了 mStorage 的 save 方法,它是一個接口 IStorage,很顯然,這里的實現類是在 ActivityTask 的 getStorage() 方法中返回的 ActivityStorage 實例,它是一個 Activity 存儲類,專門負責處理 Activity 的信息。到此,監控應用冷熱啟動耗時與生命周期耗時的部分就分析完畢了。

下面,我們再看看如何使用 AspectJ 監控 OKHttp3 的每一次網絡請求。

2、監控 OKHttp3 的每一次網絡請求

首先,我們看到 OKHttp3 的切面文件,代碼如下所示:

/*** OKHTTP3 切面文件** @author ArgusAPM Team*/@Aspectpublic class OkHttp3Aspect { // 1、定義一個切入點,用于直接調用 OkHttpClient 的 build 方法。 @Pointcut("call(public okhttp3.OkHttpClient build())") public void build() { } // 2、使用環繞通知在 build 方法執行前添加一個 NetWokrInterceptor。 @Around("build()") public Object aroundBuild(ProceedingJoinPoint joinPoint) throws Throwable { Object target = joinPoint.getTarget(); if (target instanceof OkHttpClient.Builder && Client.isTaskRunning(ApmTask.TASK_NET)) { OkHttpClient.Builder builder = (OkHttpClient.Builder) target; builder.addInterceptor(new NetWorkInterceptor()); } return joinPoint.proceed(); }}

在注釋1、2處,在調用 OkHttpClient 的 build 方法之前添加了一個 NetWokrInterceptor。我們看看它的實現代碼,如下所示:

@Overridepublic Response intercept(Chain chain) throws IOException { // 1、獲取每一個 OkHttp 請求的開始時間 long startNs = System.currentTimeMillis(); mOkHttpData = new OkHttpData(); mOkHttpData.startTime = startNs; if (Env.DEBUG) { Log.d(TAG, "okhttp request 開始時間:" + mOkHttpData.startTime); } Request request = chain.request(); // 2、記錄當前請求的請求 url 和請求數據大小 recordRequest(request); Response response; try { response = chain.proceed(request); } catch (IOException e) { if (Env.DEBUG) { e.printStackTrace(); Log.e(TAG, "HTTP FAILED: " + e); } throw e; } // 3、記錄這次請求花費的時間 mOkHttpData.costTime = System.currentTimeMillis() - startNs; if (Env.DEBUG) { Log.d(TAG, "okhttp chain.proceed 耗時:" + mOkHttpData.costTime); } // 4、記錄當前請求返回的響應碼和響應數據大小 recordResponse(response); if (Env.DEBUG) { Log.d(TAG, "okhttp chain.proceed end."); } // 5、記錄 OkHttp 的請求數據 DataRecordUtils.recordUrlRequest(mOkHttpData); return response;}

首先,在注釋1處,獲取了每一個 OkHttp 請求的開始時間。接著,在注釋2處,通過 recordRequest 方法記錄了當前請求的請求 url 和請求數據大小。然后,注釋3處,記錄了這次 請求所花費的時間。

接下來,在注釋4處,通過 recordResponse 方法記錄了當前請求返回的響應碼和響應數據大小。最后,在注釋5處,調用了 DataRecordUtils 的 recordUrlRequest ?方法記錄了 mOkHttpData 中保存好的數據。我們繼續看到 recordUrlRequest 方法,代碼如下所示:

/** * recordUrlRequest * * @param okHttpData */public static void recordUrlRequest(OkHttpData okHttpData) { if (okHttpData == null || TextUtils.isEmpty(okHttpData.url)) { return; } QOKHttp.recordUrlRequest(okHttpData.url, okHttpData.code, okHttpData.requestSize, okHttpData.responseSize, okHttpData.startTime, okHttpData.costTime); if (Env.DEBUG) { Log.d(Env.TAG, "存儲okkHttp請求數據,結束。"); }}

可以看到,這里調用了 QOKHttp 的 recordUrlRequest 方法用于記錄網絡請求信息。我們再看到 QOKHttp 的 recordUrlRequest 方法,如下所示:

/** * 記錄一次網絡請求 * * @param url 請求url * @param code 狀態碼 * @param requestSize 發送的數據大小 * @param responseSize 接收的數據大小 * @param startTime 發起時間 * @param costTime 耗時 */public static void recordUrlRequest(String url, int code, long requestSize, long responseSize, long startTime, long costTime) { NetInfo netInfo = new NetInfo(); netInfo.setStartTime(startTime); netInfo.setURL(url); netInfo.setStatusCode(code); netInfo.setSendBytes(requestSize); netInfo.setRecordTime(System.currentTimeMillis()); netInfo.setReceivedBytes(responseSize); netInfo.setCostTime(costTime); netInfo.end();}

可以看到,這里 將網絡請求信息保存在了 NetInfo 中,并最終調用了 netInfo 的 end 方法,代碼如下所示:

/** * 為什存儲的操作要寫到這里呢? * 歷史原因 */public void end() { if (DEBUG) { LogX.d(TAG, SUB_TAG, "end :"); } this.isWifi = SystemUtils.isWifiConnected(); this.costTime = System.currentTimeMillis() - startTime; if (AnalyzeManager.getInstance().isDebugMode()) { AnalyzeManager.getInstance().getNetTask().parse(this); } ITask task = Manager.getInstance().getTaskManager().getTask(ApmTask.TASK_NET); if (task != null) { // 1 task.save(this); } else { if (DEBUG) { LogX.d(TAG, SUB_TAG, "task == null"); } }}

可以看到,這里 最終還是調用了 NetTask 實例的 save 方法保存網絡請求的信息。而 NetTask 肯定是使用了與之對應的 NetStorage 實例將信息保存在了 ContentProvider 中。至此,OkHttp3 這部分的分析就結束了。

對于使用 OkHttp3 的應用來說,上述的實現可以有效地獲取網絡請求的信息,但是如果應用沒有使用 OkHttp3 呢?這個時候,我們就只能去監控 HttpConnection 的每一次網絡請求。下面,我們就看看如何去實現它。

3、監控 HttpConnection 和 HttPClient 的每一次網絡請求

在 ArgusAPM 中,使用的是 TraceNetTrafficMonitor 這個切面類對 HttpConnection 的每一次網絡請求進行監控。關鍵代碼如下所示:

@Aspectpublic class TraceNetTrafficMonitor { // 1 @Pointcut("(!within(com.argusapm.android.aop.*) && ((!within(com.argusapm.android.**) && (!within(com.argusapm.android.core.job.net.i.*) && (!within(com.argusapm.android.core.job.net.impl.*) && (!within(com.qihoo360.mobilesafe.mms.transaction.MmsHttpClient) && !target(com.qihoo360.mobilesafe.mms.transaction.MmsHttpClient)))))))") public void baseCondition() { } // 2 @Pointcut("call(org.apache.http.HttpResponse org.apache.http.client.HttpClient.execute(org.apache.http.client.methods.HttpUriRequest)) && (target(httpClient) && (args(request) && baseCondition()))") public void httpClientExecuteOne(HttpClient httpClient, HttpUriRequest request) { } // 3 @Around("httpClientExecuteOne(httpClient, request)") public HttpResponse httpClientExecuteOneAdvice(HttpClient httpClient, HttpUriRequest request) throws IOException { return QHC.execute(httpClient, request); } // 排查一些處理異常的切面代碼 // 4 @Pointcut("call(java.net.URLConnection openConnection()) && (target(url) && baseCondition())") public void URLOpenConnectionOne(URL url) { } // 5 @Around("URLOpenConnectionOne(url)") public URLConnection URLOpenConnectionOneAdvice(URL url) throws IOException { return QURL.openConnection(url); } // 排查一些處理異常的切面代碼

}

TraceNetTrafficMonitor 里面的操作分為 兩類一類是用于切 HttpClient 的 execute 方法,即注釋1、2、3處所示的切面代碼;一類是用于切 HttpConnection 的 openConnection 方法,對應的切面代碼為注釋4、5處。我們首先分析 HttpClient 的情況,這里最終 調用了 QHC 的 execute 方法進行處理,如下所示:

public static HttpResponse execute(HttpClient client, HttpUriRequest request) throws IOException { return isTaskRunning() ? AopHttpClient.execute(client, request) : client.execute(request);}

這里又 繼續調用了 AopHttpClient 的 execute 方法,代碼如下所示:

public static HttpResponse execute(HttpClient httpClient, HttpUriRequest request) throws IOException { NetInfo data = new NetInfo(); // 1 HttpResponse response = httpClient.execute(handleRequest(request, data)); // 2 handleResponse(response, data); return response;}

首先,在注釋1處,調用了 handleRequest 處理請求數據,如下所示:

private static HttpUriRequest handleRequest(HttpUriRequest request, NetInfo data) { data.setURL(request.getURI().toString()); if (request instanceof HttpEntityEnclosingRequest) { HttpEntityEnclosingRequest entityRequest = (HttpEntityEnclosingRequest) request; if (entityRequest.getEntity() != null) { // 1、將請求實體使用 AopHttpRequestEntity 進行了封裝 entityRequest.setEntity(new AopHttpRequestEntity(entityRequest.getEntity(), data)); } return (HttpUriRequest) entityRequest; } return request;}

可以看到,在注釋1處,使用 AopHttpRequestEntity 對請求實體進行了封裝,這里的目的主要是為了 便于使用封裝實體中的 NetInfo 進行數據操作。接著,在注釋2處,將得到的響應信息進行了處理,這里的實現很簡單,就是 使用 NetInfo 這個實體類將響應信息保存在了 ContentProvider 中。至此,HttpClient 的處理部分我們就分析完畢了。

下面,我們接著分析下 HTTPConnection 的切面部分代碼,如下所示:

// 4@Pointcut("call(java.net.URLConnection openConnection()) && (target(url) && baseCondition())")public void URLOpenConnectionOne(URL url) {}// 5@Around("URLOpenConnectionOne(url)")public URLConnection URLOpenConnectionOneAdvice(URL url) throws IOException { return QURL.openConnection(url);}

可以看到,這里是 調用了 QURL 的 openConnection 方法進行處理。我們來看看它的實現代碼:

public static URLConnection openConnection(URL url) throws IOException { return isNetTaskRunning() ? AopURL.openConnection(url) : url.openConnection();}

這里 又調用了 AopURL 的 openConnection 方法,繼續 看看它的實現:

public static URLConnection openConnection(URL url) throws IOException { if (url == null) { return null; } return getAopConnection(url.openConnection());}private static URLConnection getAopConnection(URLConnection con) { if (con == null) { return null; } if (Env.DEBUG) { LogX.d(TAG, "AopURL", "getAopConnection in AopURL"); } // 1 if ((con instanceof HttpsURLConnection)) { return new AopHttpsURLConnection((HttpsURLConnection) con); } // 2 if ((con instanceof HttpURLConnection)) { return new AopHttpURLConnection((HttpURLConnection) con); } return con;}

最終,在注釋1處,會判斷如果是 https 請求,則會使用 AopHttpsURLConnection 封裝 con,如果是 http 請求,則使用 AopHttpURLConnection 進行封裝。AopHttpsURLConnection 的實現與它類似,僅僅是多加了 SSL 證書驗證的部分。所以這里我們就直接分析一下 AopHttpURLConnection 的實現,這里面的代碼非常多,就不貼出來了,但是,它的 核心的處理 可以簡述為如下 兩點

  • 1)、在回調 getHeaderFields()、getInputStream()、getLastModified() 等一系列方法時會調用 inspectAndInstrumentResponse 方法把響應大小和狀態碼保存在 NetInfo 中。

  • 2)、在回調 onInputstreamComplete()、onInputstreamError()等方法時,即請求完成或失敗時,此時會直接調用 myData 的 end 方法將網絡響應信息保存在 ContentProvider 中。

至此,ArgusAPM 的 AOP 實現部分就已經全部分析完畢了。

五、總結

最后,我們再來回顧一下本篇文章中我們所學到的知識,如下所示:

  • 1、編譯插樁技術的分類與應用場景。

    • 1)、APT

    • 2)、AOP。

  • 2、AspectJ 的優勢與局限性

  • 3、AspectJ 核心語法簡介

  • 4、AspectJX 實戰。

    • 1)、最簡單的 AspectJ 示例。

    • 2)、統計 Application 中所有方法的耗時。

    • 3)、對 App 中所有的方法進行 Systrace 函數插樁

  • 5、使用 AspectJ 打造自己的性能監控框架。

    • 1)、監控應用冷熱啟動耗時與生命周期耗時。

    • 2)、監控 OKHttp3 的每一次網絡請求。

    • 3)、監控 HttpConnection 和 HttpClient 的每一次網絡請求。

可以看到,AOP 技術的確很強大,使用 AspectJ 我們能做很多事情,但是,它也有一系列的缺點,比如切入點固定、正則表達式固有的缺陷導致的使用不靈活,此外,它還生成了比較多的包裝代碼。那么,有沒有更好地實現方式,既能夠在使用上更加地靈活,也能夠避免生成包裝代碼,以減少插樁所帶來的性能損耗呢?沒錯,就是 ASM,但是它 需要通過操作 JVM 字節碼的方式來進行代碼插樁,入手難度比較大,所以,下篇文章我們將會先深入學習 JVM 字節碼的知識,敬請期待~

? ? ? ? ? ? ? ? ? ? ? ? 喜歡 就關注吧,歡迎投稿!

總結

以上是生活随笔為你收集整理的aspectj 获取方法入参_深入探索编译插桩技术(二、AspectJ)的全部內容,希望文章能夠幫你解決所遇到的問題。

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

97夜夜澡人人爽人人免费 | 欧美一区二视频在线免费观看 | 日本中文字幕在线视频 | 一级久久久 | 在线观看视频国产一区 | 综合久久精品 | 在线免费亚洲 | 99久久夜色精品国产亚洲96 | 最近最新中文字幕 | 久久精品国产免费看久久精品 | 日韩免费福利 | 国产一线在线 | 久久午夜电影网 | 亚洲一级片 | 亚洲国产欧美一区二区三区丁香婷 | 在线观看日韩视频 | 狠狠狠色丁香婷婷综合久久88 | 99热这里只有精品8 久久综合毛片 | 激情婷婷 | 99视频在线播放 | 日韩手机在线 | 精品久久久久久国产 | 午夜精品久久久久久久99 | 久久激情久久 | 久久久精品国产一区二区电影四季 | 激情五月婷婷综合网 | 日韩videos高潮hd | 成人在线一区二区三区 | 中文国产成人精品久久一 | 欧美午夜一区二区福利视频 | 国产午夜三级一区二区三桃花影视 | 在线国产专区 | 麻豆91在线播放 | 操少妇视频 | 国内亚洲精品 | 波多野结衣一区二区 | 成人黄在线观看 | 亚洲欧美国产视频 | 日韩免费电影一区二区三区 | 美女av在线免费 | 激情av网址| 精品亚洲午夜久久久久91 | 在线成人一区二区 | 毛片激情永久免费 | 国产另类av | 欧美精品国产综合久久 | av免费在线看网站 | 国产黄在线看 | 亚洲伊人网在线观看 | 91大神精品视频在线观看 | 久久国内精品99久久6app | 黄色影院在线观看 | 日韩精品一区二区三区第95 | 日本精品一区二区三区在线观看 | 在线电影a | 一级性生活片 | 国产91精品久久久久 | 亚洲成人动漫在线观看 | 在线黄网站 | 中文字幕区 | 中文字幕人成不卡一区 | 91精品国产91久久久久久三级 | 美女视频黄是免费的 | 韩国av一区二区三区在线观看 | 午夜精品av在线 | 丰满少妇在线观看网站 | 天堂av观看 | 日韩黄色一级电影 | 丁香婷婷色综合亚洲电影 | 国产免费专区 | 91精品国产九九九久久久亚洲 | 狠狠综合| 一色av| 五月婷婷黄色网 | 国产精品99久久久久久人免费 | 在线视频成人 | 99久久精| 久久精品欧美日韩精品 | av在线电影网站 | 超碰97人人射妻 | 久久精品视频免费播放 | 91久久久久久久一区二区 | 天天综合网久久综合网 | 97在线视频观看 | 337p西西人体大胆瓣开下部 | 在线观看国产日韩欧美 | 亚洲精品高清视频 | 日本在线观看一区二区三区 | 国产又黄又爽无遮挡 | 国产69精品久久99的直播节目 | 操操日日 | 视频一区二区在线观看 | 精品亚洲va在线va天堂资源站 | 操操操综合 | 韩国视频一区二区三区 | 一区二区视频在线看 | 一区二区精品视频 | 天天操天天操一操 | 伊人久久在线观看 | 精品国产91亚洲一区二区三区www | 黄色毛片在线 | 激情五月色播五月 | 天天舔夜夜操 | www亚洲精品 | 欧美超碰在线 | 毛片播放网站 | 91大神dom调教在线观看 | 中文在线免费视频 | 黄色三几片| 99精品视频在线播放观看 | 国产v亚洲v | 91网免费看| mm1313亚洲精品国产 | 国产精品ⅴa有声小说 | av中文在线观看 | 国产成人精品亚洲日本在线观看 | 亚洲精品系列 | 色五婷婷 | 欧美激情综合色综合啪啪五月 | 日韩高清免费在线观看 | 蜜桃av综合网 | 四虎成人免费观看 | 色欲综合视频天天天 | 夜夜操夜夜干 | 欧美日韩高清在线一区 | 黄色av在 | 欧美国产日韩一区 | 玖玖精品在线 | 99久久精品国 | 久久成年人网站 | 麻豆视频国产 | 最新超碰 | 婷婷资源站 | 日韩久久精品一区二区三区 | 三级黄色大片在线观看 | 在线日韩精品视频 | 国产一级一级国产 | 成人一区二区在线 | 久久日本视频 | 在线小视频| www.天天射 | 日韩精品久久久久久久电影99爱 | 亚洲电影久久久 | 日韩视频精品在线 | 亚洲精品婷婷 | 黄色三级网站 | 国产四虎影院 | 亚洲精品1区2区3区 超碰成人网 | 狠狠的日 | 欧美另类交在线观看 | 色99网| 99成人免费视频 | 精品96久久久久久中文字幕无 | 国产做爰视频 | 国产成人久久77777精品 | 在线中文字幕一区二区 | 九九九视频精品 | 国产精品自拍在线 | 色干综合| 国产一区二区久久久 | 黄色tv视频 | 91超级碰| 精品国产亚洲日本 | 国产麻豆剧传媒免费观看 | 日韩免费看片 | 久久人人爽视频 | 九九免费精品视频在线观看 | 超碰公开在线 | 在线中文字幕观看 | av一级二级| 亚洲va欧美va人人爽春色影视 | 久久狠狠婷婷 | 国产精品一区在线观看你懂的 | 中文字幕五区 | 成人网看片 | 国产福利av在线 | 婷婷国产在线观看 | 夜夜夜夜爽 | 久久久久久久久久久黄色 | av福利第一导航 | 9在线观看免费高清完整 | 久久99国产精品免费网站 | 色婷婷六月天 | 日韩高清不卡在线 | 久久久久久久国产精品 | 国产精品福利视频 | 亚洲第一区在线播放 | 成人午夜免费剧场 | av午夜电影| 一区二区视频播放 | 亚洲永久精品国产 | 国产精品久久久一区二区 | 成人宗合网 | 日韩在线观看免费 | 久久久久国产精品www | 天天综合日日夜夜 | 91禁看片| 天天干天天操天天入 | 国产高清免费在线观看 | 久久综合五月 | 成人av高清在线 | 国产中文在线播放 | 国产精品久久久久久久久岛 | 中文永久免费观看 | 亚洲91精品在线观看 | 超碰在线中文字幕 | 黄www在线观看 | 在线精品视频免费播放 | 在线之家免费在线观看电影 | 国产亚洲精品女人久久久久久 | 成人久久18免费网站麻豆 | 中字幕视频在线永久在线观看免费 | 国产精品毛片一区二区 | 91精品色| 黄色毛片视频 | 在线观看亚洲精品视频 | 国产96在线观看 | 久久精品成人欧美大片古装 | 天天色欧美| 99热在线免费观看 | 九九视频免费在线观看 | 日本午夜在线观看 | 国产精品理论片 | 日韩精品一区二区三区外面 | 国产日韩欧美在线一区 | 成人欧美一区二区三区黑人麻豆 | 国产女v资源在线观看 | 99视频精品全部免费 在线 | 波多野结衣在线播放视频 | 美女在线国产 | 99精品视频免费看 | 丰满少妇一级片 | 国产视频在线观看免费 | 97国产在线 | 在线性视频日韩欧美 | 精品国产观看 | 国产成人精品一二三区 | 免费成人黄色 | 亚洲精品美女久久 | 日本中文字幕一二区观 | 在线观看第一页 | 一区二区中文字幕在线观看 | 日本精品久久久久中文字幕5 | 久久一区二 | 国产精品久久久久久久妇 | 亚洲国产中文字幕在线 | 日本性生活一级片 | 91在线视频观看免费 | 天天曰| 天天艹天天爽 | 久久精品亚洲一区二区三区观看模式 | 日韩精品资源 | 国产无套精品久久久久久 | 中文字幕 第二区 | 中文字幕在线视频精品 | 97碰在线 | 园产精品久久久久久久7电影 | 黄色av在 | 日韩激情免费视频 | 99在线免费观看 | 久久人人97超碰国产公开结果 | 国产一区二区精品在线 | 亚洲人成人99网站 | 国产日产精品一区二区三区四区 | 欧美黄色软件 | 一区中文字幕电影 | 国产精品99久久久久久小说 | 日韩a在线| 精品国产三级 | 婷婷丁香七月 | 午夜视频99| 亚洲激情一区二区三区 | 亚洲在线网址 | 亚洲精品视频中文字幕 | 国产精品久久久久一区二区三区共 | 精品国产一区二区三区在线观看 | 亚洲三级在线免费观看 | 婷婷天天色 | 97干com| 欧美日韩视频免费看 | 国产视频日本 | av电影中文 | 精品国产福利在线 | 精品久久久久久久久久国产 | 久久婷婷一区二区三区 | 亚洲精品视频网址 | 片黄色毛片黄色毛片 | 国产精品久久久区三区天天噜 | 亚洲观看黄色网 | 激情网婷婷| 99热这里有精品 | 日韩性片 | 中文字幕二区在线观看 | 狠狠操导航| 国产黄色免费电影 | 亚洲日韩中文字幕 | 免费看国产一级片 | 精品久久久免费 | 久久99国产精品二区护士 | 亚洲a成人v | 国产日本亚洲高清 | 免费视频久久 | 热久久免费视频精品 | 日韩精品一区二区三区水蜜桃 | 久久99网站 | 亚洲天天摸日日摸天天欢 | 国产精品国产三级国产不产一地 | 中文字幕网址 | 国产精品久久久久av福利动漫 | 99久久精品国产一区二区成人 | 日韩精品视频免费在线观看 | 国产高清视频免费最新在线 | 91视频在线观看下载 | 欧美成人在线免费观看 | 久久99视频免费观看 | 国产精品99久久久久久武松影视 | 国产色在线| 亚洲午夜电影网 | 久久国产片 | 日批网站在线观看 | 中文字幕美女免费在线 | 欧美性色综合网站 | 超碰夜夜 | 91精品国产91| 天天操天天玩 | 欧美在线久久 | aav在线| 色成人亚洲网 | 欧美综合在线视频 | 丁香六月五月婷婷 | 操操操综合| av一级片在线观看 | 欧美电影在线观看 | 日韩欧美在线高清 | 久久久久久黄 | 精品字幕在线 | 欧美精品免费在线观看 | 91桃色免费视频 | 91片黄在线观看 | 精品字幕| av福利在线免费观看 | 99免费看片 | 色综合小说 | 国产精品原创 | 成人国产精品一区二区 | 国外调教视频网站 | 久久久国产精品成人免费 | 青草视频在线看 | 91麻豆国产| 91经典在线| av一区在线播放 | 亚洲aⅴ在线| 99免费在线视频观看 | 国产精品99久久久久久宅男 | 99久久99热这里只有精品 | 一级一片免费视频 | 天天操天天干天天 | 黄色精品久久 | 中文字幕 国产视频 | 日日干激情五月 | 丁香婷婷综合网 | 亚洲视频 在线观看 | 一级黄色片在线观看 | 久久久久久久久精 | 国产免费亚洲高清 | 伊甸园av在线 | 欧美日韩国产高清视频 | 久热色超碰 | 免费视频国产 | 国产精品婷婷午夜在线观看 | 人人澡人人爽欧一区 | 亚洲日本一区二区在线 | 日韩午夜高清 | 高清一区二区三区av | 国产精品99久久久久久久久久久久 | 精品一区二区在线播放 | av在线免费在线观看 | 日韩电影黄色 | 一级片免费在线 | 日韩网站中文字幕 | 黄色三级免费观看 | 亚洲欧美日韩精品久久奇米一区 | 2021国产在线视频 | 欧美在线aaa | 91在线资源| av在线播放免费 | 成人在线观看免费视频 | 欧美日韩国产一区二区三区在线观看 | 久久成人一区二区 | 国产婷婷一区二区 | 深爱激情综合 | 五月婷婷在线观看视频 | 国产亚洲精品久久久网站好莱 | 天堂在线视频免费观看 | 国产电影黄色av | 黄色精品久久 | 日日操天天操狠狠操 | 国色天香第二季 | 区一区二区三区中文字幕 | 日韩在线字幕 | 久久天堂影院 | 色视频网站在线观看一=区 a视频免费在线观看 | 国产视频一区二区在线观看 | 亚洲精品国久久99热 | 成人在线视频观看 | 三级视频国产 | 五月婷婷综合久久 | 日韩理论片在线 | 国产亚洲视频在线免费观看 | 亚洲视屏 | 欧美日韩性视频在线 | 97免费在线观看视频 | 国产中文字幕三区 | a在线观看视频 | 婷婷六月色 | 色婷久久 | 韩国一区二区三区视频 | 亚洲日本在线一区 | 亚洲日本国产精品 | 久久久www成人免费精品 | 日产中文字幕 | 高清有码中文字幕 | 69视频在线 | 国产乱对白刺激视频不卡 | 久久艹国产 | 日韩91精品 | av电影在线不卡 | 欧美色精品天天在线观看视频 | 五月综合色 | av成人免费在线看 | 亚洲成av人片一区二区梦乃 | 久久久国产日韩 | 最新色站| 久久这里只有精品23 | 国产精品夜夜夜一区二区三区尤 | 国产成人精品999在线观看 | 国产精品爽爽久久久久久蜜臀 | 免费看黄色小说的网站 | 手机av电影在线 | 国产一级片网站 | 日韩电影在线一区二区 | 18av在线视频 | 五月天久久久久 | 日韩在线观看免费 | 日韩免费看视频 | 久久精品国产久精国产 | 日韩在线观看视频中文字幕 | 国产一区二区午夜 | 国产精品伦一区二区三区视频 | 国产剧情一区二区 | 中文字幕av在线电影 | 国产亚洲精品久久久久久电影 | 射射色 | 国产精品手机视频 | 久久草草热国产精品直播 | 一级电影免费在线观看 | 国产乱对白刺激视频在线观看女王 | 日韩欧美视频在线播放 | 91丨九色丨丝袜 | 欧美电影黄色 | 超碰在线人人97 | 久久精品一区二区三区视频 | 国产精品久久久久影院 | 久久国产经典视频 | 深爱开心激情网 | 国产综合在线视频 | av导航福利 | 最近更新好看的中文字幕 | 91爱爱电影| 免费三级av | 欧美一区在线看 | 国产日韩视频在线观看 | 欧美精品国产综合久久 | 日韩精品在线看 | 中文字幕一区二区三区久久 | 五月天久久久 | 久久激情片 | 黄色毛片视频 | 五月婷婷激情综合 | 国产一区久久 | 欧美aa级| 天天色天天操综合网 | 久久国产一区二区三区 | 免费观看一级特黄欧美大片 | 日韩精品免费一线在线观看 | 日韩大片在线 | 日韩av三区 | 国产成人精品一区二区三区在线观看 | 一级一片免费看 | 亚洲电影av在线 | 看毛片网站 | 久久久久一区 | 一区二区三区av在线 | av888av.com| 狠狠躁夜夜躁人人爽视频 | 人人cao | 精品国产诱惑 | 日韩高清三区 | 欧美在线不卡一区 | 最新国产视频 | 日韩精品一区二区在线观看视频 | 色噜噜在线观看视频 | 色婷在线 | av色综合 | 亚洲视频专区在线 | avlulu久久精品 | 亚洲资源一区 | 色射爱 | 美女视频黄在线 | 亚洲国产精品传媒在线观看 | 日本精品一区二区在线观看 | 精品久久中文 | 日日狠狠 | 午夜丁香视频在线观看 | 中文字幕91 | 中文字幕乱码电影 | 精品国产亚洲在线 | 亚洲精品动漫成人3d无尽在线 | 96av视频| 久久久久久久久久久久亚洲 | 久久五月婷婷丁香 | 成人国产电影在线观看 | 亚洲精品小视频 | 日韩成人中文字幕 | 9草在线 | 99精品国产一区二区三区麻豆 | 天天曰 | 韩日色视频| 一性一交视频 | 婷婷草| 精品国产视频在线观看 | 成人av高清 | 国产在线观看一区 | 日韩精品首页 | 天天摸夜夜操 | 久久视频一区 | 玖玖在线资源 | 91桃色国产在线播放 | 精品一二三区 | 蜜臀久久99精品久久久久久网站 | 99免费在线视频观看 | 久久久免费精品 | 在线观看日韩中文字幕 | 91av视频免费在线观看 | 成人四虎影院 | 午夜视频在线观看网站 | 久草免费电影 | 在线看91| 日韩欧美综合精品 | 色成人亚洲| 成人在线视频你懂的 | 高清av中文在线字幕观看1 | 99精品国产福利在线观看免费 | 五月天激情婷婷 | 91超级碰碰 | 免费观看完整版无人区 | 五月开心六月伊人色婷婷 | 免费看网站在线 | 福利一区在线 | www.黄色片.com | 国产精品欧美一区二区 | 久久久久免费精品国产 | 五月婷婷丁香六月 | 特级西西444www大胆高清无视频 | 婷婷成人亚洲综合国产xv88 | 亚洲人片在线观看 | 国产精品麻豆果冻传媒在线播放 | 国产一区二区播放 | 午夜久久 | 国产美女精品视频 | av中文字幕不卡 | 欧美一区在线观看视频 | 天天操 夜夜操 | 亚洲成人av片在线观看 | 国产精品 中文字幕 亚洲 欧美 | 91视频传媒| 91av在线免费视频 | 7777精品伊人久久久大香线蕉 | 天天躁日日躁狠狠 | av中文字幕在线看 | 大胆欧美gogo免费视频一二区 | 亚洲v精品 | 久草干 | 永久免费精品视频 | 激情综合五月天 | 欧美精品久久久久久久久久丰满 | 免费视频网 | 中文有码在线视频 | 成人黄色电影免费观看 | 国内丰满少妇猛烈精品播 | 久草在线观看资源 | 日本精品视频免费 | 亚洲国产精品久久久久 | 911免费视频| 亚洲精品国产综合99久久夜夜嗨 | 天天射天 | 国产精品成人品 | 久久欧美视频 | 欧美国产高清 | 色a资源在线 | 国产精品精品久久久久久 | 黄色软件网站在线观看 | 色.www| av黄色免费看 | 成人黄色电影在线播放 | 首页国产精品 | 中文字幕免费一区二区 | 国产黄色片一级 | 中文字幕亚洲综合久久五月天色无吗'' | 999成人精品 | 97免费中文视频在线观看 | 91系列在线 | 99九九免费视频 | 九九九九精品 | 国产成人精品av久久 | 91电影福利 | 久草视频在线播放 | 国产五月婷 | 亚洲国产精品推荐 | 午夜精品电影一区二区在线 | 国产精品久久亚洲 | 夜夜嗨av色一区二区不卡 | 成人免费91 | 黄色日本免费 | 久久精品精品电影网 | 在线播放国产一区二区三区 | 夜夜操网| 国产网站在线免费观看 | 激情动态| 国产亚洲aⅴaaaaaa毛片 | 国产精品福利在线播放 | 国产一区 在线播放 | 国产视频1| 中文字幕日韩精品有码视频 | 亚洲精选视频在线 | 国内久久视频 | 免费看国产精品 | 精品久久影院 | 久九视频| 国产一级精品视频 | 西西www444| 亚洲免费在线观看视频 | 天天操人人要 | 黄色日批网站 | 成 人 黄 色视频免费播放 | 成人福利av | 丁香六月婷婷开心 | 人人插人人爱 | 国产破处精品 | 人人搞人人爽 | 日韩免费视频在线观看 | 一区二区三区福利 | 国产精品一区二区三区99 | 久久久久免费精品视频 | 日本在线观看一区二区三区 | 免费观看www7722午夜电影 | 日韩乱码中文字幕 | 日韩久久精品一区二区三区 | 91av在| 99久久精品国产观看 | 天天撸夜夜操 | 国内精品久久久久国产 | 五月婷婷六月丁香在线观看 | 国产亚洲精品久久19p | 91精品视频免费看 | 在线精品播放 | 97激情影院 | 国产精品久久久久久久久久了 | 欧美va电影 | 亚洲最快最全在线视频 | 久久国产a | 亚洲精品国产欧美在线观看 | 国产一级性生活视频 | 国产精品毛片一区二区在线看 | 欧美成人h版 | 久久艹人人 | 五月天视频网 | 久久久久久久久久久久亚洲 | 国产精品一区二区三区在线播放 | 91精品在线观看视频 | 国产xxxx | 视频高清| 国产精品你懂的在线观看 | 国产精品久久久久久欧美 | 国产精品久久久久久久久久久免费 | 久久精品91久久久久久再现 | 在线观看91网站 | www.干| 91热爆视频 | 中文字幕久久精品亚洲乱码 | 亚洲成aⅴ人片久久青草影院 | 国产黄在线看 | 亚洲人成在线观看 | 成人午夜精品 | 九九九国产 | 日韩网站一区 | 久久久久麻豆 | 亚洲综合在线一区二区三区 | 久久久久久久久久久久影院 | av丝袜在线 | 国产精品色视频 | 91chinesexxx| 麻豆成人精品 | 日日操网站 | 免费日韩 精品中文字幕视频在线 | 天天视频色版 | 精品国产亚洲在线 | 欧美激情在线看 | 久久夜夜操 | 亚洲 欧美 另类人妖 | 伊人天天综合 | 天天操天天射天天插 | 免费91麻豆精品国产自产在线观看 | 国产精品久久嫩一区二区免费 | 欧美国产日韩一区二区 | 99精品视频一区 | 国产中文字幕第一页 | 国产小视频国产精品 | 国产欧美最新羞羞视频在线观看 | a√天堂中文在线 | 天天搞天天干 | 久久首页| 国产日韩欧美在线观看视频 | 91在线播放国产 | 一区二区三区 亚洲 | 人人狠| 综合色综合色 | 日韩欧美视频 | 在线成人短视频 | 国内视频1区 | 91大片网站 | 五月天婷婷在线视频 | 最近av在线 | 亚洲免费国产 | 久久综合久色欧美综合狠狠 | 亚州精品视频 | 亚洲精品中文字幕在线观看 | 久草精品网 | 午夜精品区| 日本视频精品 | 精品久久久久久国产91 | 久久久久黄 | 国产福利午夜 | 午夜在线免费观看 | 人人射人人爽 | 婷婷六月激情 | 欧美污网站 | 国产精品久久久久久久免费 | 97成人在线 | av 在线观看 | 成人av av在线 | 91精品黄色 | 免费av一级电影 | 一区二区网 | 在线国产日韩 | 亚洲一区视频免费观看 | 久久婷亚洲五月一区天天躁 | 国外成人在线视频网站 | 成人av资源网 | 国产精品成人久久久久久久 | 国产中文字幕三区 | 亚洲黄色在线播放 | 色综合久久久 | 国产亚洲精品女人久久久久久 | 久久网站av| 综合久久综合久久 | 欧美孕交vivoestv另类 | 欧美亚洲国产一卡 | 久久只精品99品免费久23小说 | 日韩在线精品一区 | 国产精品美女免费视频 | 四虎永久精品在线 | 最新中文字幕 | 精品中文字幕在线 | 亚洲美女精品区人人人人 | 999久久久 | 亚洲天天做 | 中文字幕在线日本 | 黄色影院在线观看 | 人人爱人人做人人爽 | 国产精品美女免费视频 | 欧美一区二区视频97 | 成人影视片| 91在线视频免费91 | 色婷婷99| 丁香六月欧美 | 91一区一区三区 | 九九视频免费观看视频精品 | av在线一 | www.大网伊人 | 婷婷丁香花五月天 | 3d黄动漫免费看 | 色香蕉在线视频 | 97精品视频在线 | 日韩精品资源 | 五月天网页 | 国产偷国产偷亚洲清高 | 青青河边草免费观看完整版高清 | 日韩视频免费播放 | 色综合久久中文字幕综合网 | www激情网 | 久久精品99久久久久久 | 美女黄色网在线播放 | 婷婷丁香色综合狠狠色 | 久草网免费 | 婷婷5月色 | 色婷婷激情网 | 99久久久久国产精品免费 | 国产精品99久久久精品 | 在线观看国产日韩 | 夜夜爱av| 开心激情久久 | 欧美精品一区在线发布 | 久草精品视频在线播放 | 精品国模一区二区 | 狠狠狠色丁香综合久久天下网 | 久久久久亚洲精品成人网小说 | 中文字幕在线视频网站 | 精品国产乱码久久久久久天美 | 五月激情五月激情 | 久久手机免费观看 | 日日夜夜操操操操 | 久久国产精品第一页 | 国产无吗一区二区三区在线欢 | 男女激情麻豆 | 六月婷色| 国产精品亚洲片夜色在线 | 久久久久久久久久久网 | 丁香在线视频 | 热久久国产精品 | av在线不卡观看 | 国内精品久久久久久久久久 | 日本3级在线观看 | 亚洲综合在线五月天 | 最近最新mv字幕免费观看 | 成人va视频 | 国产成人精品av | 国产精品入口66mio女同 | 国产综合在线观看视频 | 337p日本欧洲亚洲大胆裸体艺术 | 日韩av电影国产 | 国产精品视频专区 | 国产精品九九视频 | 国产高清在线观看av | 在线观看中文字幕第一页 | 在线精品观看 | 午夜精品一区二区三区四区 | 蜜臀久久99精品久久久无需会员 | 在线视频欧美精品 | 99国产精品免费网站 | 成年人电影免费看 | 国产在线观看中文字幕 | 99视频免费看 | 久久免费电影网 | 欧美综合国产 | 夜夜躁日日躁 | 国产激情免费 | 欧美日韩精品免费观看 | 一级成人免费 | 国产美女视频黄a视频免费 久久综合九色欧美综合狠狠 | 久草网视频在线观看 | 九九免费在线观看视频 | 免费看国产视频 | 久草久草久草久草 | 亚洲精品黄色在线观看 | 国产精品久久伊人 | 天天添夜夜操 | 精品视频国产一区 | 黄色特级毛片 | 久久理论片 | av大全免费在线观看 | 天天操狠狠操 | 亚洲成年人av | 亚洲婷婷在线 | 久草久草在线观看 | 337p日本欧洲亚洲大胆裸体艺术 | av资源免费观看 | 黄色免费网站下载 | 亚洲午夜精品久久久久久久久 | 狠狠色香婷婷久久亚洲精品 | 一区二区成人国产精品 | 一级成人网 | 国产特级毛片aaaaaaa高清 | 在线视频区 | av在线免费观看不卡 | 国产福利91精品一区 | 日本三级人妇 | 欧美久草视频 | 天天艹天天干天天 | 国产精品久久久久影院 | 国产成人精品一区二区三区在线观看 | 国产精品99久久99久久久二8 | 国产精品美女久久久 | 亚洲乱亚洲乱亚洲 | 天天躁日日躁狠狠 | 99草视频 | av一区二区三区在线 | 99在线播放| 欧美亚洲精品在线观看 | 日韩一区二区免费播放 | 欧美极品xxx | av7777777| 五月婷婷开心中文字幕 | 免费三级影片 | 99av国产精品欲麻豆 | 一区二区三区在线免费观看视频 | 成人中文字幕在线 | 香蕉网在线 | 最新av网站在线观看 | 最新av网址在线 | 国产黄色美女 | 久久天天操 | 4hu视频| 国产理论一区二区三区 | 91看片淫黄大片一级在线观看 | 91精品婷婷国产综合久久蝌蚪 | 九九视频热 | 欧美日韩久久不卡 | 一区二区三区播放 | 精品久久久久国产 | 国产亚洲精品久久久久久无几年桃 | 91视频高清免费 | 99精品99| 天天操狠狠干 | 亚洲高清av在线 | 中文字幕一区二区三 | 精品久久久久久久久亚洲 | 美女免费电影 | 国产一级大片在线观看 | 欧美精品黑人性xxxx | 中文字幕av一区二区三区四区 | 麻豆精品视频在线 | 狠狠色噜噜狠狠狠狠2022 | 天天综合成人网 | 国产一区福利 | 91最新地址永久入口 | 黄色a一级视频 | 国产成人一区二区三区在线观看 | 国产96在线| 久久免费a | 日韩电影精品一区 | 在线免费观看黄色av | 久久亚洲热| 免费福利小视频 | 精品国产一区二区三区久久影院 | 国产成人av电影在线观看 | 亚洲三级在线免费观看 | 日韩欧美综合在线视频 | 精品国产一区二区三区四区在线观看 | 免费成人av在线看 | 亚洲综合国产精品 | 国产成年人av | 久久丝袜视频 | 91精品国产高清自在线观看 | 一区二区三区国产欧美 | 最新高清无码专区 | 黄色一及电影 | 91综合色 | 国产精品第54页 | 日韩理论在线视频 | 狠狠搞,com | 婷婷丁香六月 | 国产成人亚洲在线电影 | 久久网站最新地址 | 日本三级吹潮在线 | 欧美午夜精品久久久久久浪潮 | 久久精品日韩 | 国产黄色av | 国产精品一区二区三区在线免费观看 | www.五月婷 | 国产视频久久久久 | 国产资源精品在线观看 | 国产三级午夜理伦三级 | www日韩在线 | 成人av片免费看 | 中文字幕中文字幕 | 中文av网 | 亚洲精品九九 | 17婷婷久久www | 人人草人| 国产高清在线观看 | 天天操天天色天天射 | 91在线资源 | 天天操操操操操操 | 国产成人在线免费观看 | 精品免费视频 | 91干干干 | 日本3级在线观看 | 亚洲欧美日韩精品久久久 | 国产精品毛片久久久久久久久久99999999 | 四虎在线观看精品视频 | 黄色福利网站 | 日韩精品一区二区久久 | 亚洲精品激情 | 午夜国产福利在线观看 | 在线观看激情av | 天天天综合网 | 尤物97国产精品久久精品国产 |