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

歡迎訪問 生活随笔!

生活随笔

當前位置: 首頁 > 运维知识 > Android >内容正文

Android

【我的Android进阶之旅】Android自定义Lint实践

發布時間:2023/12/9 Android 60 豆豆
生活随笔 收集整理的這篇文章主要介紹了 【我的Android进阶之旅】Android自定义Lint实践 小編覺得挺不錯的,現在分享給大家,幫大家做個參考.

背景

2017年8月份的時候,我在公司開始推廣Lint、FindBugs等靜態代碼檢測工具。然后發現系統自帶的Lint檢測的Issue不滿足我們團隊內部的特定需求,因此去自定義了部分Lint規則。這個檢測運行了大半年,運行良好,團隊的代碼規范也有了大幅度提升。這個是基于當時Gradle2.x系列寫出來的自定義Lint實踐總結,過去大半年了,現在將它搬到CSDN博客分享給大家一起學習學習。如果要在Gradle3.x系列使用該自定義規定的話,部分代碼都得修改成最新的語法,因此此篇博客的內容請使用Gralde2.x系列編譯項目中可以加入,去定義你自己的Lint規則吧。

當時已經實現的自定義規則大概有:

我這里只介紹如何去實現你自己的Lint規則,具體源代碼的話不方便貼出了,所以不會去公布源代碼。

一、Lint介紹

android lint是一個靜態代碼分析工具,通過lint工具,你可以不用邊運行邊調試,或者通過單元測試進行代碼檢查,可以檢測代碼中不規范、不和要求的問題,解決一些潛在的bug。lint工具可以在命令行上使用,也可以在adt中使用。

比如當想檢查在manifest.xml中是否有activity,activity中是否包含了launcher activity。如果沒有進行錯誤的警告。

通過lint的這種手段,可以對代碼進行規范的控制,畢竟一個團隊每個人的風格不同,但是要注意的當然是代碼的質量,所以lint可以進行代碼的規范和質量控制。

在Android studio還沒出來時,lint和Eclipse并不能很好的結合在一起,只能作為一個獨立的工具,通過命令行去執行lint檢查。

在android studio出現之后,不再建議單獨使用lint命令,而是結合gradle進行操作,命令為* ./gradlew lint *進行執行

lint工具通過一下六個方面去檢查代碼中的問題correctness, security, performance, usability, accessibility, and internationalization。檢查的范圍包括java文件,xml文件,class文件。

lint工具在sdk16版本之后就帶有了,所以在sdk目錄/tools/可以找到lint工具?,F在建議與gradle一起使用,使用./gradlew lint進行

參考官方文檔介紹

二、使用Lint的方法

關于lint的一些命令,可以參考官網,這里簡單介紹一些。

  • lint path(項目目錄) ——進行項目的lint檢查
  • lint –disable id(MissingTranslation,UnusedIds,Usability:Icons)path ——id是lint issue(問題)的標志,檢查項目,不包括指定的issue
  • lint –check id path ——利用指定的issue進行項目檢查
  • lint –list ——列出所有的issue
  • lint –show id ——介紹指定的issue
  • lint –help ——查看幫助

2.1 使用android studio自帶的lint工具

點擊Analyze的Inspect Code選項,即可開啟lint檢查,在Inspection窗口中可以看到lint檢查的結果,lint查詢的錯誤類型包括:

  • Missing Translation and Unused Translation【缺少翻譯或者沒有】
  • Layout Peformance problems (all the issues the old layoutopt tool used to find, and more)【布局展示問題】
  • Unused resources【沒有使用的資源】
  • Inconsistent array sizes (when arrays are defined in multiple configurations)【不一致的數組大小】
  • Accessibility and internationalization problems (hardcoded strings, missing contentDescription, etc)【可訪問性和國際化問題,包括硬鏈接的字符串,缺少contentDescription,等等】
  • Icon problems (like missing densities, duplicate icons, wrong sizes, etc)【圖片問題,丟失密度,重復圖片,錯誤尺寸等】
  • Usability problems (like not specifying an input type on a text field)【使用規范,比如沒有在一個文本上指定輸入的類型】
  • Manifest errors【Manifest.xml中的錯誤】
  • and so on

android自帶的lint規則的更改可以在Setting的Edit選項下選擇Inspections(File > Settings > Project Settings),對已有的lint規則進行自定義選擇。
參考官方文檔

2.2 使用lint.xml定義檢查規則

可以通過lint.xml來自定義檢查規則,這里的自定義是指定義系統原有的操作,所以和第一個步驟的結果是一樣的,只是可以更方便的配置。

lint.xml生效的位置是要放在項目的根目錄下面,lint.xml的示例如下:

<?xml version="1.0" encoding="UTF-8"?> <lint><!-- 忽略指定的檢查 --><issue id="IconMissingDensityFolder" severity="ignore" /><!-- 忽略指定文件的指定檢查 --><issue id="ObsoleteLayoutParam"><ignore path="res/layout/activation.xml" /><ignore path="res/layout-xlarge/activation.xml" /></issue><!-- 更改檢查問題歸屬的嚴重性 --><issue id="HardcodedText" severity="error" /> </lint>

參考官方文檔

三、自定義lint檢查規則結構介紹

3.1 概述

Android Lint是Google提供給Android開發者的靜態代碼檢查工具。使用Lint對Android工程代碼進行掃描和檢查,可以發現代碼潛在的問題,提醒程序員及早修正。

為保證代碼質量,在開發流程中加入了代碼檢查,如果代碼檢測到問題,則無法合并到正式分支中,這些檢查中就包括Lint。

3.2 為什么需要自定義

我們在實際使用Lint中遇到了以下問題:

原生Lint無法滿足我們團隊特有的需求,例如:編碼規范。
原生Lint存在一些檢測缺陷或者缺少一些我們認為有必要的檢測。
基于上面的考慮,我們開始調研并開發自定義Lint。

3.3 如何加入已有的lint規則

