欧美一区二区三区老妇人-欧美做爰猛烈大尺度电-99久久夜色精品国产亚洲a-亚洲福利视频一区二区

java類(lèi)加載器的原理是什么

這篇文章給大家分享的是有關(guān)java類(lèi)加載器的原理是什么的內(nèi)容。小編覺(jué)得挺實(shí)用的,因此分享給大家做個(gè)參考,一起跟隨小編過(guò)來(lái)看看吧。

創(chuàng)新互聯(lián)公司主要為客戶提供服務(wù)項(xiàng)目涵蓋了網(wǎng)頁(yè)視覺(jué)設(shè)計(jì)、VI標(biāo)志設(shè)計(jì)、成都全網(wǎng)營(yíng)銷(xiāo)、網(wǎng)站程序開(kāi)發(fā)、HTML5響應(yīng)式網(wǎng)站建設(shè)手機(jī)網(wǎng)站開(kāi)發(fā)、微商城、網(wǎng)站托管及成都網(wǎng)站維護(hù)、WEB系統(tǒng)開(kāi)發(fā)、域名注冊(cè)、國(guó)內(nèi)外服務(wù)器租用、視頻、平面設(shè)計(jì)、SEO優(yōu)化排名。設(shè)計(jì)、前端、后端三個(gè)建站步驟的完善服務(wù)體系。一人跟蹤測(cè)試的建站服務(wù)標(biāo)準(zhǔn)。已經(jīng)為成都火鍋店設(shè)計(jì)行業(yè)客戶提供了網(wǎng)站營(yíng)銷(xiāo)服務(wù)。

一,Java類(lèi)的加載、鏈接與初始化

1,加載:查找并加載類(lèi)的二進(jìn)制數(shù)據(jù)

? 通過(guò)一個(gè)類(lèi)的全限定名來(lái)獲取定義此類(lèi)的二進(jìn)制字節(jié)流
? 將這個(gè)字節(jié)流所代表的靜態(tài)存儲(chǔ)結(jié)構(gòu)轉(zhuǎn)化為方法區(qū)的運(yùn)行時(shí)數(shù)據(jù)結(jié)構(gòu)
? 在內(nèi)存中生成一個(gè)代表這個(gè)類(lèi)的java.lang.Class類(lèi)對(duì)象,作為方法區(qū)這個(gè)類(lèi)的各種數(shù)據(jù)的訪問(wèn)入口。

2,鏈接

驗(yàn)證:

確保被加載類(lèi)的正確性

準(zhǔn)備:

為類(lèi)的靜態(tài)變量分配內(nèi)存,并將其初始化為默認(rèn)值

解析:

把類(lèi)中的符號(hào)引用轉(zhuǎn)化為直接引用

3,初始化

為類(lèi)的靜態(tài)變量賦予正確的初始值

二,JVM加載類(lèi)的主要方式

  • 從本地系統(tǒng)中直接加載
  • 通過(guò)網(wǎng)絡(luò)下載.class文件
  • 從zip,jar等歸檔文件中加載.class文件
  • 從專(zhuān)有數(shù)據(jù)庫(kù)中提取.class文件
  • 講Java的源文件動(dòng)態(tài)編譯為.class

三,JVM加載類(lèi)的種類(lèi)及功能

JVM通過(guò)CLassLoader(類(lèi)加載器)來(lái)動(dòng)態(tài)加載某個(gè)class文件到內(nèi)存當(dāng)中的。ClassLoader總共分為以下四種:

1,根(Bootstrap)類(lèi)加載器

該類(lèi)加載器沒(méi)有父加載器,他負(fù)責(zé)加載虛擬機(jī)的核心類(lèi)庫(kù),如java.lang.*等。根類(lèi)加載器從系統(tǒng)屬性sun.boot.class.path所指定的目錄中加載類(lèi)庫(kù)。跟類(lèi)加載器的實(shí)現(xiàn)依賴于底層操作系統(tǒng)屬于虛擬機(jī)實(shí)現(xiàn)的一部分,它并沒(méi)有繼承java.lang.ClassLoader類(lèi)。

