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

歡迎訪問 生活随笔!

生活随笔

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

java

Java虚拟机:深入详细分析Java ClassLoader原理与源码

發布時間:2024/9/30 java 29 豆豆
生活随笔 收集整理的這篇文章主要介紹了 Java虚拟机:深入详细分析Java ClassLoader原理与源码 小編覺得挺不錯的,現在分享給大家,幫大家做個參考.

?

一、什么是ClassLoader?

? ? ? ? ?ClassLoader就是類加載器,當我們寫好一個Java程序之后,都是由若干個.class文件組成的一個完整的Java應用程序,當程序在運行時,即會調用該程序的一個入口函數來調用系統的相關功能,而這些功能都被封裝在不同的class文件當中,所以經常要從這個class文件中要調用另外一個class文件中的方法,如果另外一個文件不存在的,則會引發系統異常。而程序在啟動的時候,并不會一次性加載程序所要用的所有class文件,而是根據程序的需要,通過Java的類加載機制(ClassLoader)來動態加載某個class文件到內存當中的,從而只有class文件被載入到了內存之后,才能被其它class所引用。如果一次性加載那么多jar包那么多class,那內存不就崩潰了?所以ClassLoader就是用來動態加載class文件到內存當中用的。

1、Class文件的認識:

我們都知道在Java中程序是運行在虛擬機中,我們平常用文本編輯器或者是IDE編寫的程序都是.java格式的文件,這是最基礎的源碼,但這類文件是不能直接運行的。如我們編寫一個簡單的程序HelloWorld.java:

public class HelloWorld{public static void main(String[] args){System.out.println("Hello world!");} }

如圖:

可以看到目錄下生成了.class文件,我們再從命令行中執行命令:

java HelloWorld

上面是基本代碼示例,這里重新拿出來是想讓大家將焦點回到class文件上,class文件是字節碼格式文件,java虛擬機并不能直接識別我們平常編寫的.java源文件,所以需要javac這個命令轉換成.class文件。另外,如果用C或者Python 編寫的程序正確轉換成.class文件后,java虛擬機也是可以識別運行的。

?

二、Java默認提供的三個ClassLoader:

1、BootStrap ClassLoader:稱為啟動類加載器,是Java類加載層次中最頂層的類加載器,負責加載JDK中的核心類庫,如:rt.jar、resources.jar、charsets.jar等,可以通過啟動jvm時指定-Xbootclasspath和路徑來改變Bootstrap ClassLoader的加載目錄。比如java -Xbootclasspath/a:path,被指定的文件追加到默認的bootstrap路徑中。也可通過如下程序獲得該類加載器從哪些地方加載了相關的jar或class文件:

URL[] urls = sun.misc.Launcher.getBootstrapClassPath().getURLs(); for (int i = 0; i < urls.length; i++) {System.out.println(urls[i].toExternalForm()); }

以下內容是上述程序從本機JDK環境所獲得的結果:

file:/C:/Program%20Files/Java/jdk1.6.0_22/jre/lib/resources.jar
file:/C:/Program%20Files/Java/jdk1.6.0_22/jre/lib/rt.jar
file:/C:/Program%20Files/Java/jdk1.6.0_22/jre/lib/sunrsasign.jar
file:/C:/Program%20Files/Java/jdk1.6.0_22/jre/lib/jsse.jar
file:/C:/Program%20Files/Java/jdk1.6.0_22/jre/lib/jce.jar
file:/C:/Program%20Files/Java/jdk1.6.0_22/jre/lib/charsets.jar
file:/C:/Program%20Files/Java/jdk1.6.0_22/jre/classes/

其實上述結果也是通過查找sun.boot.class.path這個系統屬性所得知的。?

System.out.println(System.getProperty("sun.boot.class.path"));

打印結果:

C:\Program Files\Java\jre1.8.0_91\lib\resources.jar;
C:\Program Files\Java\jre1.8.0_91\lib\rt.jar;
C:\Program Files\Java\jre1.8.0_91\lib\sunrsasign.jar;
C:\Program Files\Java\jre1.8.0_91\lib\jsse.jar;
C:\Program Files\Java\jre1.8.0_91\lib\jce.jar;
C:\Program Files\Java\jre1.8.0_91\lib\charsets.jar;
C:\Program Files\Java\jre1.8.0_91\lib\jfr.jar;
C:\Program Files\Java\jre1.8.0_91\classes

2、Extension ClassLoader:稱為擴展類加載器,負責加載Java的擴展類庫,默認加載JAVA_HOME/jre/lib/ext/目下的所有jar。還可以加載-D java.ext.dirs選項指定的目錄。

3、App ClassLoader:稱為系統類加載器,負責加載應用程序classpath目錄下的所有jar和class文件。

?????? ?注意:?除了Java默認提供的三個ClassLoader之外,用戶還可以根據需要定義自已的ClassLoader,而這些自定義的ClassLoader都必須繼承自java.lang.ClassLoader類,也包括Java提供的另外二個ClassLoader(Extension ClassLoader和App ClassLoader)在內,但是Bootstrap ClassLoader不繼承自ClassLoader,因為它不是一個普通的Java類,底層由C++編寫,已嵌入到了JVM內核當中,當JVM啟動后,Bootstrap ClassLoader也隨著啟動,負責加載完核心類庫后,并構造Extension ClassLoader和App ClassLoader類加載器。

?

三、ClassLoader加載類的原理:

?1、原理介紹:

???????ClassLoader使用的是雙親委托模型來搜索類的,每個ClassLoader實例都有一個父類加載器的引用(不是繼承的關系,是組合關系),虛擬機內置的啟動類加載器(Bootstrap ClassLoader)本身沒有父類加載器,但可以用作其它ClassLoader實例的的父類加載器。當一個ClassLoader實例需要加載某個類時,它會試圖親自搜索某個類之前,先把這個任務委托給它的父類加載器,這個過程是由上至下依次檢查的,首先由最頂層的類加載器Bootstrap ClassLoader試圖加載,如果沒加載到,則把任務轉交給Extension ClassLoader試圖加載,如果也沒加載到,則轉交給App ClassLoader?進行加載,如果它也沒有加載得到的話,則返回給委托的發起者,由它到指定的文件系統或網絡等URL中加載該類。如果它們都沒有加載到這個類時,則拋出ClassNotFoundException異常。否則將這個找到的類生成一個類的定義,并將它加載到內存當中,最后返回這個類在內存中的Class實例對象。

整個流程可以如下圖所示:

可以發現委托是從下向上,然后具體查找過程卻是自上至下

2、為什么要使用雙親委托這種模型呢?

???????因為這樣可以避免重復加載,當父親已經加載了該類的時候,就沒有必要子ClassLoader再加載一次。考慮到安全因素,我們試想一下,如果不使用這種委托模式,那我們就可以隨時使用自定義的String來動態替代java核心api中定義的類型,這樣會存在非常大的安全隱患,而雙親委托的方式,就可以避免這種情況,因為String已經在啟動時就被引導類加載器(Bootstrcp ClassLoader)加載,所以用戶自定義的ClassLoader永遠也無法加載一個自己寫的String,除非你改變JDK中ClassLoader搜索類的默認算法。

3、?但是JVM在搜索類的時候,又是如何判定兩個class是相同的呢?

? ? ?JVM在判定兩個class是否相同時,不僅要判斷兩個類名是否相同,而且要判斷是否由同一個類加載器實例加載的。只有兩者同時滿足的情況下,JVM才認為這兩個class是相同的。就算兩個class是同一份class字節碼,如果被兩個不同的ClassLoader實例所加載,JVM也會認為它們是兩個不同class。比如網絡上的一個Java類org.classloader.simple.NetClassLoaderSimple,javac編譯之后生成字節碼文件NetClassLoaderSimple.class,ClassLoaderA和ClassLoaderB這兩個類加載器并讀取了NetClassLoaderSimple.class文件,并分別定義出了java.lang.Class實例來表示這個類,對于JVM來說,它們是兩個不同的實例對象,但它們確實是同一份字節碼文件,如果試圖將這個Class實例生成具體的對象進行轉換時,就會拋運行時異常java.lang.ClassCaseException,提示這是兩個不同的類型。現在通過實例來驗證上述所描述的是否正確:

(1)在web服務器上建一個org.classloader.simple.NetClassLoaderSimple.java類

package org.classloader.simple;public class NetClassLoaderSimple {private NetClassLoaderSimple instance;public void setNetClassLoaderSimple(Object obj) {this.instance = (NetClassLoaderSimple)obj;} }

org.classloader.simple.NetClassLoaderSimple類的setNetClassLoaderSimple方法接收一個Object類型參數,并將它強制轉換成org.classloader.simple.NetClassLoaderSimple類型。?