Lint規則是基于Java寫的,是在AST抽象語法樹上去進行一個解析。所以在寫Lint規則的時候,要學習一下AST抽象語法樹。才知道如何去尋找一個類方法和其參數等。以下有兩種方法:

  • 所以自定義Lint規則應該是一個寫好的jar包,jar包生效的位置是在~/.android/lint目錄,這個是對于Mac和Linux來說的,對于Windows來說就是在C:/Users/Administrator/.android/lint下,放到這個目錄下,Lint工具會自動加載這個jar包作為lint的自定義檢查規則。

  • 放到lint目錄下著實是一件比較麻煩的事情,即使可以用腳本來代替,但是仍然不是一個特別方便的方法。也是由于當android項目直接依賴于lint.jar包時不能起作用,而無法進行直接依賴。
    而aar很好的解決了這個問題,aar能夠將項目中的資源、class文件、jar文件等都包含,所以通過將lint.jar放入lintaar中,再由項目依賴于lintaar,這時候就可以達到自定義lint檢查的目的。

  • 下面就是如何使自定義lint生效的代碼示例,使用第二個方法(第二個方法就包括了第一個方法):

    主要包括了三個Module一個是XTCLintrRules,一個是XTCLintAAR,一個是XTCLintPlugin,還有一個是測試的app項目。

    • XTCLintrRules ,主要是用來編寫自定義Lint規則,編譯后生成 lint.jar
    • XTCLintAAR,主要是將XTCLintrRules生成的lint.jar包大包成aar,方便引用
    • XTCLintPlugin,主要是后面統一管理lint.xml和lintOptions,自動添加aar,后面再講解。

    3.3.1 XTCLintrRules的 gradle配置

    在XTCLintrRules中,主要是編寫lint規則,他是一個Java工程。

    它的gradle如下:

    //java項目,該項目編譯之后生成 XTCLintRules.jarapply plugin: 'java'dependencies {compile fileTree(dir: 'libs', include: ['*.jar'])// 依賴于lint的規則的apicompile 'com.android.tools.lint:lint-api:25.0.0'compile 'com.android.tools.lint:lint-checks:25.0.0'testCompile 'com.android.tools.lint:lint-tests:24.5.0' }/*** Lint-Registry是透露給lint工具的注冊類的方法,* 也就是PermissionIssueRegistry是lint工具的入口,同時也通過這個方法進行打jar包*/ jar{manifest{attributes('Lint-Registry': 'com.xtc.lint.rules.XTCIssueRegister')} }// 創建了一個叫“lintJarOutput”的Gradle configuration, // 用于輸出我們生成的jar包。在生成aar的模塊 "XTCLintAAR" 的build.gradle中會引用此configuration。 configurations {lintJarOutput }// 指定定義方法lintJarOutput的作用,此處是獲得調用jar方法后的生成的jar包 dependencies {lintJarOutput files(jar) }defaultTasks 'assemble'//指定編譯使用JDK1.8 //sourceCompatibility = JavaVersion.VERSION_1_8 //targetCompatibility = JavaVersion.VERSION_1_8//指定編譯的編碼 tasks.withType(JavaCompile){options.encoding = "UTF-8" }
    • lint-api: 官方給出的API,API并不是最終版,官方提醒隨時有可能會更改API接口。
    • lint-checks:已有的檢查。

    3.3.2 XTCLintAAR的 gradle配置

    apply plugin: 'com.android.library'apply from: 'maven_upload.gradle' apply from: '../jenkins.gradle'android {compileSdkVersion 25buildToolsVersion "25.0.3"defaultConfig {minSdkVersion 9targetSdkVersion 25versionCode 1versionName "1.0"}buildTypes {debug {buildConfigField 'String', 'JenkinsName', "\"" + jenkinsName + "\""buildConfigField 'String', 'JenkinsRevision', "\"" + jenkinsRevision + "\""buildConfigField 'String', 'GitSHA', "\"" + gitSHA + "\""buildConfigField 'String', 'GitBranch', "\"" + gitBranch + "\""buildConfigField 'String', 'GitTag', "\"" + gitTag + "\""minifyEnabled falseproguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-rules.pro'}release {buildConfigField 'String', 'JenkinsName', "\"" + jenkinsName + "\""buildConfigField 'String', 'JenkinsRevision', "\"" + jenkinsRevision + "\""buildConfigField 'String', 'GitSHA', "\"" + gitSHA + "\""buildConfigField 'String', 'GitBranch', "\"" + gitBranch + "\""buildConfigField 'String', 'GitTag', "\"" + gitTag + "\""minifyEnabled falseproguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-rules.pro'}} }dependencies {compile fileTree(dir: 'libs', include: ['*.jar']) }/*** 定義方法lintJarImport,引入 XTCLintRules.jar** rules for including "lint.jar" in aar*/ configurations {lintJarImport }// 鏈接到lintJar中的lintJarOutput方法,調用jar方法,并獲得jar包 dependencies {//其引用了模塊 “:XTCLintRules”的 Gradle configuration “lintJarOutput”。lintJarImport project(path: ':XTCLintrRules', configuration: "lintJarOutput") }// 將得到的JAR包復制到目錄build/intermediates/lint/下,并且重命名為 lint.jar task copyLintJar(type: Copy) {from(configurations.lintJarImport) {rename {String fileName ->'lint.jar'}}into 'build/intermediates/lint/' }// 當項目build到compileLint這一步時執行copyLintJar方法 project.afterEvaluate {def compileLintTask = project.tasks.find { it.name == 'compileLint' }//對內置的Gradle task “compileLint”做了修改,讓其依賴于我們定義的一個task “copyLintJar”。compileLintTask.dependsOn(copyLintJar) }

    該Module的作用就是:

  • 鏈接到lintJar中的lintJarOutput方法,調用jar方法,并獲得jar包
  • 將得到的JAR包復制到目錄build/intermediates/lint/下,并且重命名為 lint.jar
  • 當項目build到compileLint這一步時執行copyLintJar方法,這樣的話就可以調用到我們自定義的Lint規則
  • 生成AAR方便項目調用
  • 3.3.3 XTCLintPlugin介紹

    關于 XTCLintPlugin 的介紹,等我們先將自定義規則講解完后再介紹,這里先不介紹。

    四、自定義lint檢查規則的編寫

    上面大致講解了下 XTCLintrRules和XTCLintAAR兩個Module的gradle配置和作用,下面我們來針對XTCLintrRules這個Module來編寫我們自定義的Lint檢查規則

    4.1 創建 Detector

    Detector負責掃描代碼,發現問題并報告。 我們通過一個 XTCCustomLogDetector 這個類來學習下 Detector怎么實現,

    XTCCustomLogDetector 類主要功能是:針對代碼中直接使用android.util.Log的方法 { v,d,i,w,e,wtf }或者直接使用了 System.out.print/System.err.print進行日志打印的一個判斷,然后提示各位開發人員使用我們自定義好的com.xtc.log.LogUtil類進行日志打印。

    package com.xtc.lint.rules.detectors.java;import com.android.tools.lint.client.api.JavaParser; import com.android.tools.lint.detector.api.Category; import com.android.tools.lint.detector.api.Detector; import com.android.tools.lint.detector.api.Implementation; import com.android.tools.lint.detector.api.Issue; import com.android.tools.lint.detector.api.JavaContext; import com.android.tools.lint.detector.api.Scope; import com.android.tools.lint.detector.api.Severity; import com.xtc.lint.rules.JavaPackageRelativePersonUtil;import java.util.Arrays; import java.util.Collections; import java.util.EnumSet; import java.util.List;import lombok.ast.AstVisitor; import lombok.ast.ForwardingAstVisitor; import lombok.ast.MethodInvocation; import lombok.ast.Node; /*** 定義代碼檢查規則* 這個是針對代碼中直接使用android.util.Log的方法 { v,d,i,w,e,wtf } 進行日志打印的一個判斷* </p>* created by OuyangPeng at 2017/8/31 9:55*/ public class XTCCustomLogDetector extends Detector implements Detector.JavaScanner {private static final Class<? extends Detector> DETECTOR_CLASS = XTCCustomLogDetector.class;private static final EnumSet<Scope> DETECTOR_SCOPE = Scope.JAVA_FILE_SCOPE;private static final Implementation IMPLEMENTATION = new Implementation(DETECTOR_CLASS,DETECTOR_SCOPE);private static final String ISSUE_ID = "XTC_LogUseError";private static final String ISSUE_DESCRIPTION = "警告:你應該使用我們團隊自定義的Log打印工具類工具類{com.xtc.log.LogUtil}";private static final String ISSUE_EXPLANATION = "為了能夠更好的控制Log打印的開關,你不能直接使用{android.util.Log}或者{System.out.println}直接打印日志,你應該使用我們團隊自定義的Log打印工具類工具類{com.xtc.log.LogUtil}";private static final Category ISSUE_CATEGORY = Category.CORRECTNESS;private static final int ISSUE_PRIORITY = 9;private static final Severity ISSUE_SEVERITY = Severity.WARNING;private static final String SYSTEM_OUT_PRINT = "System.out.print";private static final String SYSTEM_OUT_PRINTLN = " System.out.println";private static final String SYSTEM_ERR_PRINT = "System.err.print";private static final String SYSTEM_ERR_PRINTLN = " System.err.println";private static final String CHECK_PACKAGE = "android.util.Log";public static final Issue ISSUE = Issue.create(ISSUE_ID,ISSUE_DESCRIPTION,ISSUE_EXPLANATION,ISSUE_CATEGORY,ISSUE_PRIORITY,ISSUE_SEVERITY,IMPLEMENTATION);@Overridepublic List<String> getApplicableMethodNames() {return Arrays.asList("v", "d", "i", "w", "e", "wtf");}@Overridepublic List<Class<? extends Node>> getApplicableNodeTypes() {return Collections.singletonList(MethodInvocation.class);}@Overridepublic AstVisitor createJavaVisitor(final JavaContext context) {return new LogVisit(context);}private class LogVisit extends ForwardingAstVisitor {private final JavaContext javaContext;private LogVisit(JavaContext context) {javaContext = context;}@Overridepublic boolean visitMethodInvocation(MethodInvocation node) {String nodeString = node.toString();if (nodeString.startsWith(SYSTEM_OUT_PRINT)|| nodeString.startsWith(SYSTEM_OUT_PRINTLN)|| nodeString.startsWith(SYSTEM_ERR_PRINT)|| nodeString.startsWith(SYSTEM_ERR_PRINTLN)) {String relativePersonName = JavaPackageRelativePersonUtil.getPackageRelativePerson(javaContext,node); // System.out.println("LogVisit visitMethodInvocation() 出現lint檢測項,對應的責任人為: " + relativePersonName);String message = ISSUE_DESCRIPTION + " ,請 【" + relativePersonName + "】速度修改";javaContext.report(ISSUE, node, javaContext.getLocation(node), message);return true;}JavaParser.ResolvedNode resolve = javaContext.resolve(node);if (resolve instanceof JavaParser.ResolvedMethod) {JavaParser.ResolvedMethod method = (JavaParser.ResolvedMethod) resolve;JavaParser.ResolvedClass containingClass = method.getContainingClass();if (resolve.getName().equals("v")||resolve.getName().equals("d")||resolve.getName().equals("i")||resolve.getName().equals("w")||resolve.getName().equals("e")||resolve.getName().equals("wtf")){ // System.out.println("XTCCustomLogDetector called method one of { v,d,i,w,e,wtf }");if (containingClass.matches(CHECK_PACKAGE)) { // System.out.println("XTCCustomLogDetector called method one of { v,d,i,w,e,wtf } , and the className is : android.util.Log");String relativePersonName = JavaPackageRelativePersonUtil.getPackageRelativePerson(javaContext,node); // System.out.println("LogVisit visitMethodInvocation() 出現lint檢測項,對應的責任人為: " + relativePersonName);String message = ISSUE_DESCRIPTION + " ,請 【" + relativePersonName + "】速度修改";javaContext.report(ISSUE, node, javaContext.getLocation(node),message);return true;}}}return super.visitMethodInvocation(node);}} }

    4.2 Detector介紹

    可以看到這個Detector繼承Detector類,然后實現Scanner接口。

    自定義Detector可以實現一個或多個Scanner接口,選擇實現哪種接口取決于你想要的掃描范圍

    • Detector.XmlScanner
    • Detector.JavaScanner
    • Detector.ClassScanner
    • Detector.BinaryResourceScanner
    • Detector.ResourceFolderScanner
    • Detector.GradleScanner
    • Detector.OtherFileScanner

    這里因為我們是要針對Java代碼掃描,所以選擇使用JavaScanner。

    代碼中getApplicableNodeTypes()方法決定了什么樣的類型能夠被檢測到。這里我們想看Log以及println的方法調用,選取MethodInvocation。

    對應的,我們在createJavaVisitor()創建一個ForwardingAstVisitor通過visitMethodInvocation方法來接收被檢測到的Node。

    可以看到getApplicableNodeTypes()返回值是一個List,也就是說可以同時檢測多種類型的節點來幫助精確定位到代碼,對應的ForwardingAstVisitor接受返回值進行邏輯判斷就可以了。

    可以看到JavaScanner中還有其他很多方法,getApplicableMethodNames(指定方法名)、visitMethod(接收檢測到的方法),這種對于直接找尋方法名的場景會更方便。

    當然這種場景我們用最基礎的方式也可以完成,只是比較繁瑣。

    那么其他Scanner如何去寫呢?
    可以去查看各接口中的方法去實現,一般都是有這兩種對應:什么樣的類型需要返回、接收發現的類型。

    這里插一句,Lint是如何實現Java掃描分析的呢?Lint使用了Lombok做抽象語法樹的分析。所以在我們告訴它需要什么類型后,它就會把相應的Node返回給我們。

    回到示例,當接收到返回的Node之后需要進行判斷,如果調用方法是android.util.Log的方法 { v,d,i,w,e,wtf }或者直接使用了 System.out.print/System.err.print,則調用context.report上報。

    javaContext.report(ISSUE, node, javaContext.getLocation(node), message);

    第一個參數是Issue,這個之后會講到;
    第二個參數是當前節點;
    第三個參數location會返回當前的位置信息,便于在報告中顯示定位;
    最后的字符串用來為警告添加解釋。

    對應報告中的位置如下圖:

    4.2.1 Issue介紹

    Issue由Detector發現并報告,是Android程序代碼可能存在的bug。

    private static final Implementation IMPLEMENTATION = new Implementation(DETECTOR_CLASS,DETECTOR_SCOPE);private static final String ISSUE_ID = "XTC_LogUseError";private static final String ISSUE_DESCRIPTION = "警告:你應該使用我們團隊自定義的Log打印工具類工具類{com.xtc.log.LogUtil}";private static final String ISSUE_EXPLANATION = "為了能夠更好的控制Log打印的開關,你不能直接使用{android.util.Log}或者{System.out.println}直接打印日志,你應該使用我們團隊自定義的Log打印工具類工具類{com.xtc.log.LogUtil}";private static final Category ISSUE_CATEGORY = Category.CORRECTNESS;private static final int ISSUE_PRIORITY = 9;private static final Severity ISSUE_SEVERITY = Severity.WARNING;public static final Issue ISSUE = Issue.create(ISSUE_ID,ISSUE_DESCRIPTION,ISSUE_EXPLANATION,ISSUE_CATEGORY,ISSUE_PRIORITY,ISSUE_SEVERITY,IMPLEMENTATION);

    聲明為final class,由靜態工廠方法創建。對應參數解釋如下:

    • id : 唯一值,應該能簡短描述當前問題。利用Java注解或者XML屬性進行屏蔽時,使用的就是這個id。
    • summary : 簡短的總結,通常5-6個字符,描述問題而不是修復措施。
    • explanation : 完整的問題解釋和修復建議。
    • category : 問題類別。詳見下文詳述部分。
    • priority : 優先級。1-10的數字,10為最重要/最嚴重。
    • severity : 嚴重級別:Fatal, Error, Warning, Informational, Ignore。
    • Implementation : 為Issue和Detector提供映射關系,Detector就是當前Detector。聲明掃描檢測的范圍+ + + Scope,Scope用來描述Detector需要分析時需要考慮的文件集,包括:Resource文件或目錄、Java文件、Class文件。

    4.2.2 Issue與Lint HTML報告對應關系

    對應著 id 和 summary

    對應著 explanation 、category 、severity 、priority

    4.2.2 Category詳述

    系統現在已有的類別如下:

    • Lint
    • Correctness (incl. Messages)
    • Security
    • Performance
    • Usability (incl. Icons, Typography)
    • Accessibility
    • Internationalization
    • Icons
    • Typography
    • Messages

    Category類的部分代碼

    /** Issues related to running lint itself */public static final Category LINT = create("Lint", 110);/** Issues related to correctness */public static final Category CORRECTNESS = create("Correctness", 100);/** Issues related to security */public static final Category SECURITY = create("Security", 90);/** Issues related to performance */public static final Category PERFORMANCE = create("Performance", 80);/** Issues related to usability */public static final Category USABILITY = create("Usability", 70);/** Issues related to accessibility */public static final Category A11Y = create("Accessibility", 60);/** Issues related to internationalization */public static final Category I18N = create("Internationalization", 50);// Sub categories/** Issues related to icons */public static final Category ICONS = create(USABILITY, "Icons", 73);/** Issues related to typography */public static final Category TYPOGRAPHY = create(USABILITY, "Typography", 76);/** Issues related to messages/strings */public static final Category MESSAGES = create(CORRECTNESS, "Messages", 95);/** Issues related to right to left and bidirectional text support */public static final Category RTL = create(I18N, "Bidirectional Text", 40);

    4.2.3 自定義Category

    public class XTCCategory {public static final Category NAMING_CONVENTION = Category.create("小天才命名規范", 101); }

    使用

    public static final Issue ISSUE = Issue.create("IntentExtraKey","intent extra key 命名不規范","請在接受此參數中的Activity中定義一個按照EXTRA_<name>格式命名的常量",XTCCategory.NAMING_CONVENTION , 5, Severity.ERROR,new Implementation(IntentExtraKeyDetector.class, Scope.JAVA_FILE_SCOPE));

    4.3 IssueRegistry

    IssueRegistry就是注冊類,繼承他,并重寫getIssues的方法即可,提供需要被檢測的Issue列表

    例如我們的項目工程中的XTCIssueRegister.java代碼如下

    package com.xtc.lint.rules;import com.android.tools.lint.client.api.IssueRegistry; import com.android.tools.lint.detector.api.Issue; import com.xtc.lint.rules.detectors.binaryResource.XTCImageFileSizeDetector; import com.xtc.lint.rules.detectors.java.XTCActivityFragmentLayoutNameDetector; import com.xtc.lint.rules.detectors.java.XTCChineseStringDetector; import com.xtc.lint.rules.detectors.java.XTCCustomLogDetector; import com.xtc.lint.rules.detectors.java.XTCCloseDetector; import com.xtc.lint.rules.detectors.java.XTCCustomToastDetector; import com.xtc.lint.rules.detectors.java.XTCEnumDetector; import com.xtc.lint.rules.detectors.java.XTCHardcodedValuesDetector; import com.xtc.lint.rules.detectors.java.XTCHashMapForJDK7Detector; import com.xtc.lint.rules.detectors.java.XTCMessageObtainDetector; import com.xtc.lint.rules.detectors.java.XTCViewHolderItemNameDetector; import com.xtc.lint.rules.detectors.xml.XTCViewIdNameDetector;import java.util.Arrays; import java.util.List;public class XTCIssueRegister extends IssueRegistry {static {System.out.println("***************************************************");System.out.println("**************** lint 讀取配置文件 *****************");System.out.println("***************************************************");LoadPropertiesFile.loadPropertiesFile();}@Overridepublic List<Issue> getIssues() {System.out.println("***************************************************");System.out.println("**************** lint 開始靜態分析代碼 *****************");System.out.println("***************************************************");return Arrays.asList(XTCChineseStringDetector.ISSUE,XTCActivityFragmentLayoutNameDetector.ACTIVITY_LAYOUT_NAME_ISSUE,XTCActivityFragmentLayoutNameDetector.FRAGMENT_LAYOUT_NAME_ISSUE,XTCMessageObtainDetector.ISSUE,XTCCustomToastDetector.ISSUE,XTCCustomLogDetector.ISSUE,XTCViewIdNameDetector.ISSUE,XTCViewHolderItemNameDetector.ISSUE,XTCCloseDetector.ISSUE,XTCImageFileSizeDetector.ISSUE,XTCHashMapForJDK7Detector.ISSUE,XTCHardcodedValuesDetector.ISSUE,XTCEnumDetector.ISSUE);} }

    在getIssues()方法中返回需要被檢測的Issue List,我們剛才編寫的 XTCCustomLogDetector.ISSUE 也被注冊進去了。

    在build.grade中聲明Lint-Registry屬性

    /*** Lint-Registry是透露給lint工具的注冊類的方法,* 也就是PermissionIssueRegistry是lint工具的入口,同時也通過這個方法進行打jar包*/ jar{manifest{attributes('Lint-Registry': 'com.xtc.lint.rules.XTCIssueRegister')} }

    至此,自定義Lint的編碼部分就完成了。

    五、為自定義Lint開發plugin

    5.1 為自定義Lint開發plugin的目的

    aar雖然很方便,但是在團隊內部推廣中我們遇到了以下問題:

    • 配置繁瑣,不易推廣。每個庫都需要自行配置lint.xml、lintOptions,并且compile aar。
    • 不易統一。各庫之間需要使用相同的配置,保證代碼質量。但現在手動來回拷貝規則,且配置文件可以自己修改。

    于是我想到開發一個plugin,統一管理lint.xml和lintOptions,自動添加aar。下圖就是我們的工程XTCLintPlugin

    編寫自定義插件需要 實現 Plugin 接口,然后將插件的作用在apply(Project project)方法中實現即可。

    class XTCLintPlugin implements Plugin<Project> {@Overridevoid apply(Project project) {applyTask(project, getAndroidVariants(project))}... }

    5.2 自定義Lint開發plugin需要實現的功能

    5.2.1 統一lint.xml

    我們在plugin中內置lint.xml,執行前拷貝過去,執行完成后刪除。

    //========================== 統一 lint.xml 開始=============================================////lint任務執行前,先復制lint.xmllintTask.doFirst {//如果 lint.xml 存在,則改名為 lintOld.xmlif (lintFile.exists()) {lintOldFile = project.file("lintOld.xml")lintFile.renameTo(lintOldFile)}//進行 將plugin內置的lint.xml文件和項目下面的lint.xml進行復制合并操作def isLintXmlReady = copyLintXml(project, lintFile)//合并完畢后,將lintOld.xml 文件改名為 lint.xmlif (!isLintXmlReady) {if (lintOldFile != null) {lintOldFile.renameTo(lintFile)}throw new GradleException("lint.xml不存在")}}//lint任務執行后,刪除lint.xmlproject.gradle.taskGraph.afterTask { task, TaskState state ->if (task == lintTask) {lintFile.delete()if (lintOldFile != null) {lintOldFile.renameTo(lintFile)}}}//========================== 統一 lint.xml 結束=============================================//

    5.2.2 統一lintOptions

    Android plugin在1.3以后允許我們替換Lint Task的lintOptions

    //========================== 統一 lintOptions 開始=============================================///*lintOptions {lintConfig file("lint.xml")warningsAsErrors trueabortOnError truehtmlReport truehtmlOutput file("lint-report/lint-report.html")xmlReport false}*/def newLintOptions = new LintOptions()//配置lintConfig的配置文件路徑newLintOptions.lintConfig = lintFile//是否將所有的warnings視為errors // newLintOptions.warningsAsErrors = true//是否lint檢測出錯則停止編譯newLintOptions.abortOnError = true//htmlReport打開newLintOptions.htmlReport = truenewLintOptions.htmlOutput = project.file("${project.buildDir}/reports/lint/lint-result.html")//xmlReport打開 因為Jenkins上的插件需要xml文件newLintOptions.xmlReport = truenewLintOptions.xmlOutput = project.file("${project.buildDir}/reports/lint/lint-result.xml")//配置 lint任務的配置為 newLintOptionslintTask.lintOptions = newLintOptions//========================== 統一 lintOptions 結束=============================================//

    5.2.3 自動添加最新aar

    //========================== 統一 自動添加AAR 開始=============================================////配置project的dependencies配置,默認都自動加上 自定義lint檢測的AAR包project.dependencies {//如果是android application項目if (project.getPlugins().hasPlugin('com.android.application')) {compile('com.xtc.lint:lint-check:1.1.1') {force = true}} else {provided('com.xtc.lint:lint-check:1.1.1') {force = true}}}//去除gradle緩存的配置project.configurations.all {resolutionStrategy.cacheChangingModulesFor 0, 'seconds'resolutionStrategy.cacheDynamicVersionsFor 0, 'seconds'}//========================== 統一 自動添加AAR 結束=============================================//

    本來這段代碼

    //配置project的dependencies配置,默認都自動加上 自定義lint檢測的AAR包project.dependencies {//如果是android application項目if (project.getPlugins().hasPlugin('com.android.application')) {compile('com.xtc.lint:lint-check:1.1.1') {force = true}} else {provided('com.xtc.lint:lint-check:1.1.1') {force = true}}}

    中的 lint-check 版本號 我們寫的是 compile(‘com.xtc.lint:lint-check:+’) ,但是張亞州的依賴管理庫不準我們使用+號,因此我們這里寫的是指定好的版本。至此我們的插件功能介紹完畢了,下面是我們的Lint插件XTCLintPlugin的具體實現邏輯代碼

    package com.xtc.lint.pluginimport com.android.build.gradle.AppPlugin import com.android.build.gradle.LibraryPlugin import com.android.build.gradle.api.BaseVariant import com.android.build.gradle.internal.dsl.LintOptions import com.android.build.gradle.tasks.Lint import org.gradle.api.* import org.gradle.api.internal.artifacts.dependencies.DefaultExternalModuleDependency import org.gradle.api.tasks.TaskState/*** aar雖然很方便,但是在團隊內部推廣中我們遇到了以下問題:配置繁瑣,不易推廣。每個庫都需要自行配置lint.xml、lintOptions,并且compile aar。不易統一。各庫之間需要使用相同的配置,保證代碼質量。但現在手動來回拷貝規則,且配置文件可以自己修改。于是我們想到開發一個plugin,統一管理lint.xml和lintOptions,自動添加aar。*/ class XTCLintPlugin implements Plugin<Project> {@Overridevoid apply(Project project) {applyTask(project, getAndroidVariants(project))}private static final String sPluginMissConfiguredErrorMessage = "Plugin requires the 'android' or 'android-library' plugin to be configured."/*** 獲取project 項目 中 android項目 或者library項目 的 variant 列表* @param project 要編譯的項目* @return variants列表*/private static DomainObjectCollection<BaseVariant> getAndroidVariants(Project project) {if (project.getPlugins().hasPlugin(AppPlugin)) {return project.getPlugins().getPlugin(AppPlugin).extension.applicationVariants}if (project.getPlugins().hasPlugin(LibraryPlugin)) {return project.getPlugins().getPlugin(LibraryPlugin).extension.libraryVariants}throw new ProjectConfigurationException(sPluginMissConfiguredErrorMessage, null)}/*** 插件的實際應用:統一管理lint.xml和lintOptions,自動添加aar。* @param project 項目* @param variants 項目的variants*/private void applyTask(Project project, DomainObjectCollection<BaseVariant> variants) {//========================== 統一 自動添加AAR 開始=============================================////配置project的dependencies配置,默認都自動加上 自定義lint檢測的AAR包project.dependencies {//如果是android application項目if (project.getPlugins().hasPlugin('com.android.application')) {compile('com.xtc.lint:lint-check:1.1.1') {force = true}} else {provided('com.xtc.lint:lint-check:1.1.1') {force = true}}}//去除gradle緩存的配置project.configurations.all {resolutionStrategy.cacheChangingModulesFor 0, 'seconds'resolutionStrategy.cacheDynamicVersionsFor 0, 'seconds'}//========================== 統一 自動添加AAR 結束=============================================//def xtcLintTaskExists = falsevariants.all { variant ->//獲取Lint Taskdef variantName = variant.name.capitalize()Lint lintTask = project.tasks.getByName("lint" + variantName) as Lint//Lint 會把project下的lint.xml和lintConfig指定的lint.xml進行合并,為了確保只執行插件中的規則,采取此策略File lintFile = project.file("lint.xml")File lintOldFile = null//========================== 統一 lintOptions 開始=============================================///*lintOptions {lintConfig file("lint.xml")warningsAsErrors trueabortOnError truehtmlReport truehtmlOutput file("lint-report/lint-report.html")xmlReport false}*/def newLintOptions = new LintOptions()//配置lintConfig的配置文件路徑newLintOptions.lintConfig = lintFile//是否將所有的warnings視為errors // newLintOptions.warningsAsErrors = true//是否lint檢測出錯則停止編譯newLintOptions.abortOnError = true//htmlReport打開newLintOptions.htmlReport = truenewLintOptions.htmlOutput = project.file("${project.buildDir}/reports/lint/lint-result.html")//xmlReport打開 因為Jenkins上的插件需要xml文件newLintOptions.xmlReport = truenewLintOptions.xmlOutput = project.file("${project.buildDir}/reports/lint/lint-result.xml")//配置 lint任務的配置為 newLintOptionslintTask.lintOptions = newLintOptions//========================== 統一 lintOptions 結束=============================================////========================== 統一 lint.xml 開始=============================================////lint任務執行前,先復制lint.xmllintTask.doFirst {//如果 lint.xml 存在,則改名為 lintOld.xmlif (lintFile.exists()) {lintOldFile = project.file("lintOld.xml")lintFile.renameTo(lintOldFile)}//進行 將plugin內置的lint.xml文件和項目下面的lint.xml進行復制合并操作def isLintXmlReady = copyLintXml(project, lintFile)//合并完畢后,將lintOld.xml 文件改名為 lint.xmlif (!isLintXmlReady) {if (lintOldFile != null) {lintOldFile.renameTo(lintFile)}throw new GradleException("lint.xml不存在")}}//lint任務執行后,刪除lint.xmlproject.gradle.taskGraph.afterTask { task, TaskState state ->if (task == lintTask) {lintFile.delete()if (lintOldFile != null) {lintOldFile.renameTo(lintFile)}}}//========================== 統一 lint.xml 結束=============================================////========================== 在終端 執行命令 gradlew lintForXTC 的配置 開始=============================================//// 在終端 執行命令 gradlew lintForXTC 的時候,則會應用 lintTaskif (!xtcLintTaskExists) {xtcLintTaskExists = true//創建一個task 名為 lintForXTCproject.task("lintForXTC").dependsOn lintTask}//========================== 在終端 執行命令 gradlew lintForXTC 的配置 結束=============================================//}}/*** 復制 lint.xml 到 targetFile* @param project 項目* @param targetFile 復制到的目標文件* @return 是否復制成功*/boolean copyLintXml(Project project, File targetFile) {//創建目錄targetFile.parentFile.mkdirs()//目標文件為 resources/config/lint.xml文件InputStream lintIns = this.class.getResourceAsStream("/config/lint.xml")OutputStream outputStream = new FileOutputStream(targetFile)int retroLambdaPluginVersion = getRetroLambdaPluginVersion(project)if (retroLambdaPluginVersion >= 180) {// 加入屏蔽try with resource 檢測 1.8.0版本引入此功能InputStream retroLambdaLintIns = this.class.getResourceAsStream("/config/retrolambda_lint.xml")XMLMergeUtil.merge(outputStream, "/lint", lintIns, retroLambdaLintIns)} else {// 未使用 或 使用了不支持try with resource的版本IOUtils.copy(lintIns, outputStream)IOUtils.closeQuietly(outputStream)IOUtils.closeQuietly(lintIns)}//如果復制操作完成后,目標文件存在if (targetFile.exists()) {return true}return false}/*** 獲取 使用的 RetroLambda Plugin插件的版本* @param project 項目* @return 沒找到時返回-1 ,找到返回正常version*/def static int getRetroLambdaPluginVersion(Project project) {DefaultExternalModuleDependency retroLambdaPlugin = findClassPathDependencyVersion(project, 'me.tatarka', 'gradle-retrolambda') as DefaultExternalModuleDependencyif (retroLambdaPlugin == null) {retroLambdaPlugin = findClassPathDependencyVersion(project.getRootProject(), 'me.tatarka', 'gradle-retrolambda') as DefaultExternalModuleDependency}if (retroLambdaPlugin == null) {return -1}return retroLambdaPlugin.version.split("-")[0].replaceAll("\\.", "").toInteger()}/*** 查找Dependency的Version信息* @param project* @param group* @param attributeId* @return*/def static findClassPathDependencyVersion(Project project, group, attributeId) {return project.buildscript.configurations.classpath.dependencies.find {it.group != null && it.group.equals(group) && it.name.equals(attributeId)}} }

    關于Android團隊使用內部Lint檢測的指導文檔2 —> 集成 Lint插件自動添加Lint檢測的AAR.md

    六、應用自定義Lint到項目中去

    我們剛才說了,我們有自定義Lint的AAR,你可以直接加入到項目中,當然也可以通過自定義的插件來知己添加到項目中去。

    6.1 添加自定義Lint檢測的AAR

    6.1.1 添加自定義Lint檢測的AAR

    在項目的build.gradle 里面添加如下代碼

    compile 'com.xtc.lint:lint-check:1.0.1'

    目前版本為1.0.1,之后隨著自定義規則越來越多,版本再逐步升高。

    6.2 執行lint檢測命令

    6.2.1 本地執行lint命令

    執行如下lint檢測命令

    gradlew clean lint

    開始執行命令

    開始Lint檢測

    檢測完畢

    6.2.2 Android Studio執行 Inspect Code

    如果上面的命令你用的不習慣,你可以使用Android Studio 自帶的代碼檢測

    打開【Analyze】—>【Inspect Code】

    彈出如下所示的Scope對話框,你可以選擇檢測的范圍

    選擇完后,點擊【OK】按鈕即開始進行檢測。

    APP項目比較大,可能檢測得幾分鐘,請等待。

    檢測完后,會出現上圖所示的選項框,在Android Lint 下面可以看到我們自定義的Lint檢測規則

    6.2.3 Jenkins執行lint命令

    修改Jenkins的編譯命令為

    clean lint build --stacktrace

    編譯完后,將lint報告歸檔到FTP

    Source files 填寫

    watch/build/outputs/lint-results-debug.html,lint-results-debug.xml
    改目錄要根據自己項目實際輸出的lint報告來填寫,可以自己查看Jenkins的工作區生成的目錄

    Remote directory 填寫

    'Android/Common/${JOB_NAME}/'yyyy-MM-dd-HH-mm-ss-'build-${BUILD_NUMBER}-git-${GIT_COMMIT}'

    然后點擊【高級】選項,勾選上【Flatten files】和【Remote directory】

    這樣編譯完后,在FTP服務器對應的目錄上就會有lint檢測報告了

    6.3 查看Lint檢測報告

    6.3.1 lint報告輸出目錄

    lint命令執行完畢之后,會輸出相應的lint報告,不同的gradle版本輸出的文檔地址可能不同,APP的輸出目錄為:\watch\build\outputs

    gradle版本高的,輸出目錄可能為 \app\build\reports,如下所示

    6.3.2 報告詳情

    打開 \watch\build\outputs\lint-results-debug.html 網頁即可看到輸出的lint檢測報告。

    • 控件命名

    • 圖片太大

    • Log打印

    • Toast

    如上圖所示,我們就可以看到自定義規則出來的代碼問題。請大家逐步修改檢測出來的問題,后期的代碼編譯會將lint檢測出來的問題作為一個指標。

    6.4 添加自定義的Lint檢測插件

    上一個方法是要自己手動的添加Lint檢測的AAR包到項目中,但是aar雖然很方便,但是在團隊內部推廣中我們遇到了以下問題:

  • 配置繁瑣,不易推廣。每個庫都需要自行配置lint.xml、lintOptions,并且compile aar。
  • 不易統一。各庫之間需要使用相同的配置,保證代碼質量。但現在手動來回拷貝規則,且配置文件可以自己修改。
  • 于是我開發一個plugin,統一管理lint.xml和lintOptions,自動添加aar,下面是介紹如何添加該Plugin

    6.4.1 在根目錄下的在build.gradle 中,添加Lint檢測插件

    如上圖所示,在 buildscript 中的 dependencies 塊中,添加Lint檢測的插件地址,

    //lint 檢測插件 classpath 'com.xtc.lint:lint-check-plugin:1.0.7-Dev'

    6.4.2 在module的build.gradle中,應用Lint檢測插件

    如上圖所示,添加如下代碼即可

    apply plugin: 'XTCLintPlugin'

    并且把之前已經添加過的lint檢測aar包代碼去掉,

    6.4.3 運行命令 gradlew lintForXTC 即可正常應用插件統一配置的lint

    6.4.3.1 在Android Studio中

    gradlew clean lintForXTC

    這樣就可以使用 插件統一配置的lint,進行靜態代碼分析了。

    編譯完后的lint報告,位置位于:lint-report目錄下,如下所示:

    6.4.3.1 在Jenkins中

    構建
    在Jenkins中,修改下 構建的 任務 為

    clean lintForXTC build --stacktrace

    侯建后的操作

    1、修改 Publish Android Lint results

    修改為 watch/lint-report/lint-results*.xml

    2、修改 FTP Publishers
    增加一項,將lint檢測的html報告保存到ftp路徑

    Source files 路徑填寫

    watch/lint-report/lint-results.html

    Remote directory 遠程保存路徑填寫

    'Android/Common/${JOB_NAME}/'yyyy-MM-dd-HH-mm-ss-'build-${BUILD_NUMBER}-git-${GIT_COMMIT}'

    構建過程,lint檢測過程中

    構建后的結果
    編譯完后,位置位于:lint-report目錄下,如下所示:

    保存到FTP的報告如下

    可以將html文件 下載到本地然后使用瀏覽器打開,如下所示:

    點擊【Lint Issues】可以查看Jenkins插件的Lint報告

    查看具體細節的issue和上面介紹的一樣。

    七、參考文獻

    • https://tech.meituan.com/android_custom_lint.html
    • https://tech.meituan.com/android_custom_lint2.html
    • https://github.com/GavinCT/MeituanLintDemo
    • http://blog.csdn.net/u010360371/article/details/50189171
    • https://yq.aliyun.com/articles/6918
    • http://www.androidchina.net/5106.html
    • https://engineering.linkedin.com/android/writing-custom-lint-checks-gradle
    • https://github.com/yongce/AndroidDevNotes/blob/master/notes/knowledge/0005-custom-lint.asc
    • https://google.github.io/android-gradle-dsl/current/com.android.build.gradle.internal.dsl.LintOptions.html#com.android.build.gradle.internal.dsl.LintOptions:quiet
    • http://blog.csdn.net/itfootball/article/details/49277207
    • https://testerhome.com/topics/3105
    • http://blog.csdn.net/hwhua1986/article/details/50067089
    • http://www.jianshu.com/p/761c88d095a2

    作者:歐陽鵬 歡迎轉載,與人分享是進步的源泉!
    轉載請保留原文地址:https://blog.csdn.net/ouyang_peng/article/details/80374867

    如果覺得本文對您有所幫助,歡迎您掃碼下圖所示的支付寶和微信支付二維碼對本文進行隨意打賞。您的支持將鼓勵我繼續創作!

    總結

    以上是生活随笔為你收集整理的【我的Android进阶之旅】Android自定义Lint实践的全部內容,希望文章能夠幫你解決所遇到的問題。

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

    色久综合 | 午夜久久久影院 | 日日干av | 亚洲欧洲成人精品av97 | 在线电影日韩 | 97国产精品 | 国产成人精品一区二区 | www.干| 国产精品一区二区免费 | 久久久久久久久久免费 | 99九九视频 | 啪一啪在线 | 国产精品久久人 | 丁香六月天婷婷 | 国产中文视频 | 亚洲1级片| 久久综合九色欧美综合狠狠 | 国产手机视频 | 国产免费嫩草影院 | 国产视频精品免费 | 亚洲精品xx| 欧美日韩亚洲在线观看 | 麻豆果冻剧传媒在线播放 | 免费观看性生活大片 | 狠狠的日日 | 人人插人人搞 | 日韩久久精品一区二区 | 久草免费在线观看 | 狠狠夜夜 | 国产一级免费观看视频 | 亚洲天堂网站视频 | 成人91在线| 欧美精品久久久久久 | 国产不卡毛片 | av网址在线播放 | 国产精品99精品久久免费 | www视频免费在线观看 | 欧美激情精品久久久久久 | 91aaa在线观看| 精品av网站 | 亚洲欧美成人网 | 日本中文字幕视频 | 激情久久五月 | 国产小视频在线免费观看视频 | 免费日韩一级片 | 天天干,天天草 | 日韩精品一区二区三区中文字幕 | 国产精品理论片在线播放 | 午夜av免费观看 | 日韩欧美在线视频一区二区三区 | 国产精品一区二区无线 | 久久综合中文字幕 | 激情开心网站 | 日韩三级视频在线观看 | 在线观看亚洲电影 | 国产录像在线观看 | 免费一级片观看 | 在线观看成人 | 黄色小说在线免费观看 | 久久精品3 | 超级碰99 | 亚州精品在线视频 | 日韩午夜精品福利 | 色吊丝在线永久观看最新版本 | 免费看毛片网站 | 久久久久国产精品一区二区 | 国产在线综合视频 | 久久亚洲私人国产精品 | 日韩视频中文字幕 | 久久国产精品影片 | 超级碰碰免费视频 | 亚洲一区二区三区四区精品 | 天天摸天天操天天舔 | 国产在线p | 不卡av免费在线观看 | www.天天干.com | 色av婷婷 | 日韩精品一区二区久久 | 日本在线中文在线 | 99久久综合国产精品二区 | 色干干| 日韩不卡高清视频 | 日日夜夜免费精品视频 | 黄色高清视频在线观看 | 中文字幕刺激在线 | 国产精品一区二区三区电影 | 国产午夜精品一区二区三区 | 日韩资源在线观看 | 欧美日韩三级在线观看 | 亚洲干视频在线观看 | 久久蜜臀一区二区三区av | 91激情视频在线 | 久久久精品99 | av免费看av| 久久国产经典视频 | 天堂在线视频中文网 | 久久久久久视频 | 国产精品2020 | 日韩精品不卡在线 | 亚洲日韩中文字幕 | 国产黄色特级片 | 一区二区欧美在线观看 | 九九热久久久 | 激情五月五月婷婷 | 天天色图 | 久久成人综合视频 | 黄色a大片 | 97超碰免费在线观看 | 国内精品久久久久久 | 国产精品亚洲综合久久 | 在线a人v观看视频 | 999色视频| 99精品国产在热久久下载 | 久99久精品 | 在线观看国产成人av片 | 四虎在线观看网址 | 最近中文字幕在线播放 | 久久少妇免费视频 | 娇妻呻吟一区二区三区 | 在线www色 | 亚洲欧洲国产视频 | 蜜臀久久99精品久久久无需会员 | 国内成人精品2018免费看 | 天堂av在线 | 色吊丝在线永久观看最新版本 | 99久久久久久久久 | 亚洲 在线 | 国产999精品久久久影片官网 | 久草视频网 | 免费网站黄色 | 天堂av免费观看 | 久热免费在线观看 | 国产午夜精品一区二区三区四区 | 亚洲黄色免费电影 | 精品久久久99 | 国产97碰免费视频 | 黄a在线看 | 黄网站污 | 一区二区视频在线看 | 丁香六月天 | 婷久久 | 99热这里只有精品在线观看 | 中文字幕中文字幕中文字幕 | 日韩精品一区二区三区三炮视频 | 日韩爱爱网站 | 亚洲.www| 免费在线观看日韩视频 | 亚洲人成人99网站 | 欧美日本高清视频 | 国产成人一二三 | 国产一区精品在线 | 午夜精品久久久久久久久久 | 91刺激视频 | 精品视频久久 | 911国产精品 | 成人免费观看完整版电影 | 成人cosplay福利网站 | 亚洲精品视频偷拍 | 中文字幕人成人 | 国产一区二区视频在线 | 人人dvd| 日日弄天天弄美女bbbb | 亚洲一区二区视频在线 | 免费观看一级特黄欧美大片 | 国产毛片aaa | 国产精彩视频一区二区 | www.av免费| 一级a性色生活片久久毛片波多野 | www激情com| 成人午夜精品福利免费 | 在线播放日韩 | 91av九色| 日韩欧美黄色网址 | 亚洲日b视频 | 高清不卡免费视频 | 亚洲精品在线视频网站 | 日韩免费三区 | 人人dvd| 99精品区| 久久精品1区2区 | 黄色一级性片 | 日韩高清一区 | av千婊在线免费观看 | 热久久最新地址 | 五月婷婷综合激情 | 日韩簧片在线观看 | 久久视屏网 | 干综合网 | www.99久久.com | 婷婷色亚洲| 国产九九精品 | 欧美国产日韩一区 | 麻豆成人精品视频 | 色妞色视频一区二区三区四区 | 91亚洲精品久久久久图片蜜桃 | 日韩成人一级大片 | 精品亚洲va在线va天堂资源站 | 精品一区三区 | 成人一级影视 | 国产成人精品午夜在线播放 | 亚洲综合色av | 97视频在线观看播放 | 日本少妇高清做爰视频 | 日本久热| 欧美日韩免费一区二区 | 国产精品成人久久久久久久 | 婷婷丁香av | 99日韩精品| 麻豆影视在线播放 | 日韩一区二区三区高清免费看看 | 久久精品导航 | 国内精品视频久久 | 日韩精品一区二区三区不卡 | 日日射天天射 | 欧美人人爱 | 日韩在线电影一区二区 | 久久99国产精品 | 色婷婷久久久综合中文字幕 | 91探花国产综合在线精品 | 久久男人视频 | av一级免费 | 午夜免费久久看 | 免费看日韩 | 久精品视频 | 九九久久影视 | 五月色丁香 | bbw av| 欧美大香线蕉线伊人久久 | 97精品国产一二三产区 | 狠狠干成人综合网 | 91麻豆精品久久久久久 | 免费福利视频导航 | 亚洲综合色激情五月 | 精品国产乱码 | 日本视频精品 | 丁香婷婷久久 | 91九色精品 | 国产成人av综合色 | 国产在线资源 | 国产在线观看av | 久久婷婷一区二区三区 | 97超碰人人澡人人爱学生 | 亚洲视频网站在线观看 | 欧美日韩天堂 | 国产 日韩 在线 亚洲 字幕 中文 | 亚洲精品国产欧美在线观看 | 日韩在线视频免费播放 | 最新的av网站 | 亚洲作爱| 久久中文字幕导航 | 久久国产经典视频 | 91亚洲精品久久久中文字幕 | 欧美日韩99 | 国产一级高清 | 狠狠激情中文字幕 | 亚洲精品在线观看视频 | 国产黄色免费观看 | 欧美va天堂va视频va在线 | 日韩一二三在线 | 亚洲国产精品va在线看黑人动漫 | 久久视频6| 欧美日韩国产综合一区二区 | 天天在线免费视频 | 草久中文字幕 | 深爱激情五月网 | 婷婷色综合网 | 伊人一级| 91久色蝌蚪 | 国产在线p | 成人免费观看视频网站 | 亚洲视频一级 | 日韩免费在线观看网站 | 国产视频一区在线播放 | 中文在线字幕免费观 | 成人一级在线观看 | 国产免费视频在线 | 国产精品国产三级国产aⅴ无密码 | 国产在线免费观看 | 亚洲精品乱码白浆高清久久久久久 | 午夜影视av | 91成人在线视频观看 | 在线精品亚洲 | 免费在线观看污网站 | 成人在线免费av | 国产伦理一区 | 欧美久久久久久久久久久 | 激情综合五月 | 色美女在线| 在线观影网站 | 国产区网址 | 黄色av电影 | 久久精品久久国产 | 国产123区在线观看 国产精品麻豆91 | 亚洲精品久久激情国产片 | 九色91在线 | 亚洲欧洲一区二区在线观看 | 国产精品日韩精品 | 亚洲一级片 | 日韩精品中文字幕在线不卡尤物 | 国产精品mv在线观看 | 成年人在线播放视频 | 国产特级毛片aaaaaa | 毛片99 | 天天玩夜夜操 | 99色视频在线 | 啪啪肉肉污av国网站 | 久久色在线观看 | 极品久久久久 | 日韩视频一区二区三区 | 不卡视频在线看 | 日本精品在线视频 | 国产精品精 | 黄a在线看 | 国产精品美女久久久久久久网站 | 久久婷婷一区二区三区 | 在线免费观看黄色大片 | 国产精品专区一 | 色吊丝在线永久观看最新版本 | 久二影院 | 色视频成人在线观看免 | 久久九九影视 | 欧美日韩在线视频免费 | 国产精品久久久久久久电影 | 99精品国产一区二区三区不卡 | 午夜视频在线观看网站 | 亚洲成av人片在线观看无 | 久久人91精品久久久久久不卡 | 伊人久久av | 中文字幕视频一区 | 精品久久久久国产 | 成年人在线观看免费视频 | 欧美日韩高清在线一区 | 在线观看mv的中文字幕网站 | 国产精品久久艹 | 在线国产福利 | 免费婷婷| 精品国产aⅴ麻豆 | 人人澡视频 | 国产在线精| 青青草国产在线 | 亚洲精品中文在线资源 | 成人免费看视频 | 久久久久久久看片 | 欧美日韩国产一区二区三区在线观看 | 国产精品亚洲a | 激情网站 | 福利区在线观看 | 激情视频国产 | 国产午夜亚洲精品 | 中文在线字幕免 | 久久99深爱久久99精品 | 久草在线免费播放 | 美女视频黄在线观看 | 亚洲免费在线观看视频 | 久久精品国产成人 | 久久久久久高潮国产精品视 | 97香蕉久久超级碰碰高清版 | 丁香在线视频 | 中文字幕精品一区二区精品 | 久久精品在线免费观看 | 碰超在线 | 97超碰人人澡人人爱学生 | 91网在线观看 | 久久伊人精品一区二区三区 | 91久久精品日日躁夜夜躁国产 | 97视频在线观看网址 | 99精品国产在热久久 | 亚洲精品综合欧美二区变态 | 香蕉网在线观看 | 国产一区黄色 | 日本精品视频在线 | 国产淫片 | 国产精品高潮呻吟久久av无 | 在线观看黄网 | 欧美黑人性爽 | 91黄色影视 | 国产精品欧美久久久久三级 | 国产原厂视频在线观看 | 国产精品区免费视频 | 欧美三级高清 | 日韩中字在线 | 国产精品嫩草影院123 | 欧美日韩中文在线视频 | 激情片av| 在线看一区 | 黄色免费视频在线观看 | 探花视频在线观看免费版 | 一二区电影| 国产香蕉97碰碰碰视频在线观看 | 国产高清综合 | 日本公妇在线观看高清 | 91精品国产一区二区在线观看 | 免费欧美高清视频 | 国产在线播放一区二区 | 国产精品男女视频 | 狠狠狠色丁香婷婷综合久久88 | 丁香久久久| 国产亚洲精品久久久久5区 成人h电影在线观看 | 草久在线观看 | 国产亚洲在线视频 | 在线激情网 | av电影免费看 | 精品视频免费 | 天天草夜夜 | 国产精品久久99精品毛片三a | 日产中文字幕 | 国产精品一区二区三区在线免费观看 | 国产福利一区二区三区视频 | 国产综合在线观看视频 | 日韩av不卡在线 | 亚洲精品网站 | 欧美analxxxx| 最近中文字幕 | 日韩a级黄色片 | 最近中文字幕在线中文高清版 | 色综合久久综合中文综合网 | 在线午夜电影神马影院 | 国产精品福利无圣光在线一区 | 日韩久久精品 | 国产色视频一区二区三区qq号 | 精品久久网 | 国产日韩在线视频 | 国产香蕉97碰碰碰视频在线观看 | 日韩精品一区二区三区免费观看视频 | 精品久久久久久久久久久久久久久久 | 国产日韩欧美在线观看视频 | 国产免费视频一区二区裸体 | 欧美日韩国产网站 | 天天综合视频在线观看 | 午夜色大片在线观看 | 久久在视频 | av综合网址 | 免费成人在线电影 | 日韩高清久久 | 亚洲欧美在线视频免费 | 国产精品美女在线 | 嫩草av在线| 少妇激情久久 | 国产精品ⅴa有声小说 | 亚洲九九影院 | 亚洲成人资源在线观看 | 日韩精品中文字幕在线播放 | 欧美一级视频免费看 | 又污又黄网站 | 亚洲成人一二三 | 99re热精品视频 | 国产一级二级在线观看 | 久精品在线观看 | 午夜精品一二区 | 在线精品在线 | 69视频永久免费观看 | 国产理论一区二区三区 | 国产一区二区在线观看免费 | 国模精品一区二区三区 | 丁香六月伊人 | 最近中文字幕视频完整版 | 五月天激情综合 | 天天草天天色 | 日韩在线视 | 国产又粗又硬又爽的视频 | 久久久久久久国产精品 | 在线黄色观看 | 色婷婷欧美 | 黄色大全在线观看 | 国产精品第三页 | 欧美日韩免费看 | 国产99久久精品 | 一级精品视频在线观看宜春院 | 国产亚洲精品久久久久久 | 日韩久久久久久 | 91免费版在线 | 99久久这里有精品 | 日本中文字幕久久 | 久久精品国产一区二区 | 欧美激情精品久久久久久免费印度 | 天天爱天天操天天射 | 国产群p视频 | 天天搞天天干天天色 | 国产精品久久久久久久久蜜臀 | 国产精品69av | 五月开心激情网 | 国产成人黄色av | 欧美精品一区二区性色 | 国产婷婷久久 | 黄色的网站免费看 | 国产午夜三级一区二区三桃花影视 | 国产精品麻豆果冻传媒在线播放 | 狠狠干夜夜操天天爽 | 91精品国自产在线 | 99久久这里有精品 | 在线精品视频免费观看 | 999精品| 久久免费激情视频 | 久久精品永久免费 | 亚洲片在线资源 | 欧美性生活小视频 | 国产精品久久久久久久av电影 | 欧美激情综合色综合啪啪五月 | 中文字幕日本电影 | 日韩在线观看视频中文字幕 | 美女视频永久黄网站免费观看国产 | 日韩精品免费在线 | 深爱婷婷网 | 欧美激情综合色 | 91久久精品一区 | 国产自制av | 亚洲女同ⅹxx女同tv | 日韩欧美在线观看一区二区三区 | 在线成人免费 | 在线视频观看成人 | 中文字幕a∨在线乱码免费看 | 日韩中文在线电影 | 国产午夜三级一区二区三 | 99色在线视频 | 国产成人精品一区二三区 | 91人人视频在线观看 | 亚洲天堂自拍视频 | 亚洲精品动漫成人3d无尽在线 | 黄网站app在线观看免费视频 | 激情网站五月天 | 粉嫩av一区二区三区四区 | 免费看一级 | 日韩h在线观看 | 狠狠狠狠狠狠天天爱 | 久草热久草视频 | 国产免费小视频 | 91超在线| 91视频高清完整版 | 免费看片成人 | 国产久视频| www.香蕉视频在线观看 | 成人免费在线视频观看 | 午夜av免费在线观看 | 国产v欧美 | 在线观看视频h | 一级精品视频在线观看宜春院 | 亚洲欧洲一区二区在线观看 | 亚洲男模gay裸体gay | 97高清视频 | 免费在线观看av电影 | 久久99精品国产91久久来源 | 日日操操操 | 国产色在线 | 国产黄色免费看 | 天天天天天天天操 | 国产精品九九久久久久久久 | 亚洲精品日韩一区二区电影 | 亚洲精品一区二区精华 | a黄色大片 | 黄视频色网站 | 狠狠躁日日躁夜夜躁av | 亚洲精品乱码久久久久v最新版 | 在线综合 亚洲 欧美在线视频 | 色播五月激情综合网 | av电影中文字幕在线观看 | 国产黄色片久久久 | av动态图片 | 色综合久久久久综合体桃花网 | 18网站在线观看 | 一级做a视频 | 五月天电影免费在线观看一区 | 亚洲精品在线观 | 四虎国产精品免费 | 免费麻豆| 黄色av影院 | 国产97色在线 | 一级黄色大片在线观看 | 免费日韩av片| 91九色视频在线观看 | 欧美一区二区三区不卡 | 麻豆系列在线观看 | 黄色av网站在线免费观看 | 久久精品亚洲 | 亚洲国产理论片 | av在线播放网址 | 黄在线免费观看 | 久久久久国产a免费观看rela | 韩日av在线 | 国产剧情久久 | 四虎www com| 国产精品一区二区三区观看 | 综合天天色 | 久久99精品国产99久久 | 99在线观看视频 | 欧美淫视频 | 激情综合五月 | 精品国产99 | 美女视频黄,久久 | 国产成人精品国内自产拍免费看 | 久久优 | 国产精品私人影院 | 精品国产一区二区三区久久久蜜月 | 国产精品中文字幕在线观看 | 中文字幕视频播放 | 成人午夜影视 | 91看毛片| 97香蕉视频 | 国产精品久久久久一区二区 | 国产91精品一区二区麻豆网站 | 日韩美女高潮 | 日韩精品一区二区三区在线视频 | 中文字幕高清免费日韩视频在线 | 国产在线91在线电影 | 成人性生交大片免费观看网站 | 热久久影视 | 免费看黄色大全 | 日本午夜在线亚洲.国产 | 亚洲成人av电影 | 91香蕉亚洲精品 | 日韩va在线观看 | 国产精品一区二区电影 | 69国产精品成人在线播放 | 欧美精品在线观看 | av黄色在线观看 | 久久国产精品精品国产色婷婷 | 最近久乱中文字幕 | www国产一区 | 国产精品va最新国产精品视频 | 亚洲国产剧情av | 色99中文字幕 | 精品视频成人 | 亚州精品成人 | 韩国三级av在线 | 欧美成人影音 | 日本一区二区三区免费观看 | 亚洲精品色婷婷 | 四虎成人免费观看 | 久久嗨 | 国产精品久99 | 国产一区二区三区在线 | 园产精品久久久久久久7电影 | 国产视频精选在线 | 久久精品理论 | 干干日日 | 九九热精品国产 | 亚洲一区尤物 | 综合色综合| 欧美一区二区三区在线视频观看 | 久久曰视频 | 五月婷婷开心 | 婷婷福利影院 | 亚洲精品网址在线观看 | 国产精品自产拍在线观看蜜 | 国产精品久久久久aaaa | 国产精品女人久久久久久 | 婷婷中文在线 | 日韩在线免费 | 欧美九九九 | 国产福利在线不卡 | 97超碰资源网 | 日韩一区二区三区在线看 | 久久国产精品电影 | 欧美三级高清 | 有码中文在线 | 国产中文字幕三区 | 国产青草视频在线观看 | 精品亚洲午夜久久久久91 | 欧美日韩在线网站 | 九九热免费观看 | 悠悠av资源片 | 国产在线视频一区二区 | 天天曰天天干 | 久久99深爱久久99精品 | 懂色av一区二区三区蜜臀 | 天天爽天天搞 | 国产在线成人 | 91精品麻豆 | 成人免费一区二区三区在线观看 | 日本视频网 | 精品视频专区 | 国产一区二区精品久久 | 99久久精品免费看国产免费软件 | 久久国产精品成人免费浪潮 | 在线观看av大片 | 国模一区二区三区四区 | 国产中文字幕在线观看 | 日韩欧美有码在线 | 欧美俄罗斯性视频 | 久国产在线播放 | 免费久久片 | 精品人人人 | 欧美一区二区三区激情视频 | 国产美腿白丝袜足在线av | 国产福利在线免费 | 激情开心网站 | 久久一区二区免费视频 | 成人黄色毛片视频 | 91在线精品秘密一区二区 | 韩日电影在线观看 | 91视频成人免费 | 国产精品日韩欧美一区二区 | 成年人电影免费在线观看 | 国产精品免费成人 | 国产区免费 | 麻豆传媒精品 | 亚洲日本va中文字幕 | 久久久久久久久久久久久久av | 国产视频91在线 | 91亚色免费视频 | 伊人永久 | 亚洲日本在线视频观看 | 亚洲狠狠丁香婷婷综合久久久 | 在线观看黄污 | 国产无遮挡又黄又爽在线观看 | 九九一级片 | 国产v亚洲v | 超碰在线人人草 | 亚洲第二色 | 国产麻豆精品一区二区 | 91精品视频免费观看 | 亚洲综合在线发布 | 久久黄网站 | 免费av高清 | 免费看v片网站 | 国产精品区免费视频 | 久久精品播放 | 又湿又紧又大又爽a视频国产 | 免费视频 你懂的 | 免费三级网| 日韩中文字幕在线观看 | 国产精品久久久久久爽爽爽 | 中文字幕在线看视频国产 | 92精品国产成人观看免费 | 一区二区三区在线视频观看58 | 国产精品久久久久久久毛片 | 91成人蝌蚪 | 美女性爽视频国产免费app | 视频在线观看入口黄最新永久免费国产 | 香蕉视频91 | 国产在线精品视频 | 91精品国产高清自在线观看 | 国产精品99久久久久久小说 | 美女视频黄频大全免费 | 免费黄色小网站 | 91精彩在线视频 | 91一区在线观看 | 日韩电影在线一区 | 美女在线国产 | 国产首页| 精品久久久久久久久久久久久久久久久久 | 国产精品ssss在线亚洲 | 玖玖视频免费在线 | 欧美亚洲另类在线视频 | 欧美成人h版电影 | 国内毛片毛片 | 婷婷视频 | 天天干人人 | 亚洲三级在线免费观看 | 精品免费久久 | 久久黄色精品视频 | 亚洲深夜影院 | 粉嫩一二三区 | 国产在线精品观看 | 国产精品一区电影 | 日韩超碰在线 | 国产成人av电影在线 | 免费观看的av | 天天拍天天色 | 国产中文字幕网 | 99久久精品无码一区二区毛片 | 美女视频免费精品 | 国产中文在线观看 | 日韩二区三区 | 97在线免费观看 | 日韩av手机在线观看 | 国产精品毛片一区二区 | 国产 日韩 欧美 自拍 | 视频成人免费 | 欧美国产高清 | 91在线视频网址 | 91成人免费在线视频 | 看黄色91| 正在播放 久久 | 国产成人在线播放 | 欧美在线观看视频一区二区 | 蜜臀av性久久久久av蜜臀妖精 | 手机成人在线电影 | 日韩免费高清在线观看 | 午夜视频在线瓜伦 | 久久精品一区二区三 | 99国产免费网址 | 一区二区精品在线 | 91精品亚洲影视在线观看 | 欧美一区影院 | av解说在线 | 天天色棕合合合合合合 | 欧美 日韩 视频 | 在线性视频日韩欧美 | 天堂久久电影网 | 久久久久福利视频 | 91成人精品一区在线播放69 | 国产首页 | 日韩在线观看中文字幕 | 在线观看视频一区二区三区 | 在线看污网站 | 国产综合91 | 日韩特黄一级欧美毛片特黄 | 涩涩资源网| 在线观看深夜福利 | 99热国产在线 | 亚洲免费永久精品国产 | 久草久视频 | 欧美美女一级片 | 欧美日一级片 | 国产一区二区中文字幕 | 久久久久久网站 | 精品免费观看 | 成人va天堂 | 国产高清在线观看av | 午夜精品久久久久久久久久久久久久 | 亚洲精品在线观看视频 | 中文字幕 在线看 | 亚洲国产激情 | 精品久久久久国产免费第一页 | 黄色大全在线观看 | 欧美日韩一区二区三区在线免费观看 | 一区二区三区韩国免费中文网站 | 久久久久久久久久久黄色 | 中文在线资源 | 国产综合福利在线 | 中文在线a√在线 | 久久精品96 | 久久综合九色99 | 久久亚洲福利视频 | 日产av在线播放 | 亚洲成人软件 | 青青河边草免费观看完整版高清 | 干亚洲少妇 | 九九在线视频免费观看 | 五月天天在线 | av免费在线网站 | 欧美色图狠狠干 | 日韩av电影免费在线观看 | 狠狠网亚洲精品 | 国产精品免费av | 久草视频免费在线播放 | 成人免费xxx在线观看 | 久久国产免| 亚洲一级性 | 久草com| 一级片免费观看视频 | www.亚洲视频| 色婷婷狠狠五月综合天色拍 | 激情婷婷久久 | 国产色a在线观看 | 中文字幕你懂的 | 九草视频在线 | 久久久久成人精品免费播放动漫 | 亚洲精品午夜国产va久久成人 | 亚洲午夜精品一区 | 女人18精品一区二区三区 | 天天艹天天 | 国产综合香蕉五月婷在线 | 免费a网站 | 中文在线www | 91精品久久久久久久久 | 国内精品久久影院 | 91视频a| 处女av在线 | 精品国产一区二区三区四区在线观看 | 天天综合精品 | 91精品婷婷国产综合久久蝌蚪 | 2021国产在线视频 | 久久精品久久精品久久39 | 伊人国产女 | 四虎4hu永久免费 | 国产精品视频观看 | 免费三级黄色片 | 国产不卡毛片 | 亚洲视屏一区 | 美女网站视频一区 | 美女很黄免费网站 | 96看片| 999国产在线 | 免费看亚洲毛片 | 国产96精品 | 天天干夜夜爱 | 免费www视频 | 亚洲精品99 | 国产精品18久久久久久首页狼 | 国产亚州精品视频 | 天天搞天天干天天色 | 特级a老妇做爰全过程 | 国产在线a免费观看 | 国产传媒一区在线 | 免费av网址在线观看 | 新版资源中文在线观看 | 日日夜夜免费精品视频 | 久久伊人八月婷婷综合激情 | 亚洲精品小区久久久久久 | 亚洲精品影视在线观看 | 欧美性护士 | 丁香婷婷激情国产高清秒播 | 亚洲成a人片77777潘金莲 | 久草免费福利在线观看 | 欧美成人猛片 | 国产一级片免费播放 | 毛片基地黄久久久久久天堂 | 日韩欧美视频二区 | 国产最新在线 | 99中文字幕| 免费观看一区 | 免费成人黄色 | 韩日在线一区 | 免费观看一级 | 国产精品剧情在线亚洲 | 99久久日韩精品免费热麻豆美女 | 91福利视频网站 | 久久99最新地址 | 久久高清国产 | 特级西西人体444是什么意思 | 97免费中文视频在线观看 | 天天人人综合 | 久久精品欧美一区二区三区麻豆 | 国产成人精品一区二区三区在线观看 | 中文字幕在线成人 | 懂色av懂色av粉嫩av分享吧 | 97免费在线观看视频 | 伊人久久五月天 | 久久亚洲免费视频 | 日韩免费观看一区二区 | 天天干天天摸天天操 | 久久久久久国产精品久久 | 精品国产乱码久久久久久三级人 | 免费的国产精品 | 免费精品视频在线 | 国产高清在线不卡 | 狠狠色狠狠色 | 在线直播av| 国产日韩精品在线观看 | 黄色一级在线免费观看 | av九九九| 伊人天堂av | 国产乱对白刺激视频不卡 | 丁香六月天婷婷 | 亚洲国产午夜视频 | 国产高清视频在线观看 | 久久久美女 | 精品中文字幕在线 | 日韩网 | 欧美性黑人| 日韩在线视频一区二区三区 | 丰满少妇麻豆av | 在线亚洲天堂网 | 国产精国产精品 | 成人国产一区 | 美女免费视频黄 | 欧美一区二区三区在线播放 | 久久视奸 | 久草视频免费 | 99久久精品免费看国产 | 狠狠gao| 五月天综合网站 | 中文字幕在线免费看线人 | 黄色精品一区 | 日本黄色一级电影 | 欧美极品xxx| 麻豆视频国产精品 | 国产一区影院 | 最近免费观看的电影完整版 | 亚洲国产精品va在线 | 天天搞夜夜骑 | 亚洲精品视频在线播放 | 日本少妇久久久 | 很黄很污的视频网站 | 成人在线视频网 | av中文字幕第一页 | 国产高清免费观看 | 国产亚洲精品综合一区91 | 成人在线免费视频观看 | 中文字幕一区二区三区四区视频 | 国产成人精品一区二区三区福利 | 青草视频在线 | 国产最新在线 | a√天堂中文在线 | av大片免费在线观看 | 81精品国产乱码久久久久久 | 亚洲草视频 | 91黄色在线看 | 在线观看免费视频你懂的 | 久久久久久久久久久高潮一区二区 | 韩国av免费在线 | 国产精品久久久久久久久久免费看 | 国产 欧美 日本 | 日韩首页| 激情五月婷婷丁香 | 涩涩爱夜夜爱 | 国产成人亚洲在线电影 | 欧美狠狠操 | 久草在线免费新视频 | 涩涩伊人 | 久久综合免费视频影院 | 久久精品国产一区二区三 | 国产999在线 | 波多野结衣一区二区 | 国产区免费在线 | 狠狠色网| 天天综合天天做 |