2,擴(kuò)展(Extension)類(lèi)加載器

它的父類(lèi)加載器為根類(lèi)加載器。它從java.ext.dirs系統(tǒng)屬性所指定的目錄中加載類(lèi)庫(kù),或者從JDK安裝目錄的jrelibext子目錄(擴(kuò)展目錄加載類(lèi)庫(kù)),如何用戶創(chuàng)建的JAR問(wèn)你件放在這個(gè)目錄下也會(huì)被自動(dòng)由擴(kuò)展類(lèi)加載器加載。擴(kuò)展類(lèi)加載器是純Java類(lèi),是java.lang.ClassLoader類(lèi)的子類(lèi)。

3,系統(tǒng)類(lèi)加載器

也稱(chēng)為應(yīng)用類(lèi)加載器,它的父加載器為擴(kuò)展類(lèi)加載器。他從環(huán)境變量classpath或者系統(tǒng)屬性java.class.path所指定的目錄中加載類(lèi),它是用戶自定義的類(lèi)加載器的默認(rèn)父加載器。系統(tǒng)類(lèi)加載器是純Java類(lèi),是java.lang.ClassLoader類(lèi)的子類(lèi)。
除了以上虛擬機(jī)自帶的加載器,用戶也可以繼承java.lang.ClassLoader類(lèi)實(shí)現(xiàn)自定義加載器。

四,類(lèi)加載器的原理

1,原理介紹

ClassLoader使用的是雙親委托模型來(lái)搜索類(lèi)的,每個(gè)ClassLoader實(shí)例都有一個(gè)父類(lèi)加載器的引用(不是繼承的關(guān)系,是一個(gè)包含的關(guān)系),虛擬機(jī)內(nèi)置的類(lèi)加載器(Bootstrap ClassLoader)本身沒(méi)有父類(lèi)加載器,但可以用作其它ClassLoader實(shí)例的的父類(lèi)加載器。當(dāng)一個(gè)ClassLoader實(shí)例需要加載某個(gè)類(lèi)時(shí),它會(huì)試圖親自搜索某個(gè)類(lèi)之前,先把這個(gè)任務(wù)委托給它的父類(lèi)加載器,這個(gè)過(guò)程是由上至下依次檢查的,首先由最頂層的類(lèi)加載器Bootstrap ClassLoader試圖加載,如果沒(méi)加載到,則把任務(wù)轉(zhuǎn)交給Extension ClassLoader試圖加載,如果也沒(méi)加載到,則轉(zhuǎn)交給App ClassLoader 進(jìn)行加載,如果它也沒(méi)有加載得到的話,則返回給委托的發(fā)起者,由它到指定的文件系統(tǒng)或網(wǎng)絡(luò)等URL中加載該類(lèi)。如果它們都沒(méi)有加載到這個(gè)類(lèi)時(shí),則拋出ClassNotFoundException異常。否則將這個(gè)找到的類(lèi)生成一個(gè)類(lèi)的定義,并將它加載到內(nèi)存當(dāng)中,最后返回這個(gè)類(lèi)在內(nèi)存中的Class實(shí)例對(duì)象。

java類(lèi)加載器的原理是什么

2,使用雙親委托模型的原因

因?yàn)檫@樣可以避免重復(fù)加載,當(dāng)父親已經(jīng)加載了該類(lèi)的時(shí)候,就沒(méi)有必要子ClassLoader再加載一次。考慮到安全因素,我們?cè)囅胍幌拢绻皇褂眠@種委托模式,那我們就可以隨時(shí)使用自定義的String來(lái)動(dòng)態(tài)替代java核心api中定義的類(lèi)型,這樣會(huì)存在非常大的安全隱患,而雙親委托的方式,就可以避免這種情況,因?yàn)镾tring已經(jīng)在啟動(dòng)時(shí)就被引導(dǎo)類(lèi)加載器(Bootstrcp ClassLoader)加載,所以用戶自定義的ClassLoader永遠(yuǎn)也無(wú)法加載一個(gè)自己寫(xiě)的String,除非你改變JDK中ClassLoader搜索類(lèi)的默認(rèn)算法。