(2)測試兩個class是否相同(NetWorkClassLoader.java)

package classloader;public class NewworkClassLoaderTest {public static void main(String[] args) {try {//測試加載網絡中的class文件String rootUrl = "http://localhost:8080/httpweb/classes";String className = "org.classloader.simple.NetClassLoaderSimple";NetworkClassLoader ncl1 = new NetworkClassLoader(rootUrl);NetworkClassLoader ncl2 = new NetworkClassLoader(rootUrl);Class<?> clazz1 = ncl1.loadClass(className);Class<?> clazz2 = ncl2.loadClass(className);Object obj1 = clazz1.newInstance();Object obj2 = clazz2.newInstance();clazz1.getMethod("setNetClassLoaderSimple", Object.class).invoke(obj1, obj2);} catch (Exception e) {e.printStackTrace();}}}

首先獲得網絡上一個class文件的二進制名稱,然后通過自定義的類加載器NetworkClassLoader創建兩個實例,并根據網絡地址分別加載這份class,并得到這兩個ClassLoader實例加載后生成的Class實例clazz1和clazz2,最后將這兩個Class實例分別生成具體的實例對象obj1和obj2,再通過反射調用clazz1中的setNetClassLoaderSimple方法。

(3)查看測試結果:

結論:從結果中可以看出,雖然是同一份class字節碼文件,但是由于被兩個不同的ClassLoader實例所加載,所以JVM認為它們就是兩個不同的類。

?

三、源碼解析:

我們看到了系統的3個類加載器,他們的具體執行順序為Bootstrap CLassloder → Extention ClassLoader→AppClassLoader

為了更好的理解,我們可以查看源碼。

1、sum.misc.Launcher:

它是一個java虛擬機的入口應用。

public class Launcher {private static Launcher launcher = new Launcher();private static String bootClassPath =System.getProperty("sun.boot.class.path");public static Launcher getLauncher() {return launcher;}private ClassLoader loader;public Launcher() {// Create the extension class loaderClassLoader extcl;try {extcl = ExtClassLoader.getExtClassLoader();} catch (IOException e) {throw new InternalError("Could not create extension class loader", e);}// Now create the class loader to use to launch the applicationtry {loader = AppClassLoader.getAppClassLoader(extcl);} catch (IOException e) {throw new InternalError("Could not create application class loader", e);}//設置AppClassLoader為線程上下文類加載器,這個文章后面部分講解Thread.currentThread().setContextClassLoader(loader);}/** Returns the class loader used to launch the main application.*/public ClassLoader getClassLoader() {return loader;}/** The class loader used for loading installed extensions.*/static class ExtClassLoader extends URLClassLoader {}/*** The class loader used for loading from java.class.path.* runs in a restricted security context.*/static class AppClassLoader extends URLClassLoader {}

源碼有精簡,我們可以得到相關的信息。

(1)Launcher初始化了ExtClassLoader和AppClassLoader。

(2)Launcher中并沒有看見BootstrapClassLoader,但通過System.getProperty("sun.boot.class.path")得到了字符串bootClassPath,這個應該就是BootstrapClassLoader加載的jar包路徑。

我們可以先代碼測試一下sun.boot.class.path是什么內容。

System.out.println(System.getProperty("sun.boot.class.path"));

得到的結果是:

C:\Program Files\Java\jre1.8.0_91\lib\resources.jar;
C:\Program Files\Java\jre1.8.0_91\lib\rt.jar;
C:\Program Files\Java\jre1.8.0_91\lib\sunrsasign.jar;
C:\Program Files\Java\jre1.8.0_91\lib\jsse.jar;
C:\Program Files\Java\jre1.8.0_91\lib\jce.jar;
C:\Program Files\Java\jre1.8.0_91\lib\charsets.jar;
C:\Program Files\Java\jre1.8.0_91\lib\jfr.jar;
C:\Program Files\Java\jre1.8.0_91\classes

可以看到,這些全是JRE目錄下的jar包或者是class文件。

2、ExtClassLoader源碼:

如果你有足夠的好奇心,你應該會對它的源碼感興趣:

/** The class loader used for loading installed extensions.*/static class ExtClassLoader extends URLClassLoader {static {ClassLoader.registerAsParallelCapable();}/*** create an ExtClassLoader. The ExtClassLoader is created* within a context that limits which files it can read*/public static ExtClassLoader getExtClassLoader() throws IOException{final File[] dirs = getExtDirs();try {// Prior implementations of this doPrivileged() block supplied// aa synthesized ACC via a call to the private method// ExtClassLoader.getContext().return AccessController.doPrivileged(new PrivilegedExceptionAction<ExtClassLoader>() {public ExtClassLoader run() throws IOException {int len = dirs.length;for (int i = 0; i < len; i++) {MetaIndex.registerDirectory(dirs[i]);}return new ExtClassLoader(dirs);}});} catch (java.security.PrivilegedActionException e) {throw (IOException) e.getException();}}private static File[] getExtDirs() {String s = System.getProperty("java.ext.dirs");File[] dirs;if (s != null) {StringTokenizer st =new StringTokenizer(s, File.pathSeparator);int count = st.countTokens();dirs = new File[count];for (int i = 0; i < count; i++) {dirs[i] = new File(st.nextToken());}} else {dirs = new File[0];}return dirs;}......}

我們先前的內容有說過,可以指定-D java.ext.dirs參數來添加和改變ExtClassLoader的加載路徑。這里我們通過可以編寫測試代碼。

System.out.println(System.getProperty("java.ext.dirs"));

結果如下:

C:\Program Files\Java\jre1.8.0_91\lib\ext;C:\Windows\Sun\Java\lib\ext

3、AppClassLoader源碼:

/*** The class loader used for loading from java.class.path.* runs in a restricted security context.*/static class AppClassLoader extends URLClassLoader {public static ClassLoader getAppClassLoader(final ClassLoader extcl)throws IOException{final String s = System.getProperty("java.class.path");final File[] path = (s == null) ? new File[0] : getClassPath(s);return AccessController.doPrivileged(new PrivilegedAction<AppClassLoader>() {public AppClassLoader run() {URL[] urls =(s == null) ? new URL[0] : pathToURLs(path);return new AppClassLoader(urls, extcl);}});}......}

可以看到AppClassLoader加載的就是java.class.path下的路徑。我們同樣打印它的值。

System.out.println(System.getProperty("java.class.path"));

結果:

D:\workspace\ClassLoaderDemo\bin

這個路徑其實就是當前java工程目錄bin,里面存放的是編譯生成的class文件。

好了,自此我們已經知道了BootstrapClassLoader、ExtClassLoader、AppClassLoader實際是查閱相應的環境屬性sun.boot.class.path、java.ext.dirs和java.class.path來加載資源文件的。

4、父加載器不是父類:

(1)接下來我們先驗證ClassLoader加載類的原理:?

測試1:打印ClassLoader類的層次結構,請看下面這段代碼:

ClassLoader loader = ClassLoaderTest.class.getClassLoader(); //獲得加載ClassLoaderTest.class這個類的類加載器 while(loader != null) {System.out.println(loader);loader = loader.getParent(); //獲得父類加載器的引用 } System.out.println(loader);

?打印結果:

第一行結果說明:ClassLoaderTest的類加載器是AppClassLoader。

第二行結果說明:AppClassLoader的類加器是ExtClassLoader,即parent=ExtClassLoader。

第三行結果說明:ExtClassLoader的類加器是Bootstrap ClassLoader,因為Bootstrap ClassLoader不是一個普通的Java類,所以ExtClassLoader的parent=null,所以第三行的打印結果為null就是這個原因。(詳細原因下面會講)

測試2:將ClassLoaderTest.class打包成ClassLoaderTest.jar,放到Extension ClassLoader的加載目錄下(JAVA_HOME/jre/lib/ext),然后重新運行這個程序,得到的結果會是什么樣呢?

打印結果:

打印結果分析:為什么第一行的結果是ExtClassLoader呢?

????? 因為ClassLoader的委托模型機制,當我們要用ClassLoaderTest.class這個類的時候,AppClassLoader在試圖加載之前,先委托給Bootstrcp ClassLoader,Bootstracp ClassLoader發現自己沒找到,它就告訴ExtClassLoader,兄弟,我這里沒有這個類,你去加載看看,然后Extension ClassLoader拿著這個類去它指定的類路徑(JAVA_HOME/jre/lib/ext)試圖加載,唉,它發現在ClassLoaderTest.jar這樣一個文件中包含ClassLoaderTest.class這樣的一個文件,然后它把找到的這個類加載到內存當中,并生成這個類的Class實例對象,最后把這個實例返回。所以ClassLoaderTest.class的類加載器是ExtClassLoader。

第二行的結果為null,是因為ExtClassLoader的父類加載器是Bootstrap ClassLoader。

測試3:用Bootstrcp ClassLoader來加載ClassLoaderTest.class,有兩種方式:

1、在jvm中添加-Xbootclasspath參數,指定Bootstrcp ClassLoader加載類的路徑,并追加我們自已的jar(ClassTestLoader.jar)

2、將class文件放到JAVA_HOME/jre/classes/目錄下(上面有提到)

方式1:(我用的是Eclipse開發工具,用命令行是在java命令后面添加-Xbootclasspath參數)

打開Run配置對話框:

配置好如圖中所述的參數后,重新運行程序,產的結果如下所示:(類加載的過程,只摘下了一部份)

打印結果:

[Loaded java.io.FileReader from C:\Program Files\Java\jdk1.6.0_22\jre\lib\rt.jar]
[Loaded sun.nio.cs.StreamDecoder from C:\Program Files\Java\jdk1.6.0_22\jre\lib\rt.jar]
[Loaded java.util.ArrayList from C:\Program Files\Java\jdk1.6.0_22\jre\lib\rt.jar]
[Loaded java.lang.reflect.Array from C:\Program Files\Java\jdk1.6.0_22\jre\lib\rt.jar]
[Loaded java.util.Locale from C:\Program Files\Java\jdk1.6.0_22\jre\lib\rt.jar]
[Loaded java.util.concurrent.ConcurrentMap from C:\Program Files\Java\jdk1.6.0_22\jre\lib\rt.jar]
[Loaded java.util.concurrent.ConcurrentHashMap from C:\Program Files\Java\jdk1.6.0_22\jre\lib\rt.jar]
[Loaded java.util.concurrent.locks.Lock from C:\Program Files\Java\jdk1.6.0_22\jre\lib\rt.jar]
[Loaded java.util.concurrent.locks.ReentrantLock from C:\Program Files\Java\jdk1.6.0_22\jre\lib\rt.jar]
[Loaded java.util.concurrent.ConcurrentHashMap$Segment from C:\Program Files\Java\jdk1.6.0_22\jre\lib\rt.jar]
[Loaded java.util.concurrent.locks.AbstractOwnableSynchronizer from C:\Program Files\Java\jdk1.6.0_22\jre\lib\rt.jar]
[Loaded java.util.concurrent.locks.AbstractQueuedSynchronizer from C:\Program Files\Java\jdk1.6.0_22\jre\lib\rt.jar]
[Loaded java.util.concurrent.locks.ReentrantLock$Sync from C:\Program Files\Java\jdk1.6.0_22\jre\lib\rt.jar]
[Loaded java.util.concurrent.locks.ReentrantLock$NonfairSync from C:\Program Files\Java\jdk1.6.0_22\jre\lib\rt.jar]
[Loaded java.util.concurrent.locks.AbstractQueuedSynchronizer$Node from C:\Program Files\Java\jdk1.6.0_22\jre\lib\rt.jar]
[Loaded java.util.concurrent.ConcurrentHashMap$HashEntry from C:\Program Files\Java\jdk1.6.0_22\jre\lib\rt.jar]
[Loaded java.lang.CharacterDataLatin1 from C:\Program Files\Java\jdk1.6.0_22\jre\lib\rt.jar]
[Loaded java.io.ObjectStreamClass from C:\Program Files\Java\jdk1.6.0_22\jre\lib\rt.jar]
[Loaded sun.net.www.ParseUtil from C:\Program Files\Java\jdk1.6.0_22\jre\lib\rt.jar]
[Loaded java.util.BitSet from C:\Program Files\Java\jdk1.6.0_22\jre\lib\rt.jar]
[Loaded java.net.Parts from C:\Program Files\Java\jdk1.6.0_22\jre\lib\rt.jar]
[Loaded java.net.URLStreamHandler from C:\Program Files\Java\jdk1.6.0_22\jre\lib\rt.jar]
[Loaded sun.net.www.protocol.file.Handler from C:\Program Files\Java\jdk1.6.0_22\jre\lib\rt.jar]
[Loaded java.util.HashSet from C:\Program Files\Java\jdk1.6.0_22\jre\lib\rt.jar]
[Loaded sun.net.www.protocol.jar.Handler from C:\Program Files\Java\jdk1.6.0_22\jre\lib\rt.jar]
[Loaded sun.misc.Launcher$AppClassLoader from C:\Program Files\Java\jdk1.6.0_22\jre\lib\rt.jar]
[Loaded sun.misc.Launcher$AppClassLoader$1 from C:\Program Files\Java\jdk1.6.0_22\jre\lib\rt.jar]
[Loaded java.lang.SystemClassLoaderAction from C:\Program Files\Java\jdk1.6.0_22\jre\lib\rt.jar]
[Path C:\Program Files\Java\jdk1.6.0_22\jre\classes]
[Loaded classloader.ClassLoaderTest from C:\Program Files\Java\jdk1.6.0_22\jre\classes]
null ?//這是打印的結果
C:\Program Files\Java\jdk1.6.0_22\jre\lib\resources.jar;C:\Program Files\Java\jdk1.6.0_22\jre\lib\rt.jar;
C:\Program Files\Java\jdk1.6.0_22\jre\lib\sunrsasign.jar;C:\Program Files\Java\jdk1.6.0_22\jre\lib\jsse.jar;
C:\Program Files\Java\jdk1.6.0_22\jre\lib\jce.jar;C:\Program Files\Java\jdk1.6.0_22\jre\lib\charsets.jar;
C:\Program Files\Java\jdk1.6.0_22\jre\classes;c:\ClassLoaderTest.jar ?
//這一段是System.out.println(System.getProperty("sun.boot.class.path"));打印出來的。這個路徑就是Bootstrcp ClassLoader默認搜索類的路徑
[Loaded java.lang.Shutdown from C:\Program Files\Java\jdk1.6.0_22\jre\lib\rt.jar]
[Loaded java.lang.Shutdown$Lock from C:\Program Files\Java\jdk1.6.0_22\jre\lib\rt.jar]

方式2:將ClassLoaderTest.jar解壓后,放到JAVA_HOME/jre/classes目錄下,如下圖所示:

提示:jre目錄下默認沒有classes目錄,需要自己手動創建一個

打印結果:

從結果中可以看出,兩種方式都實現了將ClassLoaderTest.class由Bootstrcp ClassLoader加載成功了。

(2)為什么上面輸出的Bootstrcp ClassLoader的值是null呢?

難道是Bootstrcp ClassLoader類加載器沒有加載?當然不是!要想弄明白這些,我們首先得知道一個前提:父加載器不是父類

我們先前已經粘貼了ExtClassLoader和AppClassLoader的代碼。

static class ExtClassLoader extends URLClassLoader {} static class AppClassLoader extends URLClassLoader {}

可以看見ExtClassLoader和AppClassLoader同樣繼承自URLClassLoader,但上面一小節代碼中,為什么調用AppClassLoader的getParent()代碼會得到ExtClassLoader的實例呢?先從URLClassLoader說起,這個類又是什么?

先上一張類的繼承關系圖:

URLClassLoader的源碼中并沒有找到getParent()方法。這個方法在ClassLoader.java中。

public abstract class ClassLoader {// The parent class loader for delegation // Note: VM hardcoded the offset of this field, thus all new fields // must be added *after* it. private final ClassLoader parent; // The class loader for the system// @GuardedBy("ClassLoader.class") private static ClassLoader scl;private ClassLoader(Void unused, ClassLoader parent) {this.parent = parent;... } protected ClassLoader(ClassLoader parent) {this(checkCreateClassLoader(), parent); } protected ClassLoader() {this(checkCreateClassLoader(), getSystemClassLoader()); } public final ClassLoader getParent() {if (parent == null)return null;return parent; } public static ClassLoader getSystemClassLoader() {initSystemClassLoader();if (scl == null) {return null;}return scl; }private static synchronized void initSystemClassLoader() {if (!sclSet) {if (scl != null)throw new IllegalStateException("recursive invocation");sun.misc.Launcher l = sun.misc.Launcher.getLauncher();if (l != null) {Throwable oops = null;//通過Launcher獲取ClassLoaderscl = l.getClassLoader();try {scl = AccessController.doPrivileged(new SystemClassLoaderAction(scl));} catch (PrivilegedActionException pae) {oops = pae.getCause();if (oops instanceof InvocationTargetException) {oops = oops.getCause();}}if (oops != null) {if (oops instanceof Error) {throw (Error) oops;} else {// wrap the exceptionthrow new Error(oops);}}}sclSet = true;} } }

我們可以看到getParent()實際上返回的就是一個ClassLoader對象parent,parent的賦值是在ClassLoader對象的構造方法中,它有兩個情況:

(1)由外部類創建ClassLoader時直接指定一個ClassLoader為parent。

(2)由getSystemClassLoader()方法生成,也就是在sun.misc.Laucher通過getClassLoader()獲取,也就是AppClassLoader。直白的說,一個ClassLoader創建時如果沒有指定parent,那么它的parent默認就是AppClassLoader。

我們主要研究的是ExtClassLoader與AppClassLoader的parent的來源,正好它們與Launcher類有關,我們上面已經粘貼過Launcher的部分代碼。

public class Launcher {private static URLStreamHandlerFactory factory = new Factory();private static Launcher launcher = new Launcher();private static String bootClassPath =System.getProperty("sun.boot.class.path");public static Launcher getLauncher() {return launcher;}private ClassLoader loader;public Launcher() {// Create the extension class loaderClassLoader extcl;try {extcl = ExtClassLoader.getExtClassLoader();} catch (IOException e) {throw new InternalError("Could not create extension class loader", e);}// Now create the class loader to use to launch the applicationtry {//將ExtClassLoader對象實例傳遞進去loader = AppClassLoader.getAppClassLoader(extcl);} catch (IOException e) {throw new InternalError("Could not create application class loader", e);}public ClassLoader getClassLoader() {return loader;} static class ExtClassLoader extends URLClassLoader {/*** create an ExtClassLoader. The ExtClassLoader is created* within a context that limits which files it can read*/public static ExtClassLoader getExtClassLoader() throws IOException{final File[] dirs = getExtDirs();try {// Prior implementations of this doPrivileged() block supplied// aa synthesized ACC via a call to the private method// ExtClassLoader.getContext().return AccessController.doPrivileged(new PrivilegedExceptionAction<ExtClassLoader>() {public ExtClassLoader run() throws IOException {//ExtClassLoader在這里創建return new ExtClassLoader(dirs);}});} catch (java.security.PrivilegedActionException e) {throw (IOException) e.getException();}}/** Creates a new ExtClassLoader for the specified directories.*/public ExtClassLoader(File[] dirs) throws IOException {super(getExtURLs(dirs), null, factory);}}}

我們需要注意的是:

ClassLoader extcl;extcl = ExtClassLoader.getExtClassLoader();loader = AppClassLoader.getAppClassLoader(extcl);

代碼已經說明了問題AppClassLoader的parent是一個ExtClassLoader實例。

ExtClassLoader并沒有直接找到對parent的賦值。它調用了它的父類也就是URLClassLoder的構造方法并傳遞了3個參數。

public ExtClassLoader(File[] dirs) throws IOException {super(getExtURLs(dirs), null, factory); }

對應的代碼:

public URLClassLoader(URL[] urls, ClassLoader parent,URLStreamHandlerFactory factory) {super(parent); }

答案已經很明了了,ExtClassLoader的parent為null。

上面張貼這么多代碼也是為了說明AppClassLoader的parent是ExtClassLoader,ExtClassLoader的parent是null。這符合我們之前編寫的測試代碼。

不過,細心的同學發現,還是有疑問的我們只看到ExtClassLoader和AppClassLoader的創建,那么BootstrapClassLoader呢?

還有,ExtClassLoader的父加載器為null,但是Bootstrap CLassLoader卻可以當成它的父加載器這又是為何呢?

我們繼續往下進行。

(3)Bootstrap ClassLoader是由C++編寫的:

Bootstrap ClassLoader是由C/C++編寫的,它本身是虛擬機的一部分,所以它并不是一個JAVA類,也就是無法在java代碼中獲取它的引用,JVM啟動時通過Bootstrap類加載器加載rt.jar等核心jar包中的class文件,之前的int.class,String.class都是由它加載。然后呢,我們前面已經分析了,JVM初始化sun.misc.Launcher并創建Extension ClassLoader和AppClassLoader實例。并將ExtClassLoader設置為AppClassLoader的父加載器。Bootstrap沒有父加載器,但是它卻可以作用一個ClassLoader的父加載器。比如ExtClassLoader。這也可以解釋之前通過ExtClassLoader的getParent方法獲取為Null的現象。具體是什么原因,很快就知道答案了。

(4)重要方法:

①loadClass():

JDK文檔中是這樣寫的,通過指定的全限定類名加載class,它通過同名的loadClass(String,boolean)方法。

protected Class<?> loadClass(String name,boolean resolve)throws ClassNotFoundException

上面是方法原型,一般實現這個方法的步驟是

  • 執行findLoadedClass(String)去檢測這個class是不是已經加載過了。
  • 執行父加載器的loadClass方法。如果父加載器為null,則jvm內置的加載器去替代,也就是Bootstrap ClassLoader。這也解釋了ExtClassLoader的parent為null,但仍然說Bootstrap ClassLoader是它的父加載器。
  • 如果向上委托父加載器沒有加載成功,則通過findClass(String)查找。
  • 如果class在上面的步驟中找到了,參數resolve又是true的話,那么loadClass()又會調用resolveClass(Class)這個方法來生成最終的Class對象。 我們可以從源代碼看出這個步驟。

    protected Class<?> loadClass(String name, boolean resolve)throws ClassNotFoundException{synchronized (getClassLoadingLock(name)) {// 首先,檢測是否已經加載Class<?> c = findLoadedClass(name);if (c == null) {long t0 = System.nanoTime();try {if (parent != null) {//父加載器不為空則調用父加載器的loadClassc = parent.loadClass(name, false);} else {//父加載器為空則調用Bootstrap Classloaderc = findBootstrapClassOrNull(name);}} catch (ClassNotFoundException e) {// ClassNotFoundException thrown if class not found// from the non-null parent class loader}if (c == null) {// If still not found, then invoke findClass in order// to find the class.long t1 = System.nanoTime();//父加載器沒有找到,則調用findclassc = findClass(name);// this is the defining class loader; record the statssun.misc.PerfCounter.getParentDelegationTime().addTime(t1 - t0);sun.misc.PerfCounter.getFindClassTime().addElapsedTimeFrom(t1);sun.misc.PerfCounter.getFindClasses().increment();}}if (resolve) {//調用resolveClass()resolveClass(c);}return c;}}

    ?

    ?

    四、定義自已的ClassLoader

    既然JVM已經提供了默認的類加載器,為什么還要定義自已的類加載器呢?

    ? ? ? 因為Java中提供的默認ClassLoader,只加載指定目錄下的jar和class,如果我們想加載其它位置的類或jar時,比如:我要加載網絡上的一個class文件,通過動態加載到內存之后,要調用這個類中的方法實現我的業務邏輯。在這樣的情況下,默認的ClassLoader就不能滿足我們的需求了,所以需要定義自己的ClassLoader。

    1、定義自已的類加載器的步驟為:

    (1)繼承java.lang.ClassLoader抽象類;

    (2)重寫父類的findClass方法,改變搜索類的算法;

    (3)在findClass()中調用defineClass(),將class的字節碼數組轉換成Class類的實例。

    讀者可能在這里有疑問,父類有那么多方法,為什么偏偏只重寫findClass方法?

    因為JDK已經在loadClass方法中幫我們實現了ClassLoader搜索類的算法,當在loadClass方法中搜索不到類時,loadClass方法就會調用findClass方法來搜索類,所以我們只需重寫該方法即可。如沒有特殊的要求,一般不建議重寫loadClass搜索類的算法。下圖是API中ClassLoader的loadClass方法:

    注意點:

    **一個ClassLoader創建時如果沒有指定parent,那么它的parent默認就是AppClassLoader。 **

    上面說的是,如果自定義一個ClassLoader,默認的parent父加載器是AppClassLoader,因為這樣就能夠保證它能訪問系統內置加載器加載成功的class文件。

    示例:自定義一個NetworkClassLoader,用于加載網絡上的class文件:

    package classloader;import java.io.ByteArrayOutputStream; import java.io.InputStream; import java.net.URL;/*** 加載網絡class的ClassLoader*/ public class NetworkClassLoader extends ClassLoader {private String rootUrl;public NetworkClassLoader(String rootUrl) {this.rootUrl = rootUrl;}@Overrideprotected Class<?> findClass(String name) throws ClassNotFoundException {Class clazz = null;//this.findLoadedClass(name); // 父類已加載 //if (clazz == null) { //檢查該類是否已被加載過byte[] classData = getClassData(name); //根據類的二進制名稱,獲得該class文件的字節碼數組if (classData == null) {throw new ClassNotFoundException();}clazz = defineClass(name, classData, 0, classData.length); //將class的字節碼數組轉換成Class類的實例//} return clazz;}private byte[] getClassData(String name) {InputStream is = null;try {String path = classNameToPath(name);URL url = new URL(path);byte[] buff = new byte[1024*4];int len = -1;is = url.openStream();ByteArrayOutputStream baos = new ByteArrayOutputStream();while((len = is.read(buff)) != -1) {baos.write(buff,0,len);}return baos.toByteArray();} catch (Exception e) {e.printStackTrace();} finally {if (is != null) {try {is.close();} catch(IOException e) {e.printStackTrace();}}}return null;}private String classNameToPath(String name) {return rootUrl + "/" + name.replace(".", "/") + ".class";} }

    測試類:

    package classloader;public class ClassLoaderTest {public static void main(String[] args) {try {/*ClassLoader loader = ClassLoaderTest.class.getClassLoader(); //獲得ClassLoaderTest這個類的類加載器while(loader != null) {System.out.println(loader);loader = loader.getParent(); //獲得父加載器的引用}System.out.println(loader);*/String rootUrl = "http://localhost:8080/httpweb/classes";NetworkClassLoader networkClassLoader = new NetworkClassLoader(rootUrl);String classname = "org.classloader.simple.NetClassLoaderTest";Class clazz = networkClassLoader.loadClass(classname);System.out.println(clazz.getClassLoader());} catch (Exception e) {e.printStackTrace();}}}

    打印結果:

    下圖是我機器上web服務器的目錄結構:

    目前常用web服務器中都定義了自己的類加載器,用于加載web應用指定目錄下的類庫(jar或class),如:Weblogic、Jboss、tomcat等,下面我以Tomcat為例,展示該web容器都定義了哪些個類加載器:

    1、新建一個web工程httpweb

    2、新建一個ClassLoaderServletTest,用于打印web容器中的ClassLoader層次結構

    import java.io.IOException; import java.io.PrintWriter;import javax.servlet.ServletException; import javax.servlet.http.HttpServlet; import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletResponse;public class ClassLoaderServletTest extends HttpServlet {public void doGet(HttpServletRequest request, HttpServletResponse response)throws ServletException, IOException {response.setContentType("text/html");PrintWriter out = response.getWriter();ClassLoader loader = this.getClass().getClassLoader();while(loader != null) {out.write(loader.getClass().getName()+"<br/>");loader = loader.getParent();}out.write(String.valueOf(loader));out.flush();out.close();}public void doPost(HttpServletRequest request, HttpServletResponse response)throws ServletException, IOException {this.doGet(request, response);}}

    3、配置Servlet,并啟動服務:

    <?xml version="1.0" encoding="UTF-8"?> <web-app version="2.4" xmlns="http://java.sun.com/xml/ns/j2ee" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://java.sun.com/xml/ns/j2ee http://java.sun.com/xml/ns/j2ee/web-app_2_4.xsd"><servlet><servlet-name>ClassLoaderServletTest</servlet-name><servlet-class>ClassLoaderServletTest</servlet-class></servlet><servlet-mapping><servlet-name>ClassLoaderServletTest</servlet-name><url-pattern>/servlet/ClassLoaderServletTest</url-pattern></servlet-mapping><welcome-file-list><welcome-file>index.jsp</welcome-file></welcome-file-list> </web-app>

    ?4、訪問Servlet,獲得顯示結果:

    ?

    五、總結:

    ClassLoader用來加載class文件的。

    系統內置的ClassLoader通過雙親委托來加載指定路徑下的class和資源。

    可以自定義ClassLoader一般覆蓋findClass()方法。

    ContextClassLoader與線程相關,可以獲取和設置,可以繞過雙親委托的機制。

    ?

    ?

    本文整理自以下的這兩篇博客:

    https://blog.csdn.net/xyang81/article/details/7292380

    https://blog.csdn.net/briblue/article/details/54973413#commentBox

    總結

    以上是生活随笔為你收集整理的Java虚拟机:深入详细分析Java ClassLoader原理与源码的全部內容,希望文章能夠幫你解決所遇到的問題。

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

    日韩一二区在线 | 奇米影视在线99精品 | 国产打女人屁股调教97 | 久久综合九色综合久久久精品综合 | 精品国产一区二区三区久久影院 | 国产五十路毛片 | 日韩在线免费看 | 国产精品麻豆一区二区三区 | 免费 在线 中文 日本 | 久99久久| 国产精品白浆 | 最新av免费在线 | 天天干夜夜想 | 天天天干天天天操 | 久久久久久久久久久综合 | 青草视频在线免费 | 天天操天天操天天操天天操天天操 | 中文字幕在线不卡国产视频 | 欧美日韩高清 | 国产精品一区二区三区99 | 国产大片免费久久 | 亚洲精品视频在线观看免费视频 | 嫩模bbw搡bbbb搡bbbb | 久久成人综合视频 | 波多野结衣电影一区二区三区 | 国产成人精品免高潮在线观看 | 免费精品人在线二线三线 | 99久久激情| 国产一区二区三区四区大秀 | 亚洲免费永久精品国产 | 午夜天天操 | 天天爱天天 | 在线日韩中文字幕 | 中文字幕日韩一区二区三区不卡 | 国产成人精品免高潮在线观看 | 成人午夜网 | 免费久久片 | 久久久久久久久久久久久久av | 国产精品美| 国产黄色播放 | 日日夜夜天天人人 | 久草视频在线资源站 | 久久只有精品 | 国产小视频在线免费观看视频 | 日韩 在线观看 | 国产精品久久久久永久免费观看 | 欧美日韩在线电影 | 五月花激情 | 97超碰人人模人人人爽人人爱 | 久草久草久草久草 | 91精品久久久久久久久 | 2019免费中文字幕 | 天堂av免费 | 亚洲资源片 | av电影免费在线 | 狠狠色丁香九九婷婷综合五月 | 国产一级免费观看 | 久久久久久久国产精品视频 | 中文字幕文字幕一区二区 | 涩五月婷婷 | 9幺看片 | 99视频偷窥在线精品国自产拍 | 91免费版在线观看 | 日日弄天天弄美女bbbb | 久久久网页 | 国产aa精品| 国产高清精品在线 | 国产精品区免费视频 | 二区三区毛片 | 久草电影在线 | 青青河边草免费视频 | 日韩成人免费观看 | 91丨九色丨国产女 | 免费在线观看日韩欧美 | 国产专区一 | 一区二区三区视频网站 | 亚洲涩涩一区 | 国产精品久久久久久久久搜平片 | 超碰成人免费电影 | 日韩精品免费一区二区在线观看 | 国产日本亚洲 | 超碰成人网 | 97av视频| 欧美一区二区在线刺激视频 | 日日干视频 | 日日爱网址 | 精品国产一区在线观看 | 91精品国产九九九久久久亚洲 | 永久免费毛片在线观看 | 免费看久久久 | 中文字幕九九 | 日日干av | 婷婷伊人五月 | 国产精品久久久久久久久免费 | 国产不卡在线播放 | 国产精品99久久久久久有的能看 | 欧美在线视频免费 | 国产一区在线精品 | 国产亚洲精品久久久久久无几年桃 | 国产成人99av超碰超爽 | 国产无遮挡又黄又爽在线观看 | 在线观看视频一区二区三区 | 日韩在线视频精品 | 色婷婷九月| 亚洲精品裸体 | 欧美日韩高清一区二区 | 91精品在线观看入口 | 丁香花在线视频观看免费 | 高清av影院 | 国产精品一区二区久久精品爱涩 | av一级一片 | 国产福利91精品一区二区三区 | 五月婷婷丁香激情 | 黄污网站在线观看 | 国产成人福利在线观看 | 天天操夜夜操天天射 | 99久久99久久精品 | 三级av免费观看 | 亚洲理论电影 | 色综合久久五月 | 国产精品岛国久久久久久久久红粉 | 欧美人交a欧美精品 | www黄在线 | 免费日韩 精品中文字幕视频在线 | 国产黄色大全 | 精品一区二区免费 | 成人精品一区二区三区电影免费 | 日韩一区二区三区在线看 | 99麻豆久久久国产精品免费 | 在线看av的网址 | 中文字幕在线看视频国产 | 日韩 在线| 久久国产精品一区二区三区 | 午夜av一区 | 免费福利在线视频 | 久草网在线观看 | 欧美精品久久天天躁 | 99热这里只有精品1 av中文字幕日韩 | 三级视频片 | 97国产超碰 | 九九视频精品免费 | 免费看在线看www777 | 国产视频资源在线观看 | 久久精品超碰 | 国产成人精品av | 五月婷婷天堂 | 日本高清免费中文字幕 | 国产精品永久久久久久久久久 | 久久免费公开视频 | 午夜精品久久久久久久99水蜜桃 | 日韩高清在线一区二区三区 | 久久久久女人精品毛片九一 | 国产精品专区在线观看 | 青草视频免费观看 | 国产精品欧美久久久久久 | 欧美激情视频一区二区三区免费 | 亚洲日本va午夜在线电影 | 亚洲三级在线 | 在线成人免费av | 97碰碰碰| 国产精品 中文字幕 亚洲 欧美 | 中文av在线播放 | 精品久久1 | 亚州成人av在线 | 国产精品永久 | 久久tv视频| 国产精品1区 | 久久人人爽人人爽人人 | 日本色小说视频 | 亚洲国产999 | 欧美另类v | 久久久精品视频网站 | 91亚洲国产 | 国产精品久久久久久久久久直播 | 精品国产一区二区三区噜噜噜 | 久久久久久久国产精品 | 日韩精品中文字幕在线播放 | 免费看的黄网站软件 | 天天操比 | 国产在线综合视频 | 成人国产一区 | 麻豆精品国产传媒 | 天天操天天射天天爽 | 免费观看性生交大片3 | 91精品国产欧美一区二区成人 | 奇米影视777四色米奇影院 | 亚洲在线网址 | 成人黄大片 | 麻豆一区在线观看 | 九九免费在线视频 | 国产视频丨精品|在线观看 国产精品久久久久久久久久久久午夜 | 免费黄色a网站 | 福利av在线| 99在线精品免费视频九九视 | 欧美精品国产综合久久 | 国产美女精品视频 | 天天搞天天 | 高清中文字幕av | 丝袜美腿av| 国产成人久久 | a在线一区 | 很黄很黄的网站免费的 | 成人av影视在线 | 91最新地址永久入口 | 香蕉在线视频观看 | 一区二区三区在线视频观看58 | 亚洲欧洲国产日韩精品 | 中文字幕美女免费在线 | 激情影音 | 精品国产理论片 | 丁香花五月 | 久久精品人人做人人综合老师 | 九九九九精品九九九九 | 天天曰视频 | 久久三级视频 | 日韩电影黄色 | 亚洲影院国产 | 久久国产精品99国产 | 久久精品欧美一区二区三区麻豆 | 天天曰夜夜爽 | 91亚洲精品乱码久久久久久蜜桃 | 日韩欧美电影在线 | 久久综合视频网 | 手机成人免费视频 | 丁香久久久| 九色视频网址 | 久久久久久久久影视 | 色com网| av免费看在线 | 欧美色图狠狠干 | 欧美一级淫片videoshd | 国产精品福利在线观看 | 99视频精品全国免费 | 韩日色视频 | 国内成人精品2018免费看 | 欧美日韩精品在线观看视频 | 欧美一级久久久 | 一区二区三区av在线 | 欧美 日韩 国产 中文字幕 | 国产精品久久久久毛片大屁完整版 | 日韩精品不卡在线观看 | 久久精品精品电影网 | 婷婷免费视频 | 国产91大片| 狠狠狠色丁香综合久久天下网 | 免费激情网 | 97精品国自产拍在线观看 | 国产在线观看免 | 999电影免费在线观看2020 | 久久精品一区八戒影视 | 中文字幕国语官网在线视频 | 狠狠操.com| 91新人在线观看 | 亚洲人人精品 | 色综合天天综合在线视频 | 色综合天天综合在线视频 | av免费网站观看 | 日韩超碰在线 | 美女视频黄,久久 | 免费三级黄色片 | 日本久久成人中文字幕电影 | 亚洲专区欧美专区 | 国产理论片在线观看 | 99re国产视频 | 美女av在线免费 | 亚洲一区二区视频在线 | 人人涩 | 日韩高清免费无专码区 | 三级av在线 | 亚洲国内精品视频 | 看v片| 黄色a在线 | 精品国产午夜 | 久久久男人的天堂 | 99在线看 | 香蕉看片| 亚洲一区视频在线播放 | 成人国产精品久久久春色 | 91热爆在线观看 | 久久免费福利 | 大片网站久久 | 国产精品国产亚洲精品看不卡15 | 精品不卡av | 日韩成人黄色 | 国产高清小视频 | 久久精品2 | 国内精品久久久久影院日本资源 | 在线观看中文字幕av | 五月激情六月丁香 | 日韩电影在线观看一区 | 99 色| 一区二区三区四区五区在线 | 手机看片福利 | 91一区二区三区在线观看 | 97人人超碰在线 | a极黄色片| 亚洲精品玖玖玖av在线看 | 91免费视频网站在线观看 | 亚洲国产成人久久 | 色九九在线| 亚洲蜜桃av | 91高清视频在线 | 极品久久久久久久 | 精品久久久久久久久久久久久久久久 | 欧美日本不卡视频 | 91超级碰碰 | 天天艹日日干 | 99色视频| 米奇狠狠狠888 | 亚洲精品高清一区二区三区四区 | 一区二区电影在线观看 | 久久久久久免费视频 | 超碰97在线人人 | 狠狠做六月爱婷婷综合aⅴ 日本高清免费中文字幕 | 亚洲a成人v | 久草在线手机视频 | 午夜精品999 | 免费视频xnxx com | 国产69熟 | 四虎影视精品永久在线观看 | 在线中文字幕网站 | 免费日韩 精品中文字幕视频在线 | www.久久婷婷| 欧美一区中文字幕 | 涩涩网站在线观看 | 日批网站在线观看 | 日韩视频一区二区三区在线播放免费观看 | 超碰在线cao | 久久深夜福利免费观看 | 成人久久18免费网站 | 中文字幕在线免费看 | 91视频久久久久久 | 久久9999久久免费精品国产 | 玖玖在线精品 | 久久黄色影院 | 美女精品在线 | 啪啪凸凸| 成人av高清在线 | 久久成人在线视频 | 色婷婷福利 | 久久免费观看少妇a级毛片 久久久久成人免费 | 国产精品久久久久久久久久久久午夜 | 精品在线二区 | 一区二区三区在线视频111 | 久久视频一区二区 | 国产清纯在线 | 麻豆综合网| 久草男人天堂 | 在线精品一区二区 | 婷婷激情五月综合 | 深夜国产福利 | 午夜色性片 | 青草视频在线播放 | 992tv人人网tv亚洲精品 | 9999精品免费视频 | 国产99久久 | 综合在线亚洲 | 91av电影在线 | 99久久国产免费看 | 91视频 - x99av | 亚洲伊人网在线观看 | 亚洲精品综合在线观看 | 国产精品色在线 | 天天天操天天天干 | 久久免费看 | 久久99精品国产麻豆婷婷 | 粉嫩av一区二区三区四区在线观看 | 欧美人体xx| 成人久久18免费网站 | 色成人亚洲 | av综合网址| 久久久久久久久久伊人 | 高清精品在线 | 国产偷v国产偷∨精品视频 在线草 | www在线免费观看 | 国产精品不卡 | 日本久久影视 | 欧美成人精品欧美一级乱黄 | 精品在线播放视频 | 久草视频免费看 | 91传媒激情理伦片 | 日韩v欧美v日本v亚洲v国产v | 91av欧美 | 视频在线观看入口黄最新永久免费国产 | 午夜av在线播放 | 欧美成人a在线 | 国产日本三级 | 美女网站免费福利视频 | 在线免费中文字幕 | 色www.| 黄色.com| 亚洲国产资源 | 日韩免费三区 | 99精品在线视频观看 | 久久国产露脸精品国产 | 日韩精品视频第一页 | 伊人手机在线 | www黄色av | 免费观看成年人视频 | 综合色在线 | 一区二区三区高清不卡 | 亚洲黄色在线观看 | av免费网站在线观看 | 亚洲最新av网址 | 国产又粗又猛又色又黄视频 | 综合久久网 | 欧美日韩精品在线观看 | 国产中文视 | 91豆花在线 | 一区二区三区免费在线观看视频 | 亚洲综合在线观看视频 | 九色91av| 国产69精品久久久久久 | 国产日韩在线一区 | 亚洲精品免费播放 | 在线天堂视频 | 国产精品18久久久 | 日韩av播放在线 | 国产精品系列在线观看 | 欧美国产高清 | 国产三级午夜理伦三级 | 精品视频在线免费观看 | 成人试看120秒 | 在线免费黄色av | 国产高清视频免费 | 国产成人一区二区在线观看 | 国产99久久久国产精品免费看 | 夜夜视频| 在线观看中文字幕2021 | 色www免费视频 | 国内成人综合 | 国产1级毛片 | 亚洲国产精品小视频 | 国产首页 | 久久无码精品一区二区三区 | 久久国产91| 亚洲成人免费 | 三级av在线播放 | 久草手机视频 | 久久久久国产精品免费网站 | 色婷av | 在线观看香蕉视频 | 丁香综合五月 | 亚洲视频999 | a√天堂资源| 国产 在线 高清 精品 | 91麻豆精品国产91久久久更新时间 | 成人黄色小说在线观看 | 国产精品不卡在线播放 | 91高清免费观看 | 天天操操| 91大神在线观看视频 | 久久网站av | 久香蕉| 国产精品1区2区3区在线观看 | 国产伦精品一区二区三区无广告 | 中文字幕中文字幕在线中文字幕三区 | 久久99国产精品免费 | 香蕉网在线观看 | 韩国一区二区三区在线观看 | 亚洲色图 校园春色 | 六月丁香在线观看 | 国产视频在线观看一区 | 亚洲一区二区三区在线看 | 国产精品一区二区免费在线观看 | 欧美日韩在线播放一区 | 黄色a在线 | 又黄又爽又色无遮挡免费 | 天天操天天操天天操天天操天天操 | 日韩av免费在线看 | 精品久久一级片 | 天天干天天操天天拍 | 国产午夜一区二区 | 久久嗨 | 午夜在线免费观看视频 | 九九激情视频 | 综合网天天 | 97色在线视频 | 婷婷亚洲综合 | 国产精品久久久久久久久久久久 | 中文字幕在线观看视频一区二区三区 | 国产欧美精品一区二区三区四区 | 亚洲国产精品久久久久婷婷884 | 丁香婷婷激情网 | 欧美一级电影片 | 成人黄色在线 | 国产精品美女 | 欧美一级看片 | 亚洲三级视频 | 久热精品国产 | 91成人午夜| 久久99精品久久久久蜜臀 | 又爽又黄又刺激的视频 | a午夜电影 | 国产一区精品在线 | 成人免费在线播放 | 手机成人av| 欧美va在线观看 | 99热最新地址 | 91麻豆精品91久久久久同性 | 国产精品18久久久久白浆 | 国产午夜在线 | 免费成人av网站 | 亚洲一二三区精品 | 婷婷色资源 | 久久九九视频 | 91探花在线| 国产老妇av | 国产特级毛片aaaaaa毛片 | 五月婷婷一区二区三区 | 亚洲高清视频在线观看 | 日韩欧美高清视频在线观看 | 久草9视频| 99热999| 黄色a在线观看 | 日韩午夜在线播放 | 韩国在线视频一区 | 色福利网 | 天天鲁一鲁摸一摸爽一爽 | 中文字幕一区在线观看视频 | 天天摸天天操天天爽 | 久久精品国产免费看久久精品 | www操操 | 99热高清| 免费三级黄色 | 精品国产aⅴ麻豆 | 亚洲精品在线免费 | 国产综合视频在线观看 | 亚洲区色| 欧美aⅴ在线观看 | 国产精品中文字幕在线播放 | 又爽又黄又无遮挡网站动态图 | 久久成人一区二区 | 奇米影视8888在线观看大全免费 | 麻豆视频免费在线 | 999视频在线播放 | 色综合久久综合 | 成 人 黄 色 视频免费播放 | 亚洲资源在线 | 久久精品视频在线免费观看 | 国产精品去看片 | 欧洲精品在线视频 | 探花视频在线观看免费 | 成年性视频 | 丁香六月激情婷婷 | 岛国大片免费视频 | 91精品国自产在线 | 国产男女爽爽爽免费视频 | 中文字幕丝袜 | 久久精品成人欧美大片古装 | www91在线观看 | 国产97视频在线 | 国产精品无av码在线观看 | 久久好看免费视频 | 97在线资源 | 夜夜躁日日躁 | 国产麻豆视频 | 午夜91在线 | 亚洲国产精品500在线观看 | 国产91综合一区在线观看 | 91成年人在线观看 | 日韩美在线 | 三级a视频 | 国产精品乱看 | 国产亚洲日本 | 啪啪免费试看 | 久草国产精品 | 久久久网址| 天堂在线视频中文网 | 不卡av电影在线 | 中文字幕一区二区三区精华液 | 久久精品一区二 | 中文字幕在线观看第三页 | 日韩黄色一区 | 激情综合色播五月 | 97在线播放| 成人在线观看av | 99精品视频免费全部在线 | 伊人五月天婷婷 | 亚洲资源片| 91色一区二区三区 | 欧美日韩高清一区二区三区 | 久久小视频| 亚洲国产精品女人久久久 | 日韩在线观看视频一区二区三区 | 亚洲黄在线观看 | 91九色视频在线观看 | 黄色国产在线观看 | 2019免费中文字幕 | 国产成人精品亚洲日本在线观看 | 99色亚洲| 午夜成人免费影院 | 国产精品一区久久久久 | 天天色天天爱天天射综合 | 99草视频 | 一区二区三区精品在线 | 亚洲国产精品人久久电影 | 亚洲一区二区三区四区在线视频 | 91精品久久久久久综合乱菊 | 日韩精品一区二区三区在线播放 | 欧美极品一区二区三区 | 美女免费视频黄 | 三级动图 | 久久黄色美女 | 欧美在线资源 | 久久伦理电影网 | 91亚洲精品久久久蜜桃网站 | 91免费观看视频网站 | 色资源网在线观看 | 五月婷婷一级片 | 国产一区在线视频播放 | 在线国产黄色 | a久久久久 | 日韩精品字幕 | 中文字幕精品www乱入免费视频 | 久久久国产精品成人免费 | 久久综合狠狠 | 99热这里是精品 | 美女在线免费观看视频 | 国产理论免费 | 中文字幕在线一二 | 国产成人精品免高潮在线观看 | 天天爽天天摸 | av性网站| 久久久免费精品国产一区二区 | 日韩在线一区二区免费 | 色com网| 婷婷激情小说网 | 国产精品视频永久免费播放 | 国产精品欧美久久久久三级 | 久久久精品日本 | 天天操天天操天天操天天操天天操天天操 | 日韩国产高清在线 | 99精品视频免费观看视频 | 九九热在线免费观看 | 国产成人精品一区在线 | 日韩激情免费视频 | 91av视频免费在线观看 | 国产玖玖在线 | 中文字幕国产一区二区 | 丰满少妇在线观看网站 | 九九激情视频 | 国产黄色播放 | 欧美在线一二 | 丁香六月在线观看 | 九九涩涩av台湾日本热热 | 国产精品永久免费视频 | 999精品| 最新亚洲视频 | 免费在线一区二区 | 欧美在线99| 999久久久精品视频 日韩高清www | 最新久久免费视频 | 亚洲少妇激情 | 亚洲黄色三级 | 91丨九色丨国产在线 | 激情欧美网 | 中文字幕在线影院 | 色瓜| 久久图| 麻豆一二三精选视频 | 美女黄频在线观看 | 成人av播放 | 精品国内自产拍在线观看视频 | 精品一区二区电影 | 韩国在线视频一区 | 天天艹天天 | 91色影院 | 色欧美视频 | av在线免费播放网站 | 亚洲精品18日本一区app | 精品视频专区 | 日韩精品在线视频 | 婷婷资源站 | 2023亚洲精品国偷拍自产在线 | www91在线观看 | 高清不卡毛片 | 啪啪凸凸 | 丝袜精品视频 | 国内精品久久久精品电影院 | 国产69精品久久99的直播节目 | 久艹在线免费观看 | 精品国产乱码久久久久久三级人 | 国产永久免费高清在线观看视频 | 日韩欧美电影网 | 人人射| 免费成人黄色av | 国产一级免费在线观看 | 91精品婷婷国产综合久久蝌蚪 | 国产精品美女免费视频 | 97免费在线观看 | 一区二区激情视频 | 久久8精品 | 91免费观看网站 | 不卡精品视频 | 亚洲精品乱码久久久久久蜜桃不爽 | 91精彩视频在线观看 | 久久这里只有精品9 | 日韩在线观看 | 视频在线一区 | 免费黄色网址大全 | 综合激情网 | 国产久草在线 | 91日韩国产 | 天天操网站| 婷婷久久综合九色综合 | 免费看污黄网站 | 免费观看版 | 天天干天天操天天操 | 久久精品福利 | 91精品在线麻豆 | 婷婷久久久久 | 亚洲成年人在线播放 | 97香蕉久久超级碰碰高清版 | av九九九| 久久理论片 | 国产一区二区免费在线观看 | 黄色av免费看 | 久久久久国产成人免费精品免费 | 国产福利在线免费 | 久久久高清 | 男女全黄一级一级高潮免费看 | 亚洲国产网站 | 日韩天堂在线观看 | 久久久久激情视频 | 二区精品视频 | 午夜视频免费 | 中文久草 | 国内精品久久天天躁人人爽 | 国产不卡网站 | 久久久免费看片 | 中国一级特黄毛片大片久久 | 天天色官网| 日韩v在线 | 日本三级中文字幕在线观看 | 国产青青青 | 色姑娘综合天天 | 一级黄毛片 | 久久国内免费视频 | 不卡的av在线播放 | 欧美激情精品 | 免费看黄在线观看 | 97超碰人人澡人人爱 | 视频三区在线 | 国产免费xvideos视频入口 | 激情五月亚洲 | 黄色免费高清视频 | 免费网站在线观看人 | www.久艹| 久av在线| 97超碰资源网 | 免费人人干 | 久久九九久久 | 日日夜夜狠狠干 | 日韩高清精品一区二区 | 久久99久久99精品免观看粉嫩 | 最新精品国产 | 国产香蕉久久 | 国产免费xvideos视频入口 | 久久婷亚洲五月一区天天躁 | 天天操天天干天天玩 | 国产精品嫩草69影院 | 99精品欧美一区二区蜜桃免费 | 国产成人精品亚洲 | 免费观看国产精品 | 爱爱一区 | 亚洲免费公开视频 | 色综合在 | 日本在线观看一区二区三区 | 日韩欧美一区二区三区免费观看 | 国产色视频一区 | 国精产品999国精产 久久久久 | 亚洲综合视频在线 | 中文字幕在线视频一区 | 免费日韩一级片 | 韩日成人av | 综合亚洲视频 | 98超碰在线观看 | 精品国产乱码久久久久久久 | 国产精品观看在线亚洲人成网 | 久久久五月婷婷 | 亚洲国产精品99久久久久久久久 | 亚洲精品毛片一级91精品 | 国产成人精品电影久久久 | 久久69精品久久久久久久电影好 | 国产精品一区二区三区免费看 | 久久综合成人网 | 国产首页 | 伊在线视频 | 国产剧情一区二区在线观看 | 一级性av| 欧美日韩在线视频观看 | 人成在线免费视频 | 日韩免费一区二区在线观看 | 天堂视频一区 | 久久黄色影院 | 午夜精品一区二区三区在线视频 | 日本xxxxav| 97精品免费视频 | av福利网址导航 | 欧美精品久久久久久久亚洲调教 | 久久精品国产99 | 玖玖精品在线 | 亚洲精品理论 | 毛片1000部免费看 | 亚洲免费高清视频 | 日本99干网 | 亚洲一区美女视频在线观看免费 | 激情婷婷综合 | 日韩在线观看视频一区二区三区 | 天天操天天爱天天干 | 色综合天天色 | 欧美一区二区精品在线 | 五月天九九 | 欧美精品成人在线 | 日韩精品专区 | 日韩精品一区二区三区在线视频 | 亚洲 欧美 91 | 亚洲欧美在线综合 | 欧美激精品 | 国产在线视频一区二区三区 | 国产精品久久久久久久av大片 | 久久免费视频2 | 丁香婷婷久久久综合精品国产 | 91视频在线免费下载 | 久久视频国产精品免费视频在线 | 欧洲精品码一区二区三区免费看 | 欧美日韩一区二区三区免费视频 | 99精品视频在线观看免费 | 激情视频国产 | 久久久在线 | 久久在线| 成人毛片一区二区三区 | 日韩欧美在线免费 | 久久精品8 | 国产码电影 | 欧美日韩xxxxx | 国产九九九九九 | 久久伊人精品一区二区三区 | 日本黄色免费看 | av日韩在线网站 | 成人午夜免费福利 | 亚洲成a人片在线观看网站口工 | 色婷婷亚洲 | av最新资源| 天天搞天天干天天色 | 99免费精品| 久久国产精品久久精品国产演员表 | 一区二区精 | 99久久99精品 | 日韩av中文字幕在线免费观看 | 国产伦精品一区二区三区… | 91欧美精品 | 国产精品久久久久影院 | 国产精品成人自产拍在线观看 | 国产黄| 国产在线国偷精品产拍 | 超碰在线公开免费 | 亚洲国内精品视频 | 久久精品79国产精品 | www.天天综合 | 一区二区在线电影 | 91最新地址永久入口 | 91九色网站 | 国产在线观看午夜 | 欧美日韩午夜爽爽 | 欧美极度另类 | 99精品在线视频观看 | 日韩理论电影在线 | 九九热在线播放 | 在线看一区二区 | 久草在线视频新 | 97精品国产一二三产区 | 中文字幕第一页av | 国产精品永久免费在线 | 久久精品国亚洲 | 99中文字幕 | 久久久久亚洲精品国产 | 久草在线欧美 | 国产精品第一视频 | 精品国产精品一区二区夜夜嗨 | 丁香九月婷婷综合 | 久久99九九99精品 | 99色网站| 国产黄色片一级三级 | 三三级黄色片之日韩 | 少妇高潮冒白浆 | 欧美日韩p片 | 美女网站视频免费黄 | 日韩高清 一区 | 国产精品视频资源 | 99免费在线 | 手机在线中文字幕 | 一级片观看 | 日本69hd | 免费福利片2019潦草影视午夜 | 日b视频在线观看网址 | 国产精品久久久久久久久久白浆 | 免费看的av片 | 久久免费国产精品1 | 成人在线你懂得 | 一区二区三区在线免费 | 精品国产电影一区 | 久久99精品久久久久久秒播蜜臀 | av在线看网站 | 九九九在线 | 久久精品8 | 一级片在线 | 九月婷婷综合网 | 成年人黄色免费看 | 久久成电影 | 久久成人精品电影 | av电影久久 | 欧美一级久久 | 美女黄视频免费看 | 国产精品久久一区二区无卡 | 狠狠操夜夜 | 日日色综合| 日韩在线视频线视频免费网站 | 国产精品久久久久久久午夜 | 久草视频在线资源 | 色姑娘综合网 | 亚洲国产精品视频 | 国产免费又爽又刺激在线观看 | 香蕉在线视频播放网站 | 天天激情天天干 | www.91av在线| 九九九热精品免费视频观看网站 | 四川妇女搡bbbb搡bbbb搡 | 国产精品毛片一区二区 | 国产伦精品一区二区三区四区视频 | 久久久在线免费观看 | 十八岁以下禁止观看的1000个网站 | 成人国产精品一区二区 | 日韩av电影免费观看 | 天堂av在线中文在线 | 国产综合久久 | 日韩欧美在线视频一区二区三区 | 欧美视频日韩视频 | 一本一道久久a久久精品 | 天天操人人干 | 欧洲亚洲精品 | 六月丁香激情综合色啪小说 | 国产黄网站在线观看 | 99福利片| 久久99亚洲精品 | 国产精品99久久久久久有的能看 | 日韩欧美不卡 | av久久在线| 免费看十八岁美女 | 午夜视频在线观看一区二区三区 | 国产精品亚洲精品 | 97国产在线 | 久久久免费观看 | 亚洲v欧美v国产v在线观看 | 色噜噜在线观看 | 精品欧美在线视频 | 日韩欧美xxx | 国产精品国产三级国产专区53 | 日本中文乱码卡一卡二新区 | 91在线麻豆 | 亚洲欧美国产精品久久久久 | 日日夜夜精品视频天天综合网 | 黄色网在线播放 | 国产在线成人 | 四虎国产视频 | 国产精品欧美久久久久天天影视 | 人人射人人澡 | 久久不射电影院 | 欧美亚洲国产精品久久高清浪潮 | 五月天电影免费在线观看一区 | 88av色| 久久久精品福利视频 | 黄色91免费观看 | 久久久久国| 天天操天天插 | 五月婷久久| 色橹橹欧美在线观看视频高清 | 亚洲精品免费看 | 波多野结衣视频一区二区三区 | 久久99精品久久只有精品 | 国产高清网站 | 中文av在线免费观看 | 欧美精品做受xxx性少妇 | 久久精品久久久精品美女 | 久久久久视 | 91禁在线观看| 中文字幕久久精品一区 | 欧美精品v国产精品 | 亚欧洲精品视频在线观看 | 亚洲人在线视频 | 精品999国产 | 超级av在线 | 久久久91精品国产一区二区三区 | 天堂网一区二区 | 日韩v在线91成人自拍 | 午夜影院一级 | 奇米7777狠狠狠琪琪视频 |