电脑知识|欧美黑人一区二区三区|软件|欧美黑人一级爽快片淫片高清|系统|欧美黑人狂野猛交老妇|数据库|服务器|编程开发|网络运营|知识问答|技术教程文章 - 好吧啦网

您的位置:首頁技術文章
文章詳情頁

淺談Java自定義類加載器及JVM自帶的類加載器之間的交互關系

瀏覽:22日期:2022-08-16 13:54:20
JVM自帶的類加載器:

淺談Java自定義類加載器及JVM自帶的類加載器之間的交互關系

其關系如下:

淺談Java自定義類加載器及JVM自帶的類加載器之間的交互關系

其中,類加載器在加載類的時候是使用了所謂的“父委托”機制。其中,除了根類加載器以外,其他的類加載器都有且只有一個父類加載器。

關于父委托機制的說明:

淺談Java自定義類加載器及JVM自帶的類加載器之間的交互關系

當生成 一個自定義的類加載器實例時,如果沒有指定它的父加載器,那么系統類加載器將成為該類加載器的父類加載器

下面,自定義類加載器。自定義的類加載器必須繼承java.lang.ClassLoader類

import java.io.*;public class MyClassLoader extends ClassLoader { private String name; //類加載器的名字 private String path; //加載類的路徑 private final String fileType = '.class'; //class文件的擴展名 public MyClassLoader(String name){ super(); //讓系統類加載器成為該類加載器的父 類加載器,該句可省略不寫 this.name = name; } public MyClassLoader(ClassLoader parent,String name){ super(parent); //顯示指定該類加載器的父 類加載器 this.name = name; } @Override public String toString() { return this.name; } public String getPath() { return path; } public void setPath(String path) { this.path = path; } //實現自定義的類加載器必須重寫findClass方法,否則ClassLoader類中的findClass()方法是拋出了異常 @Override public Class findClass(String name)throws ClassNotFoundException{ byte[] data = this.loadClassData(name); return this.defineClass(name,data,0,data.length); } private byte[] loadClassData(String name){ InputStream is = null; byte[] data = null; ByteArrayOutputStream baos = null; try { this.name = this.name.replace('.',''); //com.dream.it---->comdreamit is = new FileInputStream(new File(path + name + fileType)); int ch; while(-1 != (ch = is.read())){baos.write(ch); //將數據寫入到字節數組輸出流對象中去 } data = baos.toByteArray(); } catch (Exception e) { e.printStackTrace(); }finally { try {is.close();baos.close(); } catch (IOException e) {e.printStackTrace(); } } return data; } public static void main(String[] args) throws Exception { MyClassLoader loader1 = new MyClassLoader('loader1'); loader1.setPath('d:/myapp/serverlib/'); MyClassLoader loader2 = new MyClassLoader(loader1,'loader2'); //loader1作為loader2的父 類加載器 loader2.setPath('d:/myapp/clientlib'); MyClassLoader loader3 = new MyClassLoader(null,'loader3');//父類加載器為null,表明其父類加載器為根類加載器 loader3.setPath('d:/myapp/otherlib'); test(loader2); test(loader3); } public static void test(ClassLoader cl) throws Exception { Class clazz = cl.loadClass('Sample'); Object object = clazz.newInstance(); }}附上findClass()方法的JDK說明

protected Class<?> findClass(String name) throws ClassNotFoundExceptionFinds the class with the specified binary name. This method should be overridden by class loaderimplementations that follow the delegation model for loading classes, and will be invoked by the loadClass method after checking the parent classloader for the requested class. The default implementation throws a ClassNotFoundException.

大致說明一下意思:通過指定的name來查找類。該方法應該被類加載器的實現類重寫,從而能夠保證在加載類的時候可以遵循委托機制模型。在loadClass()方法(該方法是由JVM調用的)中,檢查其父類加載器之后,該方法再被調用去加載請求的類。默認該方法的實現是拋出了一個ClassNotFoundException異常。

其實,所謂的加載類,無非就是讀取.class文件到內存中,所以在findClass()方法中,loadClassData()方法用于讀取.class文件的數據,并返回一個字節數組。然后利用ClassLoader類的defineClass()方法將字節數組轉換為Class對象。

上述自定義的類加載器loader1,loader2,loader3及JVM自帶的類加載器之間的關系如下:

淺談Java自定義類加載器及JVM自帶的類加載器之間的交互關系

對于各個類加載器,系統的類加載器是從環境變量classpath中讀取.class文件實現類的加載;loader1是從目錄d:/myapp/serverlib/下讀取.class文件;loader2是從目錄d:/myapp/clientlib/下讀取.class文件,loader3是從目錄d:/myapp/otherlib/下讀取.class文件

執行結果:

淺談Java自定義類加載器及JVM自帶的類加載器之間的交互關系

此處我們分析一下出現這種執行結果的原因:

當執行loader2.loadClass(“Sample”)時先由它上層的所有父類加載器嘗試加載Sample類。

loader1從D:myappserverliv目錄下成功加載了Sample類,所以loader1是Sample類的定義類加載器,loader1和loader2是Sample類的初始類加載器。

當執行loader3.loadClass(“Sample”)時,先由它上層的所有父類加載器嘗試加載Sample類。

loader3的父加載器為根類加載器,它無法加載Sample類,接著loader3從D:myappotherlib目錄下成功加載Sample類,所以loader3是Sample類的定義類加載器及初始類加載器。

在Sample類中主動使用了Dog類(new Dog()),當執行Sample類的構造方法中的new Dog()語句時,JVM需要先加載Dog類,到底用哪個類加載器家在呢?

從上述的打印結果中可以看出,加載Sample類的loader1還加載了Dog類,JVM會用Sample類的定義類加載器去加載Dog類,加載過程中也同樣采用了父親委托機制。

為了驗證這一點,可以吧D:myappserverlib目錄下Dog.class文件刪除,然后在D:myappsyslib目錄下存放一個Dog.class文件,此時打印結果如下:

Sample:loader1Dog:sun.misc.Launcher$AppClassLoader@1b84c92Sample:loader3Dog:loader3

由此可見,當由loader1加載的Sample類首次主動使用Dog類時,Dog類由系統類加載器加載,如果把D:myappserverlib和D:myappsyslib目錄下的Dog.class文件都刪除,然后在D:myappclient目錄下存放一個Dog.class文件。

此時文件結構如下圖所示:

淺談Java自定義類加載器及JVM自帶的類加載器之間的交互關系

當Loader1加載Sample類首次主動使用Dog類時,由于loader1及其父類加載器都無法加載Dog類,因此test(loader2)會拋出ClassNotFoundExcption.

這又是因為什么原因呢?

這又牽扯到命名空間的問題。

同一個命名空間內的類時相互可見的。

子加載器的命名空間包含所有父類加載器的命名空間,因此由子加載器加載的類能看見父類加載器加載的類。例如系統類加載器加載的類能看見根類加載器加載的類。由父加載器加載的類不能看見子加載器加載的類。

如果兩個加載器之間沒有直接或間接的父子關系,那么它們各自加載的類相互不可見。

對于上述問題,loader1可以加載Sample類,而Dog類只能由loader2加載Dog類,loader1是Loader2的父類加載器,父加載器loader1加載的類Sample不能看見子加載器loader2加載的類Dog,所以會拋出異常。

對于上述實例中的main方法,我們不調用test方法,換成如下代碼

Class clazz = loader1.loadClass('Sample');Object obj = clazz.newInstance();Sample sample = (Sample)obj;System.out.println(sample.v1);

MyClassLoader類由系統類加載器加載,而Sample類由loader1類加載器加載,所以MyClassLoader類看不見Sample類。在MyClassLoader類的main方法中使用Sample類,會導致NoClassFoundError錯誤。

當兩個不同命名空間內的類相互不可見時,可采用Java反射機制來訪問對象實例的屬性和方法。

將上述代碼修改:

Class clazz = loader1.loadClass('Sample');Object obj = clazz.newInstance();Field field = clazz.getField('v1');int v1 = field.getInt(obj);System.out.println(v1);

此時,可以獲取到對象中的v1屬性值。利用反射機制,我們可以跨越這種命名空間的限制。

補充:

命名空間:

淺談Java自定義類加載器及JVM自帶的類加載器之間的交互關系

運行時包:

淺談Java自定義類加載器及JVM自帶的類加載器之間的交互關系

以上為個人經驗,希望能給大家一個參考,也希望大家多多支持好吧啦網。如有錯誤或未考慮完全的地方,望不吝賜教。

標簽: Java
相關文章:
主站蜘蛛池模板: 高压分散机(高压细胞破碎仪)百科-北京天恩瀚拓| 乳化沥青设备_改性沥青设备_沥青加温罐_德州市昊通路桥工程有限公司 | ISO9001认证咨询_iso9001企业认证代理机构_14001|18001|16949|50430认证-艾世欧认证网 | 杭州货架订做_组合货架公司_货位式货架_贯通式_重型仓储_工厂货架_货架销售厂家_杭州永诚货架有限公司 | 河北中仪伟创试验仪器有限公司是专业生产沥青,土工,水泥,混凝土等试验仪器的厂家,咨询电话:13373070969 | BAUER减速机|ROSSI-MERSEN熔断器-APTECH调压阀-上海爱泽工业设备有限公司 | 伊卡洛斯软装首页-电动窗帘,别墅窗帘,定制窗帘,江浙沪1000+别墅窗帘案例 | 杭州高温泵_热水泵_高温油泵|昆山奥兰克泵业制造有限公司 | 肉嫩度仪-凝胶测试仪-国产质构仪-气味分析仪-上海保圣实业发展有限公司|总部 | 闭端端子|弹簧螺式接线头|防水接线头|插线式接线头|端子台|电源线扣+护线套|印刷电路板型端子台|金笔电子代理商-上海拓胜电气有限公司 | 风信子发稿-专注为企业提供全球新闻稿发布服务 | 碎石机设备-欧版反击破-欧版颚式破碎机(站)厂家_山东奥凯诺机械 高低温试验箱-模拟高低温试验箱订制-北京普桑达仪器科技有限公司【官网】 | 涡街流量计_LUGB智能管道式高温防爆蒸汽温压补偿计量表-江苏凯铭仪表有限公司 | 上海冠顶工业设备有限公司-隧道炉,烘箱,UV固化机,涂装设备,高温炉,工业机器人生产厂家 | 企典软件一站式企业管理平台,可私有、本地化部署!在线CRM客户关系管理系统|移动办公OA管理系统|HR人事管理系统|人力 | 水质监测站_水质在线分析仪_水质自动监测系统_多参数水质在线监测仪_水质传感器-山东万象环境科技有限公司 | 陕西安闸机-伸缩门-车牌识别-广告道闸——捷申达门业科技 | 武汉宣传片制作-视频拍摄-企业宣传片公司-武汉红年影视 | 手术室净化厂家_成都实验室装修公司_无尘车间施工单位_洁净室工程建设团队-四川华锐16年行业经验 | 回收二手冲床_金丰旧冲床回收_协易冲床回收 - 大鑫机械设备 | 济南玻璃安装_济南玻璃门_济南感应门_济南玻璃隔断_济南玻璃门维修_济南镜片安装_济南肯德基门_济南高隔间-济南凯轩鹏宇玻璃有限公司 | SOUNDWELL 编码器|电位器|旋转编码器|可调电位器|编码开关厂家-广东升威电子制品有限公司 | 多物理场仿真软件_电磁仿真软件_EDA多物理场仿真软件 - 裕兴木兰 | led太阳能路灯厂家价格_风光互补庭院灯_农村市政工程路灯-中山华可路灯品牌 | 辊道窑炉,辊道窑炉厂家-山东艾希尔| 河南包装袋厂家_河南真空袋批发价格_河南服装袋定制-恒源达包装制品 | 成都珞石机械 - 模温机、油温机、油加热器生产厂家 | 江西自考网 | 翅片管散热器价格_钢制暖气片报价_钢制板式散热器厂家「河北冀春暖气片有限公司」 | 设计圈 - 让设计更有价值!| 气力输送_输送机械_自动化配料系统_负压吸送_制造主力军江苏高达智能装备有限公司! | 制样机-密封锤式破碎机-粉碎机-智能马弗炉-南昌科鑫制样 | YJLV22铝芯铠装电缆-MYPTJ矿用高压橡套电缆-天津市电缆总厂 | 博莱特空压机|博莱特-阿特拉斯独资空压机品牌核心代理商 | 学习安徽网| 水冷散热器_水冷电子散热器_大功率散热器_水冷板散热器厂家-河源市恒光辉散热器有限公司 | 信阳市建筑勘察设计研究院有限公司| 幂简集成 - 品种超全的API接口平台, 一站搜索、试用、集成国内外API接口 | 震动筛选机|震动分筛机|筛粉机|振筛机|振荡筛-振动筛分设备专业生产厂家高服机械 | 密集架-密集柜厂家-智能档案密集架-自动选层柜订做-河北风顺金属制品有限公司 | 北京银联移动POS机办理_收银POS机_智能pos机_刷卡机_收银系统_个人POS机-谷骐科技【官网】 |