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

歡迎訪問 生活随笔!

生活随笔

當前位置: 首頁 > 编程语言 > java >内容正文

java

Java基础——类加载机制及原理

發布時間:2023/12/13 java 48 豆豆
生活随笔 收集整理的這篇文章主要介紹了 Java基础——类加载机制及原理 小編覺得挺不錯的,現在分享給大家,幫大家做個參考.

一、什么是類的加載?


? ? ? ?類的加載指的是將類的.class文件中的二進制數據讀入到內存中,將其放在運行時數據區的方法區內,然后在堆區創建一個java.lang.Class對象,用來封裝類在方法區內的數據結構。類的加載的最終產品是位于堆區中的Class對象,Class對象封裝了類在方法區內的數據結構,并且向Java程序員提供了訪問方法區內的數據結構的接口。


? ? ? ?類加載器并不需要等到某個類被“首次主動使用”時再加載它,JVM規范允許類加載器在預料某個類將要被使用時就預先加載它,如果在預先加載的過程中遇到了.class文件缺失或存在錯誤,類加載器必須在程序首次主動使用該類時才報告錯誤(LinkageError錯誤)如果這個類一直沒有被程序主動使用,那么類加載器就不會報告錯誤。

加載.class文件的方式 – 從本地系統中直接加載 – 通過網絡下載.class文件 – 從zip,jar等歸檔文件中加載.class文件 – 從專有數據庫中提取.class文件 – 將Java源文件動態編譯為.class文件
二、類的生命周期



? ? ? ?其中類加載的過程包括了加載、驗證、準備、解析、初始化五個階段。在這五個階段中,加載、驗證、準備和初始化這四個階段發生的順序是確定的,而解析階段則不一定,它在某些情況下可以在初始化階段之后開始,這是為了支持Java語言的運行時綁定(也成為動態綁定或晚期綁定)。另外注意這里的幾個階段是按順序開始,而不是按順序進行或完成,因為這些階段通常都是互相交叉地混合進行的,通常在一個階段執行的過程中調用或激活另一個階段。


1、加載:查找并加載類的二進制數據

加載時類加載過程的第一個階段,在加載階段,虛擬機需要完成以下三件事情:

? ? ? ?1)通過一個類的全限定名來獲取其定義的二進制字節流。

? ? ? ?2)將這個字節流所代表的靜態存儲結構轉化為方法區的運行時數據結構。

? ? ? ?3)在Java堆中生成一個代表這個類的java.lang.Class對象,作為對方法區中這些數據的訪問入口。

? ? ? ?相對于類加載的其他階段而言,加載階段(準確地說,是加載階段獲取類的二進制字節流的動作)是可控性最強的階段,因為開發人員既可以使用系統提供的類加載器來完成加載,也可以自定義自己的類加載器來完成加載。

? ? ? ?加載階段完成后,虛擬機外部的 二進制字節流就按照虛擬機所需的格式存儲在方法區之中,而且在Java堆中也創建一個java.lang.Class類的對象,這樣便可以通過該對象訪問方法區中的這些數據。


2、驗證確保被加載的類的正確性

? ? ? ?驗證是連接階段的第一步,這一階段的目的是為了確保Class文件的字節流中包含的信息符合當前虛擬機的要求,并且不會危害虛擬機自身的安全。驗證階段大致會完成4個階段的檢驗動作:

? ? ? ?文件格式驗證:驗證字節流是否符合Class文件格式的規范;例如:是否以0xCAFEBABE開頭、主次版本號是否在當前虛擬機的處理范圍之內、常量池中的常量是否有不被支持的類型。

? ? ? ?元數據驗證:對字節碼描述的信息進行語義分析(注意:對比javac編譯階段的語義分析),以保證其描述的信息符合Java語言規范的要求;例如:這個類是否有父類,除了java.lang.Object之外。

? ? ? ?字節碼驗證:通過數據流和控制流分析,確定程序語義是合法的、符合邏輯的。

? ? ? ?符號引用驗證:確保解析動作能正確執行。

? ? ? ?驗證階段是非常重要的,但不是必須的,它對程序運行期沒有影響,如果所引用的類經過反復驗證,那么可以考慮采用-Xverifynone參數來關閉大部分的類驗證措施,以縮短虛擬機類加載的時間。


3、準備為類的靜態變量分配內存,并將其初始化為默認值

? ? ? ?準備階段是正式為類變量分配內存并設置類變量初始值的階段,這些內存都將在方法區中分配。對于該階段有以下幾點需要注意:

? ? ? ?1)這時候進行內存分配的僅包括類變量(static),而不包括實例變量,實例變量會在對象實例化時隨著對象一塊分配在Java堆中。

? ? ? ?2)這里所設置的初始值通常情況下是數據類型默認的零值(如0、0L、null、false等),而不是被在Java代碼中被顯式地賦予的值。

? ? ? ?假設一個類變量的定義為:public static int value = 3;

? ? ? ?那么變量value在準備階段過后的初始值為0,而不是3,因為這時候尚未開始執行任何Java方法,而把value賦值為3的putstatic指令是在程序編譯后,存放于類構造器<clinit>()方法之中的,所以把value賦值為3的動作將在初始化階段才會執行。


? ? ? ?這里還需要注意如下幾點:

  • 對基本數據類型來說,對于類變量(static)和全局變量,如果不顯式地對其賦值而直接使用,則系統會為其賦予默認的零值,而對于局部變量來說,在使用前必須顯式地為其賦值,否則編譯時不通過。
  • 對于同時被static和final修飾的常量,必須在聲明的時候就為其顯式地賦值,否則編譯時不通過;而只被final修飾的常量則既可以在聲明時顯式地為其賦值,也可以在類初始化時顯式地為其賦值,總之,在使用前必須為其顯式地賦值,系統不會為其賦予默認零值。
  • 對于引用數據類型reference來說,如數組引用、對象引用等,如果沒有對其進行顯式地賦值而直接使用,系統都會為其賦予默認的零值,即null。
  • 如果在數組初始化時沒有對數組中的各元素賦值,那么其中的元素將根據對應的數據類型而被賦予默認的零值。

? ? ? ?3)如果類字段的字段屬性表中存在ConstantValue屬性,即同時被final和static修飾,那么在準備階段變量value就會被初始化為ConstValue屬性所指定的值。

? ? ? ?假設上面的類變量value被定義為: public static final int value = 3;

? ? ? ?編譯時Javac將會為value生成ConstantValue屬性,在準備階段虛擬機就會根據ConstantValue的設置將value賦值為3。回憶上一篇博文中對象被動引用的第2個例子,便是這種情況。我們可以理解為static final常量在編譯期就將其結果放入了調用它的類的常量池中。


4、解析把類中的符號引用轉換為直接引用

? ? ? ?解析階段是虛擬機將常量池內的符號引用替換為直接引用的過程,解析動作主要針對類或接口、字段、類方法、接口方法、方法類型、方法句柄和調用點限定符7類符號引用進行。符號引用就是一組符號來描述目標,可以是任何字面量。

? ? ? ?直接引用就是直接指向目標的指針、相對偏移量或一個間接定位到目標的句柄。


5、初始化

? ? ? ?初始化,為類的靜態變量賦予正確的初始值,JVM負責對類進行初始化,主要對類變量進行初始化。在Java中對類變量進行初始值設定有兩種方式:

? ? ? ?①聲明類變量是指定初始化值

? ? ? ?②使用靜態代碼塊為類變量指定初始


?JVM初始化步驟

? ? ? ?1)假如這個類還沒有被加載和連接,則程序先加載并連接該類

? ? ? ?2)假如該類的直接父類還沒有被初始化,則先初始化其直接父類

? ? ? ?3)假如類中有初始化語句,則系統依次執行這些初始化語句


類初始化時機

只有當對類的主動使用的時候才會導致類的初始化,類的主動使用包括以下六種:

? ? ? ?– 創建類的實例,也就是new的方式

? ? ? ?– 訪問某個類或接口的靜態變量,或者對該靜態變量賦值

? ? ? ?– 調用類的靜態方法

? ? ? ?– 反射(如Class.forName(“com.shengsiyuan.Test”))

? ? ? ?– 初始化某個類的子類,則其父類也會被初始化

? ? ? ?– Java虛擬機啟動時被標明為啟動類的類(Java Test),直接使用java.exe命令來運行某個主類

?

結束生命周期

在如下幾種情況下,Java虛擬機將結束生命周期:

? ? ? ?– 執行了System.exit()方法

? ? ? ?– 程序正常執行結束

? ? ? ?– 程序在執行過程中遇到了異常或錯誤而異常終止

? ? ? ?– 由于操作系統出現錯誤而導致Java虛擬機進程終止


三、Java虛擬機類加載器


1、JVM三種預定義類型類加載器

我們首先看一下JVM預定義的三種類型類加載器,當一個?JVM啟動的時候,Java缺省開始使用如下三種類型類裝入器:

? ? ? ?啟動(Bootstrap)類加載器引導類裝入器是用本地代碼實現的類裝入器,它負責將%JAVA_HOME%\lib路徑下或-Xbootclasspath參數指定路徑下的、能被虛擬機識別的類庫(僅按照文件名識別,如:rt.jar,名字不符合的類庫不會被加載)加載至虛擬機內存中由于引導類加載器涉及到虛擬機本地實現細節,開發者無法直接獲取到啟動類加載器的引用,所以不允許直接通過引用進行操作。