3,如何判斷兩個(gè)class相同

JVM在判定兩個(gè)class是否相同時(shí),不僅要判斷兩個(gè)類(lèi)名是否相同,而且要判斷是否由同一個(gè)類(lèi)加載器實(shí)例加載的。只有兩者同時(shí)滿足的情況下,JVM才認(rèn)為這兩個(gè)class是相同的。就算兩個(gè)class是同一份class字節(jié)碼,如果被兩個(gè)不同的ClassLoader實(shí)例所加載,JVM也會(huì)認(rèn)為它們是兩個(gè)不同class。比如網(wǎng)絡(luò)上的一個(gè)Java類(lèi)org.classloader.simple.NetClassLoaderSimple,javac編譯之后生成字節(jié)碼文件NetClassLoaderSimple.class,ClassLoaderA和ClassLoaderB這兩個(gè)類(lèi)加載器并讀取了NetClassLoaderSimple.class文件,并分別定義出了java.lang.Class實(shí)例來(lái)表示這個(gè)類(lèi),對(duì)于JVM來(lái)說(shuō),它們是兩個(gè)不同的實(shí)例對(duì)象,但它們確實(shí)是同一份字節(jié)碼文件,如果試圖將這個(gè)Class實(shí)例生成具體的對(duì)象進(jìn)行轉(zhuǎn)換時(shí),就會(huì)拋運(yùn)行時(shí)異常java.lang.ClassCaseException,提示這是兩個(gè)不同的類(lèi)型。

4,常用的方法

(1) loadClass方法

ClassLoader.loadClass()是ClassLoader的入口點(diǎn)。該方法的定義如下:

Class loadClass(String name,boolean resolve);

name是加載的類(lèi)的名稱(chēng),resolve是告訴方法是不中需要解析類(lèi)PS:并不是所有的類(lèi)都需要解析,如果JVM只想知道這個(gè)類(lèi)是否存在或找出該類(lèi)的超類(lèi),那么就不需要解析該類(lèi)

(2) defineClass方法

defineClass方法接受由原始字節(jié)組成的數(shù)組,并把它轉(zhuǎn)換成Class的對(duì)象。原始數(shù)組包含如從文件系統(tǒng)或網(wǎng)絡(luò)裝入的數(shù)據(jù)。defineClass管理JVM的許多復(fù)雜的實(shí)現(xiàn)層面——它把字節(jié)碼分析成運(yùn)行時(shí)數(shù)據(jù)結(jié)構(gòu)、校驗(yàn)有效性等,因?yàn)閐efineClass方法被標(biāo)記成final的,所以不能覆蓋它。

(3) findSytemClass方法

findSystemClass方法就是查找本地類(lèi)Class文件,然后裝入

(4) resolveClass方法

我們?cè)谡{(diào)用編寫(xiě)自己的loadClass方法的時(shí)候可以調(diào)用resolveClass方法來(lái)獲得resolve參數(shù)

(5) findLoadedClass方法

在調(diào)用loadClass方法之前可以調(diào)用改方法來(lái)查看地ClassLoader是否已經(jīng)裝入了這個(gè)類(lèi),這樣可以避免重新裝入這個(gè)類(lèi)

(6) findClass方法

在loadClass默認(rèn)實(shí)現(xiàn)調(diào)用這個(gè)新方法。findClass的用途包含classLoader的所有特殊代碼,而無(wú)須復(fù)制其他代碼

(7) getSystemClassLoader方法

在如果覆蓋findClass或loadClass,getSystemClassLoader能以實(shí)際的ClassLoader對(duì)象訪問(wèn)系統(tǒng)ClassLoader(而不是固定地從findSystemClass調(diào)用它)。為了將類(lèi)請(qǐng)求委托給父類(lèi)ClassLoader,這個(gè)新方法允許ClassLoader獲取它的父類(lèi)ClassLoader.當(dāng)使用特殊方法,定制的ClassLoader不能找到類(lèi)時(shí),可以使用這種方法。

  父類(lèi)ClassLoader被定義成創(chuàng)建該ClassLoader所包含代碼的對(duì)象的ClassLoader.

(8) forName方法

在Class類(lèi)中有一個(gè)靜態(tài)方法forName,這個(gè)方法和ClassLoader中的loaderClass方法的目的是一樣的,都是用來(lái)加載Class的,但是兩者在作用上卻有所區(qū)別:

loadClass加載實(shí)際上就是加載的時(shí)候并不對(duì)該類(lèi)進(jìn)行解釋?zhuān)虼瞬粫?huì)初始化該類(lèi)。而Class類(lèi)的forName方法則相反,使用forName加載的時(shí)候就會(huì)將Class進(jìn)行解釋和初始化

