類載入器

来源:http://www.cnblogs.com/zr520/archive/2016/10/29/6010533.html
-Advertisement-
Play Games

在我心中類載入一直是很迷人的,主要的原因是覺得為何能夠這個過程是如何運作的。最近在總結自己以前學過的東西,所以也就總結一下類載入器這方面的東西。 對於為何將載入類這樣一個那麼重要的事放在JVM外部在實現,我覺的最主要的原因是為了提供更好的擴展性吧,讓程式員自己去決定如何載入一個類。根據書上的說法是這 ...


在我心中類載入一直是很迷人的,主要的原因是覺得為何能夠這個過程是如何運作的。最近在總結自己以前學過的東西,所以也就總結一下類載入器這方面的東西。

對於為何將載入類這樣一個那麼重要的事放在JVM外部在實現,我覺的最主要的原因是為了提供更好的擴展性吧,讓程式員自己去決定如何載入一個類。根據書上的說法是這個類載入器最初是為了滿足JAVA Applet為設計的,但是奈何我是newbie,沒有經歷過那個時期,也只能一年矇蔽的看著老鳥們的講解了。但是如今類載入的主要方向在類層次劃分,OSGi,熱部署,代碼加密等領域有很好的發揮。

1.類與類載入器

 雖然在載入器只用於載入類得動作,但是起到的作用卻並非如此,因為每個類載入器都有自己的類名稱空間,也就是說,兩個類只有來自同一個類載入器才有意義,不然的話,即使是同一個位元組碼來的,也是不會相同的,即使是用 instanceof關鍵字來進行判斷。

package com.hotusm.classloader;

import java.io.InputStream;

import org.junit.Test;
/**
 * 
 * @author Hotusm  <br/>
 * @date 2016年10月29日   <br/>
 * @description
 */
public class ClassLoaderTest {
    
    @Test
    public void test() throws ClassNotFoundException{
        
        PathClassLoader classLoader=new PathClassLoader("D:/jeesite/ActOfJava/bin");
        Class<?> loadClass = classLoader.findClass("com.hotusm.classloader.ClassLoaderTest");
        System.out.println(loadClass);
    }
    
    @Test
    public void testClassPath(){
        
        ClassLoader classLoader=new ClassLoader() {
            
            @Override
            public Class<?> loadClass(String name) throws ClassNotFoundException {
                    if(!name.startsWith("com.hotusm"))
                        return super.loadClass(name);
                    
                    
                    String fileName=name.substring(name.lastIndexOf(".")+1)+".class";
                    try(InputStream is=getClass().getResourceAsStream(fileName)) {
                        
                        if(is==null){
                            super.loadClass(name);
                        }
                        byte[] b=new byte[is.available()];
                        
                        is.read(b);
                        
                        return defineClass(name, b, 0, b.length);
                    } catch (Exception e) {
                        e.printStackTrace();
                    }
                    return null;
            }
        };
        
        try {
            Object obj = classLoader.loadClass("com.hotusm.classloader.ClassLoaderTest").newInstance();
            //Method method = obj.getClass().getMethod("testMethod");
            //method.invoke(obj);
            System.out.println(obj instanceof com.hotusm.classloader.ClassLoaderTest);
        } catch (Exception e) {
            e.printStackTrace();
        }
        
    }
    
    public void testMethod(){
        System.out.println("method");
    }
}

 上面的輸出就是false!!(註釋的那些也就是一個代碼加密的思路,將位元組碼文件進行加密,然後自定義類載入器,實現自己的解密,將位元組碼進行解密,最後反射調用)

2.雙親委派模型:

  雙親委派講的是將職責一直向上委派,知道最頂層。如果最頂層不能解析,那麼再往下委派,直到解析成功,

3.類載入器種類:

引導類載入器(bootstrap class loader):

它用來載入 Java 的核心庫(jre/lib/rt.jar),是用原生C++代碼來實現的,並不繼承自java.lang.ClassLoader。

載入擴展類和應用程式類載入器,並指定他們的父類載入器,在java中獲取不到。 

擴展類載入器(extensions class loader):

