Was ist ClassLoader? Unter allen Programmiersprachen ist Java einzigartig, da es auf der Java Virtual Machine läuft. Dies bedeutet, dass das kompilierte Programm auf dem Zielcomputer in einer eindeutigen, plattformunabhängigen Form ausgeführt wird, die nicht dem Format des Zielcomputers entspricht. Dieses Format unterscheidet sich in vielerlei Hinsicht stark von herkömmlichen ausführbaren Programmen.
Java ClassLoader ist eine wichtige, aber oft übersehene Komponente im Java-Betriebssystem. Es ist dafür verantwortlich, Klassendateien zur Laufzeit zu finden und zu laden. Durch das Erstellen eines benutzerdefinierten ClassLoaders kann die Art und Weise, wie Klassendateien in das System geladen werden, völlig neu definiert werden.
Dieses Tutorial bietet einen allgemeinen Überblick über den ClassLoader von Java und gibt ein Beispiel für einen benutzerdefinierten ClassLoader. Dieser ClassLoader wird automatisch kompiliert, bevor der Code geladen wird. Sie erfahren, was ein ClassLoader tut und wie Sie einen benutzerdefinierten ClassLoader erstellen.
Für dieses Tutorial müssen die Leser über grundlegende Kenntnisse der Java-Programmierung verfügen, einschließlich der Erstellung, Kompilierung und Ausführung einfacher Java-Befehlszeilenprogramme.
Nachdem Sie dieses Tutorial gelesen haben, wissen Sie, wie Sie:
die Funktionalität der JVM erweitern
einen Benutzerdefiniert erstellen ClassLoader
Erfahren Sie, wie Sie ClassLoader in Java-Anwendungen integrieren
Ändern Sie ClassLoader, um es an die Java2-Version anzupassen
Unter allen Programmiersprachen ist Java insofern einzigartig, als es auf der Java Virtual Machine läuft. Dies bedeutet, dass das kompilierte Programm auf dem Zielcomputer in einer eindeutigen, plattformunabhängigen Form ausgeführt wird, die nicht dem Format des Zielcomputers entspricht. Dieses Format unterscheidet sich in vielerlei Hinsicht stark von herkömmlichen ausführbaren Programmen.
Der größte Unterschied zwischen einem Java-Programm und einem C- oder C++-Programm besteht darin, dass es sich nicht um eine einzelne ausführbare Datei handelt, sondern aus vielen separaten Klassendateien besteht, wobei jede Klassendatei einer Java-Klasse entspricht.
Darüber hinaus werden diese Klassendateien nicht auf einmal, sondern bei Bedarf in den Speicher geladen. ClassLoader ist der Teil der JVM, der Klassen in den Speicher lädt.
Darüber hinaus ist Java ClassLoader in Java geschrieben. Dies bedeutet, dass Sie ganz einfach Ihren eigenen ClassLoader erstellen können, ohne weitere Details über die JVM zu kennen.
Wenn die JVM bereits einen ClassLoader hat, warum müssen Sie dann einen weiteren schreiben? Gute Frage, der Standard-ClassLoader weiß nur, wie man Klassendateien aus dem lokalen Dateisystem lädt. In allgemeinen Szenarien ist es völlig ausreichend, Code lokal zu schreiben und lokal zu kompilieren.
Eines der neuartigsten Merkmale der JAVA-Sprache ist jedoch, dass Kurse von der lokalen Festplatte oder außerhalb des Internets abgerufen werden können. Beispielsweise verwendet der Browser einen benutzerdefinierten ClassLoader, um ausführbare Inhalte von der Website abzurufen.
Es gibt viele andere Möglichkeiten, Klassendateien zu erhalten. Zusätzlich zum lokalen oder Online-Laden von Klassendateien können Sie Klassenlader auch verwenden, um:
Digitale Signaturen automatisch zu überprüfen, bevor nicht vertrauenswürdiger Code ausgeführt wird
Verwenden Sie den vom Benutzer bereitgestellten passworttransparenten Entschlüsselungscode.
Erstellen Sie benutzerdefinierte dynamische Klassen basierend auf den spezifischen Anforderungen des Benutzers.
Jede Generation. Der Inhalt von Java-Bytecode kann in Ihre Anwendung integriert werden.
Wenn Sie jemals ein Applet verwendet haben, müssen Sie einen benutzerdefinierten Klassenlader verwendet haben.
Als Sun die Java-Sprache veröffentlichte, war es eines der aufregendsten Dinge, zu beobachten, wie die Technologie das rechtzeitige Laden von Code von einem Remote-Webserver durchführte. Sie senden Bytecode über eine HTTP-Verbindung von einem Remote-Webserver und führen ihn lokal aus, was spannend ist.
Die Fähigkeit der Java-Sprache, benutzerdefinierte ClassLoader zu unterstützen, macht diese Idee möglich. Es gibt einen benutzerdefinierten ClassLoader im Applet. Er lädt die Klassendatei nicht aus dem lokalen Dateisystem, sondern ruft sie vom Remote-Webserver ab, lädt den ursprünglichen Bytecode über HTTP und konvertiert ihn dann in eine Klasse in der JVM.
Klassenlader in Browsern und Applets haben auch andere Funktionen: Sicherheitsverwaltung, verhindern, dass sich Applets auf verschiedenen Seiten gegenseitig beeinflussen usw.
Als nächstes erstellen wir einen benutzerdefinierten Klassenlader namens CompilingClassLoader(CCL)
, und CCL hilft uns beim Kompilieren von Java-Code. Es ist im Grunde so, als würde man ein einfaches Make-Programm direkt in das laufende System einbauen.
Der Hauptzweck von ClassLoader besteht darin, Dienste für Klassenanfragen bereitzustellen. Die JVM benötigt eine Klasse und fordert daher den ClassLoader auf, die Klasse anhand ihres Namens zu laden. ClassLoader versucht, ein Objekt zurückzugeben, das die Klasse darstellt.
Ein benutzerdefinierter ClassLoader kann durch Überschreiben der Methoden erstellt werden, die den verschiedenen Phasen dieses Prozesses entsprechen.
Im Rest dieses Artikels erfahren Sie mehr über einige wichtige Methoden in ClassLoader. Sie erfahren, was jede Methode bewirkt und wie sie beim Laden der Klasse aufgerufen wird. Außerdem erfahren Sie, was zu tun ist, wenn Sie einen ClassLoader anpassen. Die
loadClass
Methode ## und die ClassLoader.loadClass()
-Methode sind die Eingänge zum ClassLoader. Sein Methoden-Tag lautet wie folgt:
Class loadClass(String name, boolean resolve)
name
Der Parameter stellt den Namen der von der JVM benötigten Klasse dar, z. B. Foo
oder java.lang.Object
.
resolve
Der Parameter gibt an, ob die Klasse analysiert werden muss. Das Parsen von Klassen kann als vollständige Vorbereitung der Klasse auf die Ausführung verstanden werden. Ein Parsen ist nicht erforderlich. Wenn die JVM nur die Existenz der Klasse oder ihre übergeordnete Klasse ermitteln muss, ist keine Analyse erforderlich.
Vor Java1.1 musste der benutzerdefinierte ClassLoader nur die loadClass
-Methode überschreiben. Die Methode
defineClass
ist der Kern des gesamten ClassLoader. Diese Methode konvertiert das ursprüngliche Byte-Array in ein Class
-Objekt. Das Rohbyte-Array enthält lokal oder remote erhaltene Daten.
defineClass
ist für die Handhabung vieler komplexer, mysteriöser und umsetzungsabhängiger Teile der JVM verantwortlich. Es analysiert den Bytecode in Laufzeitdatenstrukturen, prüft deren Gültigkeit usw. Keine Sorge, Sie müssen dies nicht selbst umsetzen. Tatsächlich können Sie es überhaupt nicht überschreiben, da die Methode endgültig ist. Die Methode
findSystemClass方法
findSysetmClass
lädt Dateien aus dem lokalen Dateisystem. Es sucht nach einer Klassendatei im lokalen Dateisystem und konvertiert sie, falls vorhanden, mit defineClass
von Rohbytes in ein Klassenobjekt. Dies ist der Standardmechanismus der JVM zum Laden von Klassen beim Ausführen einer Java-Anwendung.
Für den benutzerdefinierten ClassLoader rufen wir die Methode findSystemClass
erst auf, nachdem wir andere Methoden ausprobiert haben, um den Klasseninhalt zu laden. Der Grund ist einfach: Ein benutzerdefinierter ClassLoader enthält einige Schritte zum Laden spezieller Klassen, aber nicht alle Klassen sind spezielle Klassen. Selbst wenn ClassLoader beispielsweise einige Klassen von der Remote-Website abrufen muss, müssen immer noch viele Klassen aus der lokalen Java-Bibliothek geladen werden. Diese Klassen stehen nicht in unserem Fokus, daher benötigen wir die JVM, um sie auf die Standardmethode abzurufen.
Der gesamte Prozess ist wie folgt:
Fordern Sie einen benutzerdefinierten ClassLoader an, um eine Klasse zu laden
Überprüfen Sie, ob der Remote-Server hat die Klasse
Wenn ja, holen und zurückgeben
Wenn nicht, gehen wir davon aus, dass die Klasse eine lokale Basisklasse ist und rufen <🎜 auf > aus dem Dateisystem geladen. findSystemClass
verwenden, um den Zugriff auf Remote-Websites zu reduzieren, da sich die meisten Java-Klassen in lokalen Klassenbibliotheken befinden. Wie Sie im nächsten Abschnitt sehen werden, möchten wir jedoch nicht, dass die JVM Klassen aus dem lokalen Dateisystem lädt, bevor sie den Anwendungscode automatisch kompiliert. findSystemClass
resolveClass
-Methode implementieren, müssen wir möglicherweise die loadClass
-Methode aufrufen, abhängig vom Wert des resolveClass
-Parameters in loadClass
. Die resolve
findLoadedClass
-Methode fungiert als Caching-Aufrufmechanismus: Wenn die findLoadedClass
-Methode aufgerufen wird, ruft sie diese Methode auf, um zu überprüfen, ob die Klasse geladen wurde . Eliminiert wiederholtes Laden. Diese Methode sollte zuerst aufgerufen werden. loadClass
führen Sie die folgenden Schritte aus (hier werden wir nicht besonders darauf achten, welche magische Methode verwendet wird, um die Klassendatei zu erhalten. Sie kann von lokal stammen, Aus dem Netzwerk oder aus komprimierten Dateien erhalten, kurz gesagt, wir haben den Bytecode der ursprünglichen Klassendatei erhalten): loadClass
auf, um zu überprüfen, ob die Klasse geladen wurde findLoadedClass
auf, um ihn in ein defineClass
Objekt Class
umzuwandeln
auf, um zu sehen, ob die Klasse findSystemClass
>Wenn Der Wert ist wahr. Rufen Sie resolve
auf, um das resolveClass
-Objekt zu analysieren. Class
ClassNotFoundException
CompilingClassLoader
wird verwendet, um sicherzustellen, dass der Code kompiliert wurde und die neueste Version ist. CCL
Das Folgende ist eine Beschreibung der Klasse:
ClassNotFoundException
auf, um zu sehen, ob sie nützlich ist findSystemClass
ClassNotFoundException
zurück
在深入研究之前,我们应该回过头来看一下Java的编译机制。总的来说,当你请求一个类的时候,Java不只是编译各种类信息,它还编译了别的相关联的类。
CCL会按需一个接一个的编译相关的类。但是,当CCL编译完一个类之后试着去编译其它相关类的时候会发现,其它的类已经编译完成了。为什么呢?Java编译器遵循一个规则:如果一个类不存在,或者它相对于源码已经过时了,就需要编译它。从本质上讲,Java编译器先CCL一步完成了大部分的工作。
CCL在编译类的时候会打印其编译的应用程序。在大多数场景里面,你会看到它在程序的主类上调用编译器。
但是,有一种情况是不会在第一次调用时编译所有类的的。如果你通过类名Class.forNasme
加载一个类,Java编译器不知道该类需要哪些信息。在这种场景下,你会看到CCL会再次运行Java编译器。
CompilingClassLoader
为了使用CCL,我们需要用一种独特的方式启动程序。正常的启动程序如下:
% java Foo arg1 arg2
而我们启动方式如下:
% java CCLRun Foo arg1 arg2
CCLRun是一个特殊的桩程序,它会创建一个CompilingClassLoader并使用它来加载程序的main方法,确保整个程序的类会通过CompilingClassLoader加载。CCLRun使用Java反射API来调用main方法并传参
Java1.2以后ClassLoader有一些变动。原有版本的ClassLoader还是兼容的,而且在新版本下开发ClassLoader更容易了
新的版本下采用了delegate模型。ClassLoader可以将类的请求委托给父类。默认的实现会先调用父类的实现,在自己加载。但是这种模式是可以改变的。所有的ClassLoader的根节点是系统ClassLoader。它默认会从文件系统中加载类。
loadClass
默认实现一个自定义的loadClass
方法通常会尝试用各种方法来获得一个类的信息。如果你写了大量的ClassLoader,你会发现基本上是在重复写复杂而变化不大的代码。
java1.2的loadClass
的默认实现中允许你直接重写findClass
方法,loadClass
将会在合适的时候调用该方法。
这种方式的好处在于你无须重写loadClass
方法。
findClass
该方法会被loadClass
的默认实现调用。findClass
是为了包含ClassLoader所有特定的代码,而无需写大量重负的其他代码
getSystenClassLoader
无论你是否重写了findClass
或是loadClass
方法,getSystemClassLoader
允许你直接获得系统的ClassLoader(而不是隐式的用findSystemClass
获得)
getParent
该方法允许类加载器获取其父类加载器,从而将请求委托给它。当你自定义的加载器无法找到类时,可以使用该方法。父类加载器是指包含创建该类加载代码的加载器。
// $Id$ import java.io.*; /* A CompilingClassLoader compiles your Java source on-the-fly. It checks for nonexistent .class files, or .class files that are older than their corresponding source code. */ public class CompilingClassLoader extends ClassLoader { // Given a filename, read the entirety of that file from disk // and return it as a byte array. private byte[] getBytes( String filename ) throws IOException { // Find out the length of the file File file = new File( filename ); long len = file.length(); // Create an array that's just the right size for the file's // contents byte raw[] = new byte[(int)len]; // Open the file FileInputStream fin = new FileInputStream( file ); // Read all of it into the array; if we don't get all, // then it's an error. int r = fin.read( raw ); if (r != len) throw new IOException( "Can't read all, "+r+" != "+len ); // Don't forget to close the file! fin.close(); // And finally return the file contents as an array return raw; } // Spawn a process to compile the java source code file // specified in the 'javaFile' parameter. Return a true if // the compilation worked, false otherwise. private boolean compile( String javaFile ) throws IOException { // Let the user know what's going on System.out.println( "CCL: Compiling "+javaFile+"..." ); // Start up the compiler Process p = Runtime.getRuntime().exec( "javac "+javaFile ); // Wait for it to finish running try { p.waitFor(); } catch( InterruptedException ie ) { System.out.println( ie ); } // Check the return code, in case of a compilation error int ret = p.exitValue(); // Tell whether the compilation worked return ret==0; } // The heart of the ClassLoader -- automatically compile // source as necessary when looking for class files public Class loadClass( String name, boolean resolve ) throws ClassNotFoundException { // Our goal is to get a Class object Class clas = null; // First, see if we've already dealt with this one clas = findLoadedClass( name ); //System.out.println( "findLoadedClass: "+clas ); // Create a pathname from the class name // E.g. java.lang.Object => java/lang/Object String fileStub = name.replace( '.', '/' ); // Build objects pointing to the source code (.java) and object // code (.class) String javaFilename = fileStub+".java"; String classFilename = fileStub+".class"; File javaFile = new File( javaFilename ); File classFile = new File( classFilename ); //System.out.println( "j "+javaFile.lastModified()+" c "+ // classFile.lastModified() ); // First, see if we want to try compiling. We do if (a) there // is source code, and either (b0) there is no object code, // or (b1) there is object code, but it's older than the source if (javaFile.exists() && (!classFile.exists() || javaFile.lastModified() > classFile.lastModified())) { try { // Try to compile it. If this doesn't work, then // we must declare failure. (It's not good enough to use // and already-existing, but out-of-date, classfile) if (!compile( javaFilename ) || !classFile.exists()) { throw new ClassNotFoundException( "Compile failed: "+javaFilename ); } } catch( IOException ie ) { // Another place where we might come to if we fail // to compile throw new ClassNotFoundException( ie.toString() ); } } // Let's try to load up the raw bytes, assuming they were // properly compiled, or didn't need to be compiled try { // read the bytes byte raw[] = getBytes( classFilename ); // try to turn them into a class clas = defineClass( name, raw, 0, raw.length ); } catch( IOException ie ) { // This is not a failure! If we reach here, it might // mean that we are dealing with a class in a library, // such as java.lang.Object } //System.out.println( "defineClass: "+clas ); // Maybe the class is in a library -- try loading // the normal way if (clas==null) { clas = findSystemClass( name ); } //System.out.println( "findSystemClass: "+clas ); // Resolve the class, if any, but only if the "resolve" // flag is set to true if (resolve && clas != null) resolveClass( clas ); // If we still don't have a class, it's an error if (clas == null) throw new ClassNotFoundException( name ); // Otherwise, return the class return clas; } }
import java.lang.reflect.*; /* CCLRun executes a Java program by loading it through a CompilingClassLoader. */ public class CCLRun { static public void main( String args[] ) throws Exception { // The first argument is the Java program (class) the user // wants to run String progClass = args[0]; // And the arguments to that program are just // arguments 1..n, so separate those out into // their own array String progArgs[] = new String[args.length-1]; System.arraycopy( args, 1, progArgs, 0, progArgs.length ); // Create a CompilingClassLoader CompilingClassLoader ccl = new CompilingClassLoader(); // Load the main class through our CCL Class clas = ccl.loadClass( progClass ); // Use reflection to call its main() method, and to // pass the arguments in. // Get a class representing the type of the main method's argument Class mainArgType[] = { (new String[0]).getClass() }; // Find the standard main method in the class Method main = clas.getMethod( "main", mainArgType ); // Create a list containing the arguments -- in this case, // an array of strings Object argsArray[] = { progArgs }; // Call the method main.invoke( null, argsArray ); } }
public class Foo { static public void main( String args[] ) throws Exception { System.out.println( "foo! "+args[0]+" "+args[1] ); new Bar( args[0], args[1] ); } }
import baz.*; public class Bar { public Bar( String a, String b ) { System.out.println( "bar! "+a+" "+b ); new Baz( a, b ); try { Class booClass = Class.forName( "Boo" ); Object boo = booClass.newInstance(); } catch( Exception e ) { e.printStackTrace(); } } }
package baz; public class Baz { public Baz( String a, String b ) { System.out.println( "baz! "+a+" "+b ); } }
public class Boo { public Boo() { System.out.println( "Boo!" ); } }
相关文章:
基于Java类的加载方式之classloader类加载器详解
相关视频:
Das obige ist der detaillierte Inhalt vonWarum ClassLoader schreiben? Tiefes Verständnis des Klassenladers in Java. Für weitere Informationen folgen Sie bitte anderen verwandten Artikeln auf der PHP chinesischen Website!