? ? ? ?擴展(Extension)類加載器擴展類加載器是由SunExtClassLoadersun.misc.Launcher$ExtClassLoader實現的。它負責將java.ext.dirs參數(默認值是%JAVA_HOME%\jre\lib\ext,可由VM參數-Djava.ext.dirs指定)指定路徑中的所有類庫加載到內存中。開發者可以直接使用標準擴展類加載器。

? ? ? ?系統(System)類加載器系統類加載器是由?SunAppClassLoadersun.misc.Launcher$AppClassLoader)實現的,由于這個類加載器是ClassLoader中的getSystemClassLoader()方法的返回值,也稱為系統類加載器。它負責將系統類路徑java -classpath或-Djava.class.path變量所指的目錄下的類庫加載到內存中。開發者可以直接使用系統類加載器。

? ? ? ?注意:上述三個JDK提供的類加載器雖然父子類加載器關系,但是沒有使用繼承,而是使用了組合關系


2、類加載雙親委派機制介紹和分析

從JDK1.2開始,java虛擬機規范推薦開發者使用雙親委派模式(ParentsDelegation Model)進行類加載,其加載過程如下:

  • 如果一個類加載器收到了類加載請求,它首先不會自己去嘗試加載這個類,而是把類加載請求委派給父類加載器去完成,每一個層次的加載器都是如此
  • 因此所有的類加載請求都會傳給頂層的啟動類加載器,只有當父加載器反饋自己無法完成該加載請求(該加載器的搜索范圍中沒有找到對應的類)時,子加載器才會嘗試自己去加載。
  • 如果連最初發起類加載請求的類加載器也無法完成加載請求時,將會拋出ClassNotFoundException,而不再調用其子類加載器去進行類加載。
  • ? ? ? ?關于虛擬機默認的雙親委派機制,我們可以從系統類加載器和擴展類加載器為例作簡單分析。


    圖一 標準擴展類加載器繼承層次圖


    圖二系統類加載器繼承層次圖

    ? ? ? ?通過圖一和圖二我們可以看出,類加載器均是繼承自java.lang.ClassLoader抽象類。我們下面我們就看簡要介紹一下java.lang.ClassLoader中幾個最重要的方法:

    //加載指定名稱(包括包名)的二進制類型,供用戶調用的接口 public Class<?> loadClass(String name) throws ClassNotFoundException{ … } //加載指定名稱(包括包名)的二進制類型,同時指定是否解析(但是這里的resolve參數不一定真正能達到解析的效果),供繼承用 protected synchronized Class<?> loadClass(String name, boolean resolve) throws ClassNotFoundException{ … } //findClass方法一般被loadClass方法調用去加載指定名稱類,供繼承用 protected Class<?> findClass(String name) throws ClassNotFoundException { … } //定義類型,一般在findClass方法中讀取到對應字節碼后調用,可以看出不可繼承 //(說明:JVM已經實現了對應的具體功能,解析對應的字節碼,產生對應的內部數據結構放置到方法區,所以無需覆寫,直接調用就可以了) protected final Class<?> defineClass(String name, byte[] b, int off, int len) throws ClassFormatError{ … } ? ? ? ?通過進一步分析標準擴展類加載器( sun.misc.Launcher$ExtClassLoader )和系統類加載器( sun.misc.Launcher$AppClassLoader )的代碼以及其公共父類(java.net.URLClassLoader java.security.SecureClassLoader )的代碼可以看出,都沒有覆寫 java.lang.ClassLoader中默認的加載委派規則---loadClass )方法既然這樣,我們就可以通過分析java.lang.ClassLoader 中的loadClass String name)方法的代碼就可以分析出虛擬機默認采用的雙親委派機制到底是什么模樣:
    public Class<?> loadClass(String name) throws ClassNotFoundException { return loadClass(name, false); } protected synchronized Class<?> loadClass(String name, boolean resolve) throws ClassNotFoundException { // 首先判斷該類型是否已經被加載 Class c = findLoadedClass(name); if (c == null) { //如果沒有被加載,就委托給父類加載或者委派給啟動類加載器加載 try { if (parent != null) { //如果存在父類加載器,就委派給父類加載器加載 c = parent.loadClass(name, false); } else { //如果不存在父類加載器,就檢查是否是由啟動類加載器加載的類, //通過調用本地方法native findBootstrapClass0(String name) c = findBootstrapClass0(name); } } catch (ClassNotFoundException e) { // 如果父類加載器和啟動類加載器都不能完成加載任務,才調用自身的加載功能 c = findClass(name); } } if (resolve) { resolveClass(c); } return c; } ? ? ? ?通過上面的代碼分析,我們可以對JVM 采用的雙親委派類加載機制有了更感性的認識,下面我們就接著分析一下啟動類加載器、標準擴展類加載器和系統類加載器三者之間的關系。可能大家已經從各種資料上面看到了如下類似的一幅圖片:

    圖三 類加載器默認委派關系圖
    ? ? ? ?上面圖片給人的直觀印象是系統類加載器的父類加載器是標準擴展類加載器,標準擴展類加載器的父類加載器是啟動類加載器,下面我們就用代碼具體測試一下:

    public class LoaderTest { public static void main(String[] args) { try { System.out.println(ClassLoader.getSystemClassLoader()); System.out.println(ClassLoader.getSystemClassLoader().getParent()); System.out.println(ClassLoader.getSystemClassLoader().getParent().getParent()); } catch (Exception e) { e.printStackTrace(); } } } ? ? ? ?說明:通過java.lang.ClassLoader.getSystemClassLoader()可以直接獲取到系統類加載器。
    代碼輸出如下:
    sun.misc.Launcher$AppClassLoader@6d06d69c sun.misc.Launcher$ExtClassLoader@70dea4e null ? ? ? ?通過以上的代碼輸出,我們可以判定系統類加載器的父加載器是標準擴展類加載器,但是我們試圖獲取標準擴展類加載器的父類加載器時確得到了 null,就是說標準擴展類加載器本身強制設定父類加載器為null 我們還是借助于代碼分析一下。

    我們首先看一下java.lang.ClassLoader抽象類中默認實現的兩個構造函數:

    protected ClassLoader() { SecurityManager security = System.getSecurityManager(); if (security != null) { security.checkCreateClassLoader(); } //默認將父類加載器設置為系統類加載器,getSystemClassLoader()獲取系統類加載器 this.parent = getSystemClassLoader(); initialized = true; } protected ClassLoader(ClassLoader parent) { SecurityManager security = System.getSecurityManager(); if (security != null) { security.checkCreateClassLoader(); } //強制設置父類加載器 this.parent = parent; initialized = true; } 我們再看一下 ClassLoader 抽象類中 parent 成員的聲明:
    // The parent class loader for delegation private ClassLoader parent;? ? ? ?聲明為私有變量的同時并沒有對外提供可供派生類訪問的public或者protected設置器接口(對應的setter方法),結合前面的測試代碼的輸出,我們可以推斷出:

    • 系統類加載器(AppClassLoader)調用ClassLoader(ClassLoader parent)構造函數將父類加載器設置為標準擴展類加載器(ExtClassLoader)。(因為如果不強制設置,默認會通過調用getSystemClassLoader()方法獲取并設置成系統類加載器,這顯然和測試輸出結果不符。)
    • 擴展類加載器(ExtClassLoader)調用ClassLoader(ClassLoader parent)構造函數將父類加載器設置為null。(因為如果不強制設置,默認會通過調用getSystemClassLoader()方法獲取并設置成系統類加載器,這顯然和測試輸出結果不符。)

    ? ? ? ?現在我們可能會有這樣的疑問:擴展類加載器(ExtClassLoader)的父類加載器被強制設置為null了,那么擴展類加載器為什么還能將加載任務委派給啟動類加載器呢?

    圖四 標準擴展類加載器和系統類加載器成員大綱視圖 圖五?擴展類加載器和系統類加載器公共父類成員大綱視圖

    ? ? ? ?通過圖四和圖五可以看出,標準擴展類加載器和系統類加載器及其父類(java.net.URLClassLoader和java.security.SecureClassLoader)都沒有覆寫java.lang.ClassLoader中默認的加載委派規則---loadClass(…)方法有關java.lang.ClassLoader中默認的加載委派規則前面已經分析過,如果父加載器為null,則會調用本地方法進行啟動類加載嘗試。所以,圖三中,啟動類加載器、標準擴展類加載器和系統類加載器之間的委派關系事實上是仍就成立的。(在后面的用戶自定義類加載器部分,還會做更深入的分析)。

    ? ? ? ?雙親委派模式的類加載機制的優點:java類它的類加載器一起具備了一種帶優先級的層次關系,越是基礎的類,越是被上層的類加載器進行加載,保證了java程序的穩定運行。


    3、類加載雙親委派示例

    ? ? ? ?以上已經簡要介紹了虛擬機默認使用的啟動類加載器、標準擴展類加載器和系統類加載器,并以三者為例結合JDK代碼對JVM默認使用的雙親委派類加載機制做了分析。下面我們就來看一個綜合的例子。首先在IDE中建立一個簡單的java應用工程,然后寫一個簡單的JavaBean如下:

    package classloader.test.bean; public class TestBean { public TestBean() { } } 在現有當前工程中另外建立一測試類(ClassLoaderTest.java)內容如下:

    測試一:

    package classloader.test.bean; public class ClassLoaderTest { public static void main(String[] args) { try { //查看當前系統類路徑中包含的路徑條目 System.out.println(System.getProperty("java.class.path")); //調用加載當前類的類加載器(這里即為系統類加載器)加載TestBean Class typeLoaded = Class.forName("classloader.test.bean.TestBean"); //查看被加載的TestBean類型是被那個類加載器加載的 System.out.println(typeLoaded.getClassLoader()); } catch (Exception e) { e.printStackTrace(); } } } 輸出結果:

    C:\Users\Administrator\Documents\NetBeansProjects\ClassLoaderTest\build\classes sun.misc.Launcher$AppClassLoader@73d16e93 說明:當前類路徑默認的含有的一個條目就是工程的輸出目錄。

    測試二:

    ? ? ? ?將當前工程輸出目錄下的TestBean.class打包進test.jar剪貼<Java_Runtime_Home>/lib/ext目錄下(現在工程輸出目錄下和JRE擴展目錄下都有待加載類型的class文件)。再運行測試一測試代碼,結果如下:

    C:\Users\Administrator\Documents\NetBeansProjects\ClassLoaderTest\build\classes sun.misc.Launcher$ExtClassLoader@15db9742

    ? ? ? ?對比測試一和測試二,我們明顯可以驗證前面說的雙親委派機制,系統類加載器在接到加載classloader.test.bean.TestBean類型的請求時,首先將請求委派給父類加載器(標準擴展類加載器),標準擴展類加載器搶先完成了加載請求。

    測試三:

    ? ? ? ?將test.jar拷貝一份到<Java_Runtime_Home>/lib下,運行測試代碼,輸出如下:
    C:\Users\Administrator\Documents\NetBeansProjects\ClassLoaderTest\build\classes sun.misc.Launcher$ExtClassLoader@15db9742? ? ? ?測試三和測試二輸出結果一致。那就是說,放置到<Java_Runtime_Home>/lib目錄下的TestBean對應的class字節碼并沒有被加載,這其實和前面講的雙親委派機制并不矛盾。 虛擬機出于安全等因素考慮,不會加載<Java_Runtime_Home>/lib存在的陌生類,開發者通過將要加載的非JDK自身的類放置到此目錄下期待啟動類加載器加載是不可能的。做個進一步驗證,刪除<Java_Runtime_Home>/lib/ext目錄下和工程輸出目錄下的TestBean對應的class文件,然后再運行測試代碼,則將會有ClassNotFoundException異常拋出。有關這個問題,大家可以在java.lang.ClassLoader中的loadClass(String name, boolean resolve)方法中設置相應斷點運行測試三進行調試,會發現findBootstrapClass0()會拋出異常,然后在下面的findClass方法中被加載,當前運行的類加載器正是擴展類加載器(sun.misc.Launcher$ExtClassLoader),這一點可以通過JDT中變量視圖查看驗證。

    四、Java程序動態擴展方式

    ? ? ? ?Java的連接模型允許用戶運行時擴展引用程序,既可以通過當前虛擬機中預定義的加載器加載編譯時已知的類或者接口,又允許用戶自行定義類裝載器,在運行時動態擴展用戶的程序。通過用戶自定義的類裝載器,你的程序可以裝載在編譯時并不知道或者尚未存在的類或者接口,并動態連接它們并進行有選擇的解析。

    運行時動態擴展java應用程序有如下兩個途徑:


    1、調用java.lang.Class.forName(…)加載類

    ? ? ? ?這個方法其實在前面已經討論過,在后面的問題2解答中說明了該方法調用會觸發哪個類加載器開始加載任務。這里需要說明的是多參數版本的forName(…)方法:

    public static Class<?> forName(String name, boolean initialize, ClassLoader loader) throws ClassNotFoundException? ? ? ?這里的initialize參數是很重要的。它表示在加載同時是否完成初始化的工作(說明:單參數版本的forName方法默認是完成初始化的)。 有些場景下需要將initialize設置為true來強制加載同時完成初始化。例如典型的就是利用DriverManager進行JDBC驅動程序類注冊的問題因為每一個JDBC驅動程序類的靜態初始化方法都用DriverManager注冊驅動程序,這樣才能被應用程序使用。這就要求驅動程序類必須被初始化,而不單單被加載。Class.forName的一個很常見的用法就是在加載數據庫驅動的時候。如 Class.forName("org.apache.derby.jdbc.EmbeddedDriver").newInstance()用來加載 Apache Derby 數據庫的驅動。


    2、用戶自定義類加載器

    ? ? ? ?通過前面的分析,我們可以看出,除了和本地實現密切相關的啟動類加載器之外,包括標準擴展類加載器和系統類加載器在內的所有其他類加載器我們都可以當做自定義類加載器來對待,唯一區別是是否被虛擬機默認使用。前面的內容中已經對java.lang.ClassLoader抽象類中的幾個重要的方法做了介紹,這里就簡要敘述一下一般用戶自定義類加載器的工作流程吧(可以結合后面問題解答一起看):

  • 首先檢查請求的類型是否已經被這個類裝載器裝載到命名空間中了,如果已經裝載,直接返回;否則轉入步驟2;
  • 委派類加載請求給父類加載器(更準確的說應該是雙親類加載器,真實虛擬機中各種類加載器最終會呈現樹狀結構),如果父類加載器能夠完成,則返回父類加載器加載的Class實例;否則轉入步驟3;
  • 調用本類加載器的findClass(…)方法,試圖獲取對應的字節碼,如果獲取的到,則調用defineClass(…)導入類型到方法區;如果獲取不到對應的字節碼或者其他原因失敗,返回異常給loadClass(…), loadClass(…)轉而拋異常,終止加載過程(注意:這里的異常種類不止一種)。
  • ? ? ? ?說明:這里說的自定義類加載器是指JDK 1.2以后版本的寫法,即不覆寫改變java.lang.loadClass(…)已有委派邏輯情況下。

    整個加載類的過程如下圖:


    圖六 自定義類加載器加載類的過程


    五、常見問題


    1、由不同的類加載器加載的指定類還是相同的類型嗎?

    ? ? ? ?在Java中,一個類用其完全匹配類名(fully qualified class name)作為標識,這里指的完全匹配類名包括包名和類名。但在JVM中一個類用其全名和一個加載類ClassLoader的實例作為唯一標識,不同類加載器加載的類將被置于不同的命名空間我們可以用兩個自定義類加載器去加載某自定義類型(注意不要將自定義類型的字節碼放置到系統路徑或者擴展路徑中,否則會被系統類加載器或擴展類加載器搶先加載),然后用獲取到的兩個Class實例進行java.lang.Object.equals(…)判斷,將會得到不相等的結果。這個大家可以寫兩個自定義的類加載器去加載相同的自定義類型,然后做個判斷;同時,可以測試加載java.*類型,然后再對比測試一下測試結果。


    2、在代碼中直接調用Class.forName(String name)方法,到底會觸發那個類加載器進行類加載行為?

    Class.forName(String name)默認會使用調用類的類加載器來進行類加載。我們直接來分析一下對應的jdk的代碼:

    //java.lang.Class.java publicstatic Class<?> forName(String className) throws ClassNotFoundException { return forName0(className, true, ClassLoader.getCallerClassLoader()); } //java.lang.ClassLoader.java // Returns the invoker's class loader, or null if none. static ClassLoader getCallerClassLoader() { // 獲取調用類(caller)的類型 Class caller = Reflection.getCallerClass(3); // This can be null if the VM is requesting it if (caller == null) { return null; } // 調用java.lang.Class中本地方法獲取加載該調用類(caller)的ClassLoader return caller.getClassLoader0(); } //java.lang.Class.java //虛擬機本地實現,獲取當前類的類加載器,前面介紹的Class的getClassLoader()也使用此方法 native ClassLoader getClassLoader0();


    3、 在編寫自定義類加載器時,如果沒有設定父加載器,那么父加載器是誰?
    ? ? ? ?前面講過,在不指定父類加載器的情況下,默認采用系統類加載器可能有人覺得不明白,現在我們來看一下JDK對應的代碼實現。眾所周知,我們編寫自定義的類加載器直接或者間接繼承自java.lang.ClassLoader抽象類,對應的無參默認構造函數實現如下:
    //摘自java.lang.ClassLoader.java protected ClassLoader() { SecurityManager security = System.getSecurityManager(); if (security != null) { security.checkCreateClassLoader(); } this.parent = getSystemClassLoader(); initialized = true; } 我們再來看一下對應的getSystemClassLoader()方法的實現:
    private static synchronized void initSystemClassLoader() { //... sun.misc.Launcher l = sun.misc.Launcher.getLauncher(); scl = l.getClassLoader(); //... } 我們可以寫簡單的測試代碼來測試一下:
    System.out.println(sun.misc.Launcher.getLauncher().getClassLoader()); 輸出結果:

    sun.misc.Launcher$AppClassLoader@73d16e93 ? ? ? ?所以,我們現在可以相信當自定義類加載器沒有指定父類加載器的情況下,默認的父類加載器即為系統類加載器。同時,我們可以得出如下結論: 即使用戶自定義類加載器不指定父類加載器,那么,同樣可以加載如下三個地方的類:

  • <Java_Runtime_Home>/lib下的類;
  • < Java_Runtime_Home >/lib/ext下或者由系統變量java.ext.dir指定位置中的類;
  • 當前工程類路徑下或者由系統變量java.class.path指定位置中的類。

  • 4、在編寫自定義類加載器時,如果將父類加載器強制設置為null,那么會有什么影響?如果自定義的類加載器不能加載指定類,就肯定會加載失敗嗎?

    ? ? ? ?JVM規范中規定如果用戶自定義的類加載器將父類加載器強制設置為null,那么會自動將啟動類加載器設置為當前用戶自定義類加載器的父類加載器(這個問題前面已經分析過了)。同時,我們可以得出如下結論:

    ? ? ? ?即使用戶自定義類加載器不指定父類加載器,那么,同樣可以加載到<Java_Runtime_Home>/lib下的類,但此時就不能夠加載<Java_Runtime_Home>/lib/ext目錄下的類了。

    ? ? ? ?說明:問題3和問題4的推斷結論是基于用戶自定義的類加載器本身延續了java.lang.ClassLoader.loadClass(…)默認委派邏輯,如果用戶對這一默認委派邏輯進行了改變,以上推斷結論就不一定成立了,詳見問題5。


    5、編寫自定義類加載器時,一般有哪些注意點?

    1)一般盡量不要覆寫已有的loadClass(...)方法中的委派邏輯

    ? ? ? ?一般在JDK 1.2之前的版本才這樣做,而且事實證明,這樣做極有可能引起系統默認的類加載器不能正常工作。在JVM規范和JDK文檔中(1.2或者以后版本中),都沒有建議用戶覆寫loadClass(…)方法,相比而言,明確提示開發者在開發自定義的類加載器時覆寫findClass(…)邏輯。舉一個例子來驗證該問題:

    //用戶自定義類加載器WrongClassLoader.Java(覆寫loadClass邏輯) public class WrongClassLoader extends ClassLoader { public Class<?> loadClass(String name) throws ClassNotFoundException { return this.findClass(name); } protected Class<?> findClass(String name) throws ClassNotFoundException { // 假設此處只是到工程以外的特定目錄D:\library下去加載類 // 具體實現代碼省略 } } ? ? ? ?通過前面的分析我們已經知道,這個自定義類加載器WrongClassLoader的默認類加載器是系統類加載器,但是現在問題4種的結論就不成立了。大家可以簡單測試一下,現在<Java_Runtime_Home>/lib、< Java_Runtime_Home >/lib/ext和工程類路徑上的類都加載不上了。
    //問題5測試代碼一 public class WrongClassLoaderTest { publicstaticvoid main(String[] args) { try { WrongClassLoader loader = new WrongClassLoader(); Class classLoaded = loader.loadClass("beans.Account"); System.out.println(classLoaded.getName()); System.out.println(classLoaded.getClassLoader()); } catch (Exception e) { e.printStackTrace(); } } } 這里D:"classes"beans"Account.class是物理存在的。輸出結果:
    java.io.FileNotFoundException: D:"classes"java"lang"Object.class (系統找不到指定的路徑。) at java.io.FileInputStream.open(Native Method) at java.io.FileInputStream.<init>(FileInputStream.java:106) at WrongClassLoader.findClass(WrongClassLoader.java:40) at WrongClassLoader.loadClass(WrongClassLoader.java:29) at java.lang.ClassLoader.loadClassInternal(ClassLoader.java:319) at java.lang.ClassLoader.defineClass1(Native Method) at java.lang.ClassLoader.defineClass(ClassLoader.java:620) at java.lang.ClassLoader.defineClass(ClassLoader.java:400) at WrongClassLoader.findClass(WrongClassLoader.java:43) at WrongClassLoader.loadClass(WrongClassLoader.java:29) at WrongClassLoaderTest.main(WrongClassLoaderTest.java:27) Exception in thread "main" java.lang.NoClassDefFoundError: java/lang/Object at java.lang.ClassLoader.defineClass1(Native Method) at java.lang.ClassLoader.defineClass(ClassLoader.java:620) at java.lang.ClassLoader.defineClass(ClassLoader.java:400) at WrongClassLoader.findClass(WrongClassLoader.java:43) at WrongClassLoader.loadClass(WrongClassLoader.java:29) at WrongClassLoaderTest.main(WrongClassLoaderTest.java:27) ? ? ? ?這說明,連要加載的類型的超類型java.lang.Object都加載不到了。這里列舉的由于覆寫loadClass()引起的邏輯錯誤明顯是比較簡單的,實際引起的邏輯錯誤可能復雜的多。
    //問題5測試二 //用戶自定義類加載器WrongClassLoader.Java(不覆寫loadClass邏輯) public class WrongClassLoader extends ClassLoader { protected Class<?> findClass(String name) throws ClassNotFoundException { //假設此處只是到工程以外的特定目錄D:\library下去加載類 //具體實現代碼省略 } } 將自定義類加載器代碼WrongClassLoader.Java做以上修改后,再運行測試代碼,輸出結果如下:
    beans.Account WrongClassLoader@1c78e57 2)正確設置父類加載器

    ? ? ? ?通過上面問題4和問題5的分析我們應該已經理解,個人覺得這是自定義用戶類加載器時最重要的一點,但常常被忽略或者輕易帶過。有了前面JDK代碼的分析作為基礎,我想現在大家都可以隨便舉出例子了。

    3)保證findClass(String name)方法的邏輯正確性

    ? ? ? ?事先盡量準確理解待定義的類加載器要完成的加載任務,確保最大程度上能夠獲取到對應的字節碼內容。

    6、如何在運行時判斷系統類加載器能加載哪些路徑下的類?

    ? ? ? ?一是可以直接調用ClassLoader.getSystemClassLoader()或者其他方式獲取到系統類加載器(系統類加載器和擴展類加載器本身都派生自URLClassLoader),調用URLClassLoader中的getURLs()方法可以獲取到。

    ? ? ? ?二是可以直接通過獲取系統屬性java.class.path來查看當前類路徑上的條目信息 :System.getProperty("java.class.path")。


    7、如何在運行時判斷標準擴展類加載器能加載哪些路徑下的類?

    方法一:

    import java.net.URL; import java.net.URLClassLoader; public class ClassLoaderTest { /** * @param args the command line arguments */ public static void main(String[] args) { try { URL[] extURLs = ((URLClassLoader) ClassLoader.getSystemClassLoader().getParent()).getURLs(); for (int i = 0; i < extURLs.length; i++) { System.out.println(extURLs[i]); } } catch (Exception e) { //… } } } 輸出結果:

    file:/C:/Program%20Files/Java/jdk1.8.0_05/jre/lib/ext/access-bridge-64.jar file:/C:/Program%20Files/Java/jdk1.8.0_05/jre/lib/ext/cldrdata.jar file:/C:/Program%20Files/Java/jdk1.8.0_05/jre/lib/ext/dnsns.jar file:/C:/Program%20Files/Java/jdk1.8.0_05/jre/lib/ext/jaccess.jar file:/C:/Program%20Files/Java/jdk1.8.0_05/jre/lib/ext/jfxrt.jar file:/C:/Program%20Files/Java/jdk1.8.0_05/jre/lib/ext/localedata.jar file:/C:/Program%20Files/Java/jdk1.8.0_05/jre/lib/ext/nashorn.jar file:/C:/Program%20Files/Java/jdk1.8.0_05/jre/lib/ext/sunec.jar file:/C:/Program%20Files/Java/jdk1.8.0_05/jre/lib/ext/sunjce_provider.jar file:/C:/Program%20Files/Java/jdk1.8.0_05/jre/lib/ext/sunmscapi.jar file:/C:/Program%20Files/Java/jdk1.8.0_05/jre/lib/ext/sunpkcs11.jar file:/C:/Program%20Files/Java/jdk1.8.0_05/jre/lib/ext/zipfs.jar
    六、開發自己的類加載器


    ? ? ? ?在前面介紹類加載器的代理委派模式的時候,提到過類加載器會首先代理給其它類加載器來嘗試加載某個類。這就意味著真正完成類的加載工作的類加載器和啟動這個加載過程的類加載器,有可能不是同一個。真正完成類的加載工作是通過調用defineClass來實現的;而啟動類的加載過程是通過調用loadClass來實現的。前者稱為一個類的定義加載器(defining loader),后者稱為初始加載器(initiating loader)。在Java虛擬機判斷兩個類是否相同的時候,使用的是類的定義加載器。也就是說,哪個類加載器啟動類的加載過程并不重要,重要的是最終定義這個類的加載器。兩種類加載器的關聯之處在于:一個類的定義加載器是它引用的其它類的初始加載器。如類 com.example.Outer引用了類 com.example.Inner,則由類 com.example.Outer的定義加載器負責啟動類 com.example.Inner的加載過程。

    ? ? ? ?方法 loadClass()拋出的是 java.lang.ClassNotFoundException異常;方法 defineClass()拋出的是 java.lang.NoClassDefFoundError異常。

    ? ? ? ?類加載器在成功加載某個類之后,會把得到的 java.lang.Class類的實例緩存起來。下次再請求加載該類的時候,類加載器會直接使用緩存的類的實例,而不會嘗試再次加載。也就是說,對于一個類加載器實例來說,相同全名的類只加載一次,即 loadClass方法不會被重復調用

    ? ? ? ?在絕大多數情況下,系統默認提供的類加載器實現已經可以滿足需求。但是在某些情況下,您還是需要為應用開發出自己的類加載器。比如您的應用通過網絡來傳輸Java類的字節代碼,為了保證安全性,這些字節代碼經過了加密處理。這個時候您就需要自己的類加載器來從某個網絡地址上讀取加密后的字節代碼,接著進行解密和驗證,最后定義出要在Java虛擬機中運行的類來。下面將通過兩個具體的實例來說明類加載器的開發。

    1、文件系統類加載器

    第一個類加載器用來加載存儲在文件系統上的Java字節代碼。完整的實現如下所示。

    package classloader; import java.io.ByteArrayOutputStream; import java.io.File; import java.io.FileInputStream; import java.io.IOException; import java.io.InputStream; // 文件系統類加載器 public class FileSystemClassLoader extends ClassLoader { private String rootDir; public FileSystemClassLoader(String rootDir) { this.rootDir = rootDir; } // 獲取類的字節碼 @Override protected Class<?> findClass(String name) throws ClassNotFoundException { byte[] classData = getClassData(name); // 獲取類的字節數組 if (classData == null) { throw new ClassNotFoundException(); } else { return defineClass(name, classData, 0, classData.length); } } private byte[] getClassData(String className) { // 讀取類文件的字節 String path = classNameToPath(className); try { InputStream ins = new FileInputStream(path); ByteArrayOutputStream baos = new ByteArrayOutputStream(); int bufferSize = 4096; byte[] buffer = new byte[bufferSize]; int bytesNumRead = 0; // 讀取類文件的字節碼 while ((bytesNumRead = ins.read(buffer)) != -1) { baos.write(buffer, 0, bytesNumRead); } return baos.toByteArray(); } catch (IOException e) { e.printStackTrace(); } return null; } private String classNameToPath(String className) { // 得到類文件的完全路徑 return rootDir + File.separatorChar + className.replace('.', File.separatorChar) + ".class"; } } ? ? ? ?如上所示,類 FileSystemClassLoader繼承自類java.lang.ClassLoader。在java.lang.ClassLoader類的常用方法中,一般來說,自己開發的類加載器只需要覆寫 findClass(String name)方法即可。java.lang.ClassLoader類的方法loadClass()封裝了前面提到的代理模式的實現。該方法會首先調用findLoadedClass()方法來檢查該類是否已經被加載過;如果沒有加載過的話,會調用父類加載器的loadClass()方法來嘗試加載該類;如果父類加載器無法加載該類的話,就調用findClass()方法來查找該類。 因此,為了保證類加載器都正確實現代理模式,在開發自己的類加載器時,最好不要覆寫 loadClass()方法,而是覆寫 findClass()方法。

    ? ? ? ?類 FileSystemClassLoader的 findClass()方法首先根據類的全名在硬盤上查找類的字節代碼文件(.class 文件),然后讀取該文件內容,最后通過defineClass()方法來把這些字節代碼轉換成 java.lang.Class類的實例。

    加載本地文件系統上的類,示例如下:

    package com.example; public class Sample { private Sample instance; public void setSample(Object instance) { System.out.println(instance.toString()); this.instance = (Sample) instance; } } package classloader; import java.lang.reflect.Method; public class ClassIdentity { public static void main(String[] args) { new ClassIdentity().testClassIdentity(); } public void testClassIdentity() { String classDataRootPath = "C:\\Users\\JackZhou\\Documents\\NetBeansProjects\\classloader\\build\\classes"; FileSystemClassLoader fscl1 = new FileSystemClassLoader(classDataRootPath); FileSystemClassLoader fscl2 = new FileSystemClassLoader(classDataRootPath); String className = "com.example.Sample"; try { Class<?> class1 = fscl1.loadClass(className); // 加載Sample類 Object obj1 = class1.newInstance(); // 創建對象 Class<?> class2 = fscl2.loadClass(className); Object obj2 = class2.newInstance(); Method setSampleMethod = class1.getMethod("setSample", java.lang.Object.class); setSampleMethod.invoke(obj1, obj2); } catch (Exception e) { e.printStackTrace(); } } } 輸出結果:
    com.example.Sample@7852e922


    2、網絡類加載器
    ? ? ? ?下面將通過一個網絡類加載器來說明如何通過類加載器來實現組件的動態更新。即基本的場景是:Java 字節代碼(.class)文件存放在服務器上,客戶端通過網絡的方式獲取字節代碼并執行。當有版本更新的時候,只需要替換掉服務器上保存的文件即可。通過類加載器可以比較簡單的實現這種需求。

    ? ? ? ?類 NetworkClassLoader負責通過網絡下載Java類字節代碼并定義出Java類。它的實現與FileSystemClassLoader類似。

    package classloader; import java.io.ByteArrayOutputStream; import java.io.InputStream; import java.net.URL; public class NetworkClassLoader extends ClassLoader { private String rootUrl; public NetworkClassLoader(String rootUrl) { // 指定URL this.rootUrl = rootUrl; } // 獲取類的字節碼 @Override protected Class<?> findClass(String name) throws ClassNotFoundException { byte[] classData = getClassData(name); if (classData == null) { throw new ClassNotFoundException(); } else { return defineClass(name, classData, 0, classData.length); } } private byte[] getClassData(String className) { // 從網絡上讀取的類的字節 String path = classNameToPath(className); try { URL url = new URL(path); InputStream ins = url.openStream(); ByteArrayOutputStream baos = new ByteArrayOutputStream(); int bufferSize = 4096; byte[] buffer = new byte[bufferSize]; int bytesNumRead = 0; // 讀取類文件的字節 while ((bytesNumRead = ins.read(buffer)) != -1) { baos.write(buffer, 0, bytesNumRead); } return baos.toByteArray(); } catch (Exception e) { e.printStackTrace(); } return null; } private String classNameToPath(String className) { // 得到類文件的URL return rootUrl + "/" + className.replace('.', '/') + ".class"; } } ? ? ? ?在通過NetworkClassLoader加載了某個版本的類之后,一般有兩種做法來使用它。第一種做法是使用Java反射API。另外一種做法是使用接口。 需要注意的是,并不能直接在客戶端代碼中引用從服務器上下載的類,因為客戶端代碼的類加載器找不到這些類。使用Java反射API可以直接調用Java類的方法。而使用接口的做法則是把接口的類放在客戶端中,從服務器上加載實現此接口的不同版本的類。在客戶端通過相同的接口來使用這些實現類。我們使用接口的方式。示例如下:

    客戶端接口:

    package classloader; public interface Versioned { String getVersion(); } package classloader; public interface ICalculator extends Versioned { String calculate(String expression); }

    網絡上的不同版本的類:

    package com.example; import classloader.ICalculator; public class CalculatorBasic implements ICalculator { @Override public String calculate(String expression) { return expression; } @Override public String getVersion() { return "1.0"; } } package com.example; import classloader.ICalculator; public class CalculatorAdvanced implements ICalculator { @Override public String calculate(String expression) { return "Result is " + expression; } @Override public String getVersion() { return "2.0"; } } 在客戶端加載網絡上的類的過程:
    package classloader; public class CalculatorTest { public static void main(String[] args) { String url = "http://localhost:8080/ClassloaderTest/classes"; NetworkClassLoader ncl = new NetworkClassLoader(url); String basicClassName = "com.example.CalculatorBasic"; String advancedClassName = "com.example.CalculatorAdvanced"; try { Class<?> clazz = ncl.loadClass(basicClassName); // 加載一個版本的類 ICalculator calculator = (ICalculator) clazz.newInstance(); // 創建對象 System.out.println(calculator.getVersion()); clazz = ncl.loadClass(advancedClassName); // 加載另一個版本的類 calculator = (ICalculator) clazz.newInstance(); System.out.println(calculator.getVersion()); } catch (Exception e) { e.printStackTrace(); } } }



    總結

    以上是生活随笔為你收集整理的Java基础——类加载机制及原理的全部內容,希望文章能夠幫你解決所遇到的問題。

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

    免费在线观看一区二区三区 | 欧美日韩性视频在线 | 蜜臀av性久久久久蜜臀aⅴ流畅 | 免费观看国产精品视频 | 久久综合久久综合久久 | 久久久亚洲国产精品麻豆综合天堂 | 国产免费嫩草影院 | 中文字幕亚洲欧美日韩2019 | 国产在线免费观看 | 日本精品一区二区在线观看 | 天天躁日日躁狠狠躁av麻豆 | 日韩免费看视频 | 久久区二区 | 欧美成人一区二区 | 亚洲欧洲国产日韩精品 | 9999国产| 欧美另类成人 | 久久久久久久久久国产精品 | 亚洲国产欧美在线看片xxoo | 日韩高清片| 亚洲经典视频在线观看 | 亚洲闷骚少妇在线观看网站 | 五月婷婷综| 国产精品久久久久影院日本 | 成人av教育 | 亚洲国产精品传媒在线观看 | 91av99| 欧美激情一区不卡 | 国产小视频福利在线 | 国产亚洲免费观看 | 99精品视频免费 | 香蕉网在线观看 | 国产精品不卡在线 | 亚洲人成在线电影 | 日韩免费在线观看视频 | 日本中文字幕网址 | 激情婷婷色 | 最近能播放的中文字幕 | 97在线视频免费 | 久久精品亚洲 | 亚洲欧洲精品一区二区 | 久久高清| 免费一级特黄毛大片 | 亚洲干视频在线观看 | 久久久精品亚洲 | 中文字幕免费看 | 国产不卡免费av | 999成人| 欧美色操| 不卡的av在线播放 | 久久久久国产视频 | 日韩com | 成年人在线视频观看 | 久久99这里只有精品 | 亚洲高清av在线 | 黄色网址中文字幕 | 日本在线视频一区二区三区 | 国产高清网站 | 一区二区三区四区在线 | 久久不卡国产精品一区二区 | 片网站| 久草在在线 | 中文字幕在线视频一区 | 国产精品激情在线观看 | 黄色中文字幕在线 | 国产二区免费视频 | 亚洲国产欧美一区二区三区丁香婷 | 色综合久久天天 | 成片人卡1卡2卡3手机免费看 | 三级黄色片子 | 蜜臀久久99精品久久久酒店新书 | 成人免费网站在线观看 | 亚洲欧洲日韩在线观看 | 91亚瑟视频 | 国产精品国内免费一区二区三区 | 久久黄色影视 | 国产欧美精品一区二区三区 | 久久久久在线观看 | 亚洲在线色| 中文字幕久久亚洲 | av成人免费网站 | 91精品麻豆 | 久久视频这里只有精品 | 亚洲国产欧洲综合997久久, | 综合久久精品 | 丝袜美女视频网站 | 国产精品手机在线观看 | 日韩av在线资源 | 亚洲天堂网站视频 | 日日摸日日爽 | 天天草av| 免费成人在线电影 | 久久一区二区三区超碰国产精品 | 成人黄色免费在线观看 | 视频三区在线 | 成人av一区二区在线观看 | 97精品国产一二三产区 | 亚州精品一二三区 | 黄色一级片视频 | 在线观看视频一区二区三区 | 欧美日产在线观看 | www.久久久精品 | 玖玖在线免费视频 | 亚洲资源片 | 一级一片免费看 | 久久久久久蜜av免费网站 | av在线免费播放网站 | 色天天中文 | 久久人人插 | 欧美另类高清 videos | 国产特级毛片aaaaaaa高清 | 91精品免费视频 | 天天天操操操 | 精品日韩在线一区 | 一色屋精品视频在线观看 | 成人国产精品免费观看 | 一级性视频 | 天天色天天射综合网 | 午夜精选视频 | 日韩二区在线观看 | 久久好看 | 免费一级黄色 | 开心激情五月网 | 黄色毛片视频免费观看中文 | 久久激情精品 | 亚洲电影第一页av | 黄色a三级 | 婷婷久久综合网 | 天天操天天射天天舔 | 精品国产三级 | 久亚洲 | 免费福利在线观看 | 日韩免费二区 | 探花视频在线观看+在线播放 | 少妇搡bbbb搡bbb搡69 | aⅴ精品av导航 | 久草精品在线播放 | 免费男女羞羞的视频网站中文字幕 | 在线观看视频你懂 | 99婷婷狠狠成为人免费视频 | 免费网站黄色 | 五月花婷婷 | 久久久久久久久综合 | 99热国产精品 | 亚洲区另类春色综合小说校园片 | 91高清免费看 | 天天色天天 | 最近中文字幕大全中文字幕免费 | 免费看短| 五月婷婷六月丁香 | 久久爱www. | 欧美日韩视频网站 | 亚洲激情综合 | 免费看污污视频的网站 | 国产亚洲欧美日韩高清 | 久久综合电影 | 免费日韩一区二区三区 | 97精品伊人 | 国产成人精品免费在线观看 | 在线免费av观看 | 日日日日干| 在线免费观看麻豆视频 | 精品美女在线视频 | 国产精品欧美 | 欧美大香线蕉线伊人久久 | 91精品天码美女少妇 | 久草免费新视频 | 亚洲国内精品 | 成在人线av | 国产尤物在线观看 | 97超级碰碰碰视频在线观看 | 一二区精品 | 午夜视频在线瓜伦 | 在线天堂亚洲 | av线上免费看 | 久草视频在线免费看 | 久草免费色站 | 国产在线视频资源 | 亚洲va综合va国产va中文 | 久久国产网站 | 在线观看国产日韩 | 久久精品99国产精品亚洲最刺激 | 亚洲精品午夜国产va久久成人 | 精品国产一区二区三区日日嗨 | 日本高清中文字幕有码在线 | 日韩精品91偷拍在线观看 | 国产xvideos免费视频播放 | 超碰在线天天 | 最近免费中文字幕mv在线视频3 | 国产午夜精品一区 | 亚洲精品午夜久久久久久久久久久 | 国产精品免费看久久久8精臀av | 日韩欧美国产精品 | 人人爱爱| 中文字幕亚洲在线观看 | 国产亚洲精品久久久久久网站 | 国产精品男女啪啪 | 国产精品视频在线观看 | 美女视频黄是免费的 | 国产亚洲精品av | 久久久三级视频 | 在线观看播放av | 免费久久网 | 91桃色在线观看视频 | 久久免费电影网 | 国产91免费看 | 91黄在线看 | 久久黄色免费视频 | 亚洲午夜大片 | 日韩精品一区二区三区水蜜桃 | 国产精品第三页 | 日韩色av色资源 | 国产男女无遮挡猛进猛出在线观看 | 91试看 | www.av在线播放 | 日韩高清片 | 亚洲91网站 | 日本在线观看视频一区 | 九月婷婷人人澡人人添人人爽 | 深爱激情开心 | 人人干狠狠干 | 国产亚洲在线 | 国产免费叼嘿网站免费 | 亚洲精品电影在线 | 黄色毛片网站在线观看 | 国产精品久久久久久久久久久久 | 国产精品久久久久国产精品日日 | www黄色软件| 超碰在线观看av.com | 色吊丝在线永久观看最新版本 | 久久精品视频国产 | 天天射天天干天天 | 国产精品理论片在线观看 | 在线小视频国产 | 日日爽夜夜操 | 99久久精品国产毛片 | 天天草天天摸 | 天天激情天天干 | 国产精品久久久久一区二区三区 | 国产精品久久久久久999 | 国产色综合天天综合网 | 日韩一区二区免费视频 | 欧美极品xxxxx| 91精品一区二区三区蜜臀 | 国产亚洲在线观看 | 天天撸夜夜操 | 九九热免费观看 | 国产午夜精品一区二区三区欧美 | 2024av | 美女网站色 | 成人免费在线观看电影 | 欧美日韩三区二区 | 日韩高清不卡在线 | 胖bbbb搡bbbb擦bbbb | 91久久久国产精品 | 日韩国产精品毛片 | av3级在线| 天天躁天天操 | 人人澡人人添人人爽一区二区 | 免费看日韩片 | 免费视频久久久 | 亚洲永久字幕 | 国产福利一区二区在线 | 亚洲一区二区三区毛片 | 一区二区 不卡 | 88av色| 国产又粗又长又硬免费视频 | 日韩欧美精品在线观看 | 国产精品综合在线 | 一区二区三区国 | 久草免费福利在线观看 | 婷婷丁香七月 | 91久久国产精品 | 久草视频在线免费 | 成 人 免费 黄 色 视频 | 日韩中文字幕第一页 | 成人久久 | 美女黄频在线观看 | 亚洲精品国产精品久久99热 | 国产精品综合久久久 | 色先锋av资源中文字幕 | 日韩电影在线一区二区 | 久久福利影视 | av一二三区 | 97在线免费视频观看 | 丝袜足交在线 | 91精品国产高清自在线观看 | 五月婷婷在线视频观看 | 中文字幕国产精品 | 中文字幕在线日亚洲9 | 日韩乱码中文字幕 | 亚洲精品五月天 | 亚洲视频在线免费看 | 99精品欧美一区二区三区 | 国产成年免费视频 | 热久久精品在线 | 永久免费的啪啪网站免费观看浪潮 | 91精品爽啪蜜夜国产在线播放 | 99精品一区二区三区 | 欧美极品在线播放 | 97超碰免费| 色婷婷免费 | 亚洲精选在线观看 | 欧美日韩亚洲在线观看 | 国产99久久久国产精品免费二区 | 国产一级三级 | 一区二区三区精品在线视频 | 天天伊人狠狠 | av网址在线播放 | 亚洲dvd | 天天干天天操av | 国色天香av | 亚洲天堂网在线视频 | 免费网站v| 国产视频精选在线 | 久久精精品 | 99精品国产兔费观看久久99 | 久久国产精品久久精品国产演员表 | 日韩在线网 | 丁香色综合 | 亚洲欧美色婷婷 | 国产999久久久 | 夜添久久精品亚洲国产精品 | 久久久精品免费观看 | 97超碰免费在线观看 | 很黄很污的视频网站 | 四虎永久免费网站 | 国产你懂的在线 | 午夜视频一区二区三区 | 激情伊人五月天久久综合 | www激情com | 日韩在线观看av | 欧美一区二区视频97 | 国产精品高清在线观看 | 99 色| 日韩视频免费观看高清完整版在线 | 在线视频18在线视频4k | av免费黄色| 国产高清成人在线 | 在线蜜桃视频 | 成人播放器 | 91精品在线视频观看 | 亚洲精品在线观看免费 | 午夜av电影院 | 久久国产精品免费观看 | 久久久久99精品国产片 | 日本高清免费中文字幕 | 日韩精品一区二区三区在线视频 | 看黄色.com | 中文字幕日韩国产 | 狠狠撸电影 | 国产无套精品久久久久久 | 日本黄色免费播放 | 久操久| 91毛片在线| 国产一区二区手机在线观看 | 涩涩网站在线播放 | 在线观看国产www | 日韩在线三区 | 日韩三级一区 | 中文字幕一区二区三区在线观看 | av免费成人| 91在线91拍拍在线91 | 久久精品中文字幕少妇 | 国产v亚洲v | 丁香伊人网 | 97成人在线免费视频 | 国产精品欧美日韩 | 91经典在线 | 精品国产成人av在线免 | 日韩三级精品 | 91视频一8mav | 久久看毛片 | 国产视频亚洲精品 | 美女视频是黄的免费观看 | 亚洲综合在线一区二区三区 | 色综合天天视频在线观看 | 亚洲欧美国产精品va在线观看 | 激情中文在线 | 最新免费av在线 | 日韩视频 一区 | 亚洲美女视频在线观看 | 视频一区亚洲 | 激情偷乱人伦小说视频在线观看 | 日韩专区一区二区 | 国产综合香蕉五月婷在线 | 亚洲国产高清视频 | 久久这里只有精品1 | 激情五月婷婷激情 | 久久久国产精品视频 | 人人草在线观看 | www.成人精品 | 91亚洲精品久久久蜜桃 | 亚洲免费av一区二区 | 天天超碰 | 免费网址你懂的 | 天天玩天天干天天操 | 超碰在线97国产 | a成人v| 九月婷婷综合网 | 久久免费视频2 | 国产精品成人品 | 天天操天天拍 | 欧美日韩亚洲一 | 97高清免费视频 | 国产片免费在线观看视频 | 2019精品手机国产品在线 | 亚洲精品国产精品国自 | 一区二区国产精品 | 成人av高清在线观看 | 成年人免费看 | 在线三级播放 | 婷婷综合网 | 1区2区视频 | 日本高清dvd | 在线观看国产日韩欧美 | 久久99久久99精品免费看小说 | 久久久久久国产精品美女 | 美腿丝袜av | 日韩视频一区二区在线观看 | 国产伦精品一区二区三区照片91 | 粉嫩av一区二区三区四区在线观看 | 欧美一区在线看 | 久草香蕉在线 | 狠狠操狠狠干2017 | 激情综合五月婷婷 | 九九精品在线观看 | 日本精品一区二区在线观看 | 在线va网站| 中文字幕无吗 | 欧美a级在线| 在线看片成人 | 麻豆va一区二区三区久久浪 | 嫩嫩影院理论片 | 最新av在线播放 | 91九色性视频| 日本精品一区二区三区在线观看 | 91自拍视频在线观看 | 国产在线观看你懂的 | 欧美aaa大片 | av看片在线观看 | 曰韩精品 | 中文字幕高清 | 一区二区欧美在线观看 | 欧美一区二区三区免费看 | 黄色毛片视频 | 久久久视屏| 亚洲在线视频免费观看 | 手机成人免费视频 | 亚洲美女视频网 | 久久精品小视频 | 手机在线永久免费观看av片 | 四虎在线影视 | 97综合网| 99精品视频观看 | 日韩国产精品一区 | 亚洲精品国产精品乱码在线观看 | 亚洲激情影院 | 久艹视频免费观看 | 久久国产91| 蜜臀久久99精品久久久无需会员 | 国内揄拍国内精品 | 亚洲成人av在线电影 | 国产成人福利在线观看 | www五月婷婷| 亚洲电影成人 | 中文字幕精品一区久久久久 | 精品国产a| 字幕网资源站中文字幕 | 久热免费在线观看 | 国产一区在线免费观看视频 | 天天综合在线观看 | 成人 亚洲 欧美 | 免费男女羞羞的视频网站中文字幕 | 狂野欧美激情性xxxx欧美 | 成人a级大片 | 我要看黄色一级片 | 成人小视频在线观看免费 | 人人爽久久久噜噜噜电影 | 欧美一级性生活 | 福利视频网站 | 性色视频在线 | 久草线 | 久久久久久黄 | 国产探花视频在线播放 | 天天天干天天射天天天操 | 中文字幕人成乱码在线观看 | 天天干,夜夜爽 | 黄色av高清| 国产精品成人一区 | 在线天堂日本 | 日韩久久在线 | 99精品在这里| 欧美性超爽 | 成人av免费 | 久久99日韩 | wwwwww国产| 免费精品久久久 | 中文字幕亚洲不卡 | 91亚洲精品久久久中文字幕 | 久久久久亚洲精品中文字幕 | 婷婷在线观看视频 | 在线视频观看成人 | 少妇性色午夜淫片aaaze | 在线影院中文字幕 | 国产精品久久婷婷六月丁香 | 久久免费视频这里只有精品 | 日韩精品高清不卡 | 免费日韩高清 | 国产日韩在线看 | 不卡av电影在线观看 | 国产精品久久在线观看 | 九色精品免费永久在线 | 91.麻豆视频 | 日b视频国产 | 欧美日韩精品综合 | 国产亚洲精品久久网站 | 五月天六月丁香 | 欧美成年人在线视频 | 亚洲日本中文字幕在线观看 | 奇米影视在线99精品 | 日韩欧美一区二区在线播放 | 国产精品久久久久久超碰 | 日本久久中文 | 九九热在线视频免费观看 | 婷婷久久综合网 | 日韩精品欧美专区 | 亚洲精品乱码久久久久久蜜桃动漫 | av电影免费在线播放 | 超碰在线人人草 | 五月婷婷综合激情 | av大全免费在线观看 | 色网站在线免费观看 | av在线进入 | 国产一线在线 | 欧美久久久久 | 午夜精品在线看 | 久久国产影视 | 免费黄色一区 | 911免费视频 | 日本午夜免费福利视频 | 91久久黄色 | 欧美少妇的秘密 | 久久www免费人成看片高清 | 国产麻豆精品久久一二三 | 久久成人免费视频 | 精品一区二区精品 | 中文字幕在线看人 | 亚洲资源一区 | 日韩精品中文字幕在线观看 | 日韩欧美一区二区在线播放 | 亚洲资源视频 | 日韩高清无线码2023 | av一区二区三区在线 | 国产在线观看99 | 97成人精品视频在线播放 | 免费h漫在线观看 | 又大又硬又黄又爽视频在线观看 | 99精品久久精品一区二区 | 欧美日韩午夜在线 | 久久草在线精品 | 在线视频欧美亚洲 | 99视频在线免费看 | 国产香蕉视频在线观看 | 久久欧美在线电影 | 视频在线国产 | 国产xxxxx在线观看 | 国产精品成人在线观看 | 日韩精品一区二区三区丰满 | 中文字幕网站 | 在线播放 日韩专区 | 美女免费黄视频网站 | 欧美精品国产综合久久 | 欧美在线视频一区二区三区 | 欧美精品999 | 日日操日日插 | 丁香六月中文字幕 | 天天爽夜夜爽人人爽曰av | 久久色亚洲 | 一区二区高清在线 | www.五月激情.com | 国产精品久久久久久久毛片 | 在线精品在线 | 亚洲精品久久久久久久蜜桃 | 特级毛片在线免费观看 | 亚洲电影免费 | 久久综合精品国产一区二区三区 | 国产精品都在这里 | 免费观看国产成人 | 九色自拍视频 | 亚洲精品乱码久久久久 | 欧美日韩国产一二三区 | 日韩免费成人 | 91免费视频网站在线观看 | 国产成年免费视频 | 国产福利不卡视频 | www.狠狠操 | 天堂av一区二区 | 91亚洲国产成人久久精品网站 | 久久三级毛片 | 成人久久视频 | 日本精品一区二区三区在线观看 | 亚洲视频在线观看免费 | 亚洲va综合va国产va中文 | 又色又爽又黄 | 天天射天天艹 | 国产99在线免费 | 国产精品麻豆免费版 | 一区二区伦理电影 | 久久视频这里只有精品 | 人成午夜视频 | 岛国大片免费视频 | 美女精品 | 视频在线日韩 | 中文字幕123区 | 夜夜视频 | 天天干天天操天天干 | 九九热免费在线视频 | 九九热免费视频在线观看 | 亚洲欧洲在线视频 | 久久婷婷色综合 | 在线免费视频你懂的 | 久久久精品国产免费观看一区二区 | 国产精品自产拍在线观看中文 | 美女久久久久 | 精品国产免费人成在线观看 | 欧美夫妻性生活电影 | 中文字幕一区二区三区精华液 | 五月婷婷综合网 | 最近日韩免费视频 | 成 人 黄 色 片 在线播放 | 一色屋精品视频在线观看 | 精品国产区 | 成人av播放 | 三三级黄色片之日韩 | 亚洲永久在线 | 成人黄色大片网站 | 国产亚洲视频在线 | 91精品国产高清自在线观看 | 欧美视频在线二区 | 国产精品一区二区av麻豆 | 黄在线免费看 | 手机av电影在线 | 国产理论影院 | 久久伦理电影网 | 久久只精品99品免费久23小说 | 91网免费看| 亚洲激情 欧美激情 | 特及黄色片| 在线观看蜜桃视频 | 欧美精品免费在线观看 | 日韩av手机在线观看 | 中文字幕 在线看 | 欧美日韩国产一二 | 一区二区三区在线不卡 | 黄色免费网战 | 亚洲精品动漫成人3d无尽在线 | 欧美一级片在线免费观看 | 色婷婷在线播放 | 欧美日韩一区二区三区不卡 | 亚洲成av人片 | 手机在线永久免费观看av片 | 国产黄色片免费观看 | 精品久久久久久亚洲综合网站 | 国产在线观看国语版免费 | 免费黄在线观看 | 色爱区综合激月婷婷 | 天天插狠狠干 | 草久久久 | 亚洲色图22p | 91爱爱视频 | 永久精品视频 | 国产精品美女视频 | 香蕉在线播放 | av看片在线观看 | 久视频在线| 亚洲国产三级在线观看 | 日韩一区二区三区高清在线观看 | 欧美视频不卡 | av超碰在线 | 欧美成人在线网站 | 久久婷婷网 | 久久久综合九色合综国产精品 | 久久另类小说 | 亚洲成人在线免费 | 国产综合91 | av九九| 色五月成人 | 国产高清亚洲 | 国产高清绿奴videos | 久久免费视频99 | 四虎在线免费观看 | 亚洲码国产日韩欧美高潮在线播放 | 亚洲精品三级 | 在线黄色观看 | 国产成人三级一区二区在线观看一 | 久久av免费| 国产精品av在线免费观看 | 精品一二三区视频 | 久久久精品欧美 | 在线韩国电影免费观影完整版 | 久久国产午夜精品理论片最新版本 | 久久久久久久久久久高潮一区二区 | 欧美视频国产视频 | 亚洲精品白浆高清久久久久久 | 日韩美视频 | 成人免费xyz网站 | 亚洲一区 av | 天天天天爱天天躁 | 免费网站色 | 伊人久在线 | 亚洲精品视频在线观看免费视频 | www.狠狠插.com | 久久久久亚洲精品中文字幕 | 国产黄色成人 | 久久社区视频 | www.夜夜骑.com| 偷拍精品一区二区三区 | 成片免费观看视频999 | 伊人资源站 | 日韩在线视频免费看 | 色多视频在线观看 | 成在线播放 | 开心激情五月婷婷 | 久久久久久久影院 | 婷婷六月久久 | 国产69久久久欧美一级 | 亚洲jizzjizz日本少妇 | 黄色亚洲大片免费在线观看 | 免费黄色小网站 | 天天干天天做 | 18网站在线观看 | 久久久精品国产一区二区三区 | 狠狠色丁香婷婷综合基地 | 国产精品成人一区二区三区 | 国产高清视频网 | 一区二区在线电影 | 精品伦理一区二区三区 | 亚洲综合激情五月 | 综合色婷婷| 婷婷丁香激情综合 | 国产精品不卡一区 | 久久免费激情视频 | 91精品综合在线观看 | av成人在线网站 | 国产一区精品在线 | 黄色网址中文字幕 | 五月激情久久久 | 91人人在线 | 蜜臀一区二区三区精品免费视频 | 国产一区二区在线免费播放 | 亚洲国产合集 | 在线免费观看国产黄色 | 精品在线观看一区二区三区 | 99视频精品视频高清免费 | 精品国产一区二区三区四区在线观看 | 日韩午夜网站 | 91成人在线网站 | 国产精品嫩草在线 | 精品五月天 | 香蕉在线观看视频 | 中文字幕在线观看免费 | 日本中文字幕网 | av解说在线观看 | 国产精品久久艹 | 亚洲免费在线 | 国产成人精品在线 | 久久久久久综合 | 在线国产视频一区 | 天天干夜夜夜 | 国产在线更新 | 国产精品99久久免费观看 | 国产成人一区二区三区影院在线 | 久久伊人综合 | avhd高清在线谜片 | 精品视频国产 | 欧美日韩p片 | 日韩免费av在线 | 亚欧日韩av | 91亚州| 中文字幕免费成人 | 国产乱码精品一区二区三区介绍 | 国产精品久久久毛片 | 日韩中文字幕免费在线观看 | 天天色天天搞 | 天天操天天干天天摸 | 久久综合9988久久爱 | 91亚色免费视频 | 97国产超碰| 中文字幕免费高清在线 | 亚洲不卡在线 | 亚洲欧美日本A∨在线观看 青青河边草观看完整版高清 | 日韩精品不卡在线观看 | 91在线观看欧美日韩 | 93久久精品日日躁夜夜躁欧美 | 国产精品国产亚洲精品看不卡15 | 麻豆91在线观看 | 国产男女无遮挡猛进猛出在线观看 | 狠狠色丁香婷婷综合久久片 | 一区二区三区免费在线观看视频 | 有码中文在线 | 久久精品福利视频 | 一区二区三区精品在线视频 | 国产一级一片免费播放放a 一区二区三区国产欧美 | 在线有码中文 | 97偷拍视频 | 亚洲精品乱码久久久久久高潮 | 色com | 免费av一级电影 | 欧美性春潮 | 毛片美女网站 | 91自拍视频在线 | 日韩精品一区二区三区免费视频观看 | 超碰免费公开 | 久久精品三 | 99热九九这里只有精品10 | 色婷婷视频在线 | www.久久久com| 波多野结衣亚洲一区二区 | 午夜久久福利 | 日韩天堂在线观看 | 成人一区不卡 | 国产成人精品亚洲精品 | 国产一性一爱一乱一交 | 成人久久影院 | 日韩av看片| 免费a网址| 色丁香综合 | 久久久国产成人 | 伊人手机在线 | 亚洲性少妇性猛交wwww乱大交 | 在线观看国产v片 | 精品久久久久一区二区国产 | 亚洲精品啊啊啊 | 国产高清av在线播放 | 黄色影院在线观看 | 亚洲国产中文字幕在线 | 国产首页 | 亚洲国产播放 | 中文字幕一区二区三区在线播放 | 人人澡超碰碰97碰碰碰软件 | 最新av网址在线观看 | 91在线精品播放 | 女人18毛片a级毛片一区二区 | 西西444www | 91在线看黄| 久久艹人人 | 中文字幕在线免费播放 | 日本少妇视频 | 欧美日韩aa | 亚洲激情电影在线 | 啪啪肉肉污av国网站 | 精品国产伦一区二区三区观看体验 | 国产中文在线视频 | 婷婷视频导航 | 国产精品中文字幕在线观看 | 欧美精品免费在线观看 | 91在线精品一区二区 | 国产精品一区在线观看你懂的 | 国语对白少妇爽91 | 成年人视频在线免费播放 | 亚洲九九九在线观看 | 奇米四色影狠狠爱7777 | 狠狠网 | 日韩高清无线码2023 | 99热这里只有精品国产首页 | 99久热精品 | 中文字幕国产精品 | 狠狠躁夜夜a产精品视频 | 久久69精品久久久久久久电影好 | 欧美国产精品久久久久久免费 | 99久久精品国产观看 | 丝袜美腿在线视频 | 97精品国产91久久久久久久 | 天天干天天做天天爱 | 欧美一区二区三区不卡 | 伊人婷婷网 | 亚州精品成人 | 婷婷精品国产欧美精品亚洲人人爽 | 欧美在线你懂的 | 精品国产成人在线影院 | 久久久在线免费观看 | 色婷在线| www.av在线.com | 麻豆视频免费在线观看 | 欧美高清视频不卡网 | 免费在线观看av网址 | 免费观看的av网站 | 久青草视频在线观看 | 91探花在线视频 | 国产91aaa| 国产精品入口66mio女同 | 狂野欧美激情性xxxx欧美 | 中文字幕在线网址 | 一级黄色片毛片 | 在线一二三四区 | 国产在线看 | 国产精品嫩草在线 | 色片网站在线观看 | 免费看日韩| 91网站在线视频 | 日韩免费在线视频 | 欧美日韩在线视频一区二区 | 日本视频高清 | 极品久久久久 | 国产一区二区高清视频 | 91亚洲精品久久久久图片蜜桃 | 日韩特黄av| 丰满少妇对白在线偷拍 | 精品一区二区6 | 国产精品美女www爽爽爽视频 | 在线视频国产区 | 在线观看免费版高清版 | 亚洲国产精品人久久电影 | av理论电影| 免费在线观看一区 | www色,com | 91精品国产一区二区三区 | 国产精品嫩草在线 | 欧美精品一区在线 | 亚洲免费av网站 | 久草91视频 | 久久av影院| 婷婷六月在线 | 超碰97公开| av动态图片 | 综合色综合| 五月婷久| 中文久草 | 日韩天堂在线观看 | 亚洲视屏| 欧美一区日韩精品 | 国产精品一区二区久久精品爱涩 | 欧洲亚洲精品 | 亚洲九九精品 | 狠狠综合久久 | 久久国产精品久久久 | 国内精品在线一区 | 久久久精品免费观看 | 国产二区视频在线观看 | 丰满少妇在线观看网站 | 久久久激情网 | 黄色日本免费 | 精品一区91 | 91成人国产| 中文不卡视频 | 国产91精品久久久久久 | 91福利社区在线观看 | 久草免费资源 | 成年人视频免费在线 | 精品久久久久久一区二区里番 | 免费福利视频网站 | 国产国产人免费人成免费视频 | 91大神电影 | 在线蜜桃视频 | 欧美中文字幕第一页 | 草久在线视频 | 免费成人结看片 | 天天操夜操 | 黄色视屏免费在线观看 | 狠狠网 | 99精品久久久久久久久久综合 | 五月婷婷开心中文字幕 | 最新极品jizzhd欧美 | 久久av影视| av福利在线导航 | 激情欧美在线观看 | 一区二区三区免费在线播放 | 一区免费视频 | 久草剧场| 久久免费久久 | 蜜桃av观看 | 国产免费一区二区三区最新 | 国产精品不卡在线播放 | 国产成人精品免费在线观看 | 久草电影免费在线观看 | 久一在线 | av在线播放中文字幕 | 免费99精品国产自在在线 | 天天摸天天舔天天操 | 99r国产精品 | 免费在线一区二区三区 | 国产一线二线三线性视频 | 亚洲精品av中文字幕在线在线 | 亚洲一区二区三区精品在线观看 | 免费在线国产黄色 | av在线电影网站 | 在线观看亚洲a | ww亚洲ww亚在线观看 | 美女福利视频在线 | 久草在线精品观看 | 成人免费共享视频 | 国产一级大片免费看 | 精品视频久久久久久 | 99 视频 高清| 一级大片在线观看 | 日韩精品视频在线观看免费 | 亚洲经典视频 |