它用來載入 Java 的擴展庫(jre/ext/*.jar)。Java 虛擬機的實現會提供一個擴展庫目錄。該類載入器在此目錄裡面查找並載入 Java 類。 

系統類載入器(system class loader):

它根據 Java 應用的類路徑(CLASSPATH)來載入 Java 類。一般來說,Java 應用的類都是由它來完成載入的。可以通過 ClassLoader.getSystemClassLoader()來獲取它。

自定義類載入器(custom class loader):

除了系統提供的類載入器以外,開發人員可以通過繼承 java.lang.ClassLoader類的方式實現自己的類載入器,以滿足一些特殊的需求。

下麵是自己定義的實現的類載入器:

package com.hotusm.classloader;

import java.io.ByteArrayOutputStream;
import java.io.File;
import java.io.FileInputStream;
import java.io.InputStream;
import java.util.Map;
import java.util.concurrent.ConcurrentHashMap;
/**
 * 
 * @author Hotusm  <br/>
 * @date 2016年10月28日   <br/>
 * @description 類載入器
 * http://www.cnblogs.com/sunniest/p/4574080.html
 * 
 * JVM中類載入器的樹狀層次結構
 *         引導類載入器(bootstrap class loader)
 *         擴展類載入器(extensions class loader)
 *         系統類載入器(system class loader)
 *         自定義類載入器(custom class loader)
 * 
 */
public class PathClassLoader extends ClassLoader{
    
    private String classPath;
    private String packageNames="com.hotusm";
    
    private Map<String,Class<?>> cache=new ConcurrentHashMap<>();
    
    public PathClassLoader(String classPath){
        this.classPath=classPath;
    }

    @Override
    protected Class<?> findClass(String name) throws ClassNotFoundException {
        
        if(name.startsWith(packageNames)){
            return cacheClass(name);
        }else{
            return super.findClass(name);
        }
        
    }
    
    private byte[] getData(String className){
        String path=classPath+File.separatorChar+className.replace('.', File.separatorChar)
                +".class";
        try(InputStream is=new FileInputStream(path)) {
            
            ByteArrayOutputStream stream=new ByteArrayOutputStream();
            byte[] buffer=new byte[2014];
            int num=0;
            while((num=is.read(buffer))!=-1){
                stream.write(buffer,0,num);
            }
            return stream.toByteArray();
        } catch (Exception e) {
            e.printStackTrace();
        }
        return null;
    }
    
    private Class<?> cacheClass(final String name) throws ClassNotFoundException{
        Class<?> clazz = cache.get(name);
            synchronized (name) {
                if(clazz==null){
                    byte[] classData = getData(name);
                    if(classData==null){
                        throw new ClassNotFoundException();
                    }else{
                        clazz = defineClass(name, classData, 0, classData.length);
                        cache.put(name, clazz);
                    }
                }
            }
        return clazz;
    }
}

 


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 文檔目錄 本節內容: 簡介 關於 IFeatureValueStore 功能類型 Boolean 功能 Value 功能 定義功能 基本功能屬性 其它功能屬性 功能層次 檢查功能 使用RequiresFeature特性 RequiresFeature特性註意事項 使用 IFeatureChecker ...
  • UEditor For ASP.Net Core Use Qiniu 此項目為UEditor提供文件管理; 後端服務使用 ASP.Net Core; 使用七牛提供的雲存儲; 項目地址 http://git.oschina.net/joesjiang/BaiduEditorUseQiniu.NetCo ...
  • 列印一個介面、方法的運行時間在程式中是很容易遇到的一件事情;現在,我就分享一個我在工作中使用的臨時列印日誌的方法和結合 Stopwatch 列印測量某個時間間隔的運行時間的方法。 Stopwatch 實例可以很好的測量一個時間間隔的運行時間;以下例子是比較常用到的: 引用命名空間: using Sy ...
  • using System; using System.Collections.Generic; using System.Linq; using System.Text; //本人英語不太好,嘿嘿。。 //其實介面多繼承和單繼承的形式是一樣的。 //博客里介面單繼承,單繼承會了,看這個就很輕鬆了 n ...
  • 上一章筆者講到關於Asp.NET的知識點。瞭解Asp.NET基本的知識點之後,我們在來學習關於C#的MVC框架就簡單多了。顯然本章就是來介紹一下關於Asp.NET MVC。對於MVC的思想筆者相信大家已經很清楚了。筆者在這裡就不多說了。我有個同學做Asp.NET開發的。他跟我講過一句話——我都快忘了 ...
  • 只要根據錯誤提示,找到相應文件夾下的\obj\Debug文件夾下的 相應名字.csproj.FileListAbsolute.txt, 打開並刪除含有'<<<<<<< .mine'或類似的欄位,重新編譯即可。 項目的.csproj 下相關'<<<<<<< .mine'或類似的欄位也需要刪除! ...
  • 在項目中遇到列印日誌或者處理異常這再常見不過了,通常service層處理日誌每個方法都寫一遍logger.info()太麻煩了。而aop恰好可以解決這個問題,寫一個aop的類就能處理這個問題了,spring更是擁有強大的aop功能。那就看看springaop到底多麼簡便吧! ...
  • 測試環境:php5.3.29 sort_flags 可以用以下值改變排序的行為: 排序類型標記: SORT_REGULAR - 正常比較單元(不改變類型) SORT_NUMERIC - 單元被作為數字來比較 SORT_STRING - 單元被作為字元串來比較 SORT_LOCALE_STRING - ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...