五,類(lèi)加載器的使用

使用URLClassLoader去加載類(lèi)

import java.lang.reflect.Constructor;import java.lang.reflect.Method;import java.net.URL;import java.net.URLClassLoader;
public class ClassLoaderTest {    public static void main(String args[]) {        try {                URL url = new URL("file:/C:/Users/spark/Desktop/logs-analyzer.jar");                URLClassLoader myClassLoader1 = new URLClassLoader(new URL[]{url}, Thread.currentThread().getContextClassLoader());                Class<?> clazz = myClassLoader1.loadClassQ("study.ClassLoaderTest.TestAction");                Method mainClass = clazz.getMethod("action");                Constructor<?> constructor = clazz.getConstructor();                Object obj = constructor.newInstance();                System.out.println(mainClass.invoke(obj));        } catch (Exception e) {            e.printStackTrace();        }    }}

首先定義好一個(gè)類(lèi),然后打包成jar

public class TestAction{    public String action()    {        return "this ActionTest class";    }}

六,Spark中的URLClassLoader簡(jiǎn)述

Spark使用內(nèi)部使用的最多的類(lèi)加載器就是URLClassloader。

private[spark] class MutableURLClassLoader(urls: Array[URL], parent: ClassLoader)  extends URLClassLoader(urls, parent) {
 override def addURL(url: URL): Unit = {    super.addURL(url)  }
 override def getURLs(): Array[URL] = {    super.getURLs()  }
}

這樣取決于Spark分布式計(jì)算的特性,后面源碼系列講述到運(yùn)行環(huán)境的時(shí)候會(huì)詳細(xì)說(shuō)道這個(gè)問(wèn)題。


感謝各位的閱讀!關(guān)于“java類(lèi)加載器的原理是什么”這篇文章就分享到這里了,希望以上內(nèi)容可以對(duì)大家有一定的幫助,讓大家可以學(xué)到更多知識(shí),如果覺(jué)得文章不錯(cuò),可以把它分享出去讓更多的人看到吧!

網(wǎng)站標(biāo)題:java類(lèi)加載器的原理是什么
標(biāo)題鏈接:http://www.chinadenli.net/article36/jsihsg.html

成都網(wǎng)站建設(shè)公司_創(chuàng)新互聯(lián),為您提供面包屑導(dǎo)航外貿(mào)網(wǎng)站建設(shè)商城網(wǎng)站微信小程序網(wǎng)站排名品牌網(wǎng)站設(shè)計(jì)

廣告

聲明:本網(wǎng)站發(fā)布的內(nèi)容(圖片、視頻和文字)以用戶投稿、用戶轉(zhuǎn)載內(nèi)容為主,如果涉及侵權(quán)請(qǐng)盡快告知,我們將會(huì)在第一時(shí)間刪除。文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如需處理請(qǐng)聯(lián)系客服。電話:028-86922220;郵箱:631063699@qq.com。內(nèi)容未經(jīng)允許不得轉(zhuǎn)載,或轉(zhuǎn)載時(shí)需注明來(lái)源: 創(chuàng)新互聯(lián)

外貿(mào)網(wǎng)站制作