|
|||||||
|
|
|
![]() |
|
|
Strumenti |
|
|
#1 |
|
Senior Member
Iscritto dal: Nov 2004
Messaggi: 691
|
[JAVA] Problema con custom classloader
Buongiorno a tutti. Sto cercando di scrivere un classloader custom (estende URLClassLoader) per caricare/ricaricare a runtime certe classi.
Ho provato in vari modi ma ho una serie di problemi. Ad esempio, non riesco a "collegare" correttamente il custom classloader alla gerarchia di classloader. In pratica, riesco a fargli caricare un jar a runtime, ma quando carico una classe specifica dentro quel jar mi va in ClassNotFoundException su classi che avrebbe dovuto trovare nei classloader padri (ad es org.apache.log4j)... Qualcuno ha già dovuto sbattere la testa sui custom classloader e mi sa dare una mano? |
|
|
|
|
|
#2 |
|
Senior Member
Iscritto dal: Oct 2007
Città: Padova
Messaggi: 4131
|
Non posso darti consigli circostanziati, perchè fin'ora non mi sono mai cimentato nello studio del meccanismo di class loading e nell'implementazione di un custom ClasslLoader, però prova a vedere se questa lettura aiuta:
http://books.google.com/books?id=EhX...loader&f=false
__________________
As long as you are basically literate in programming, you should be able to express any logical relationship you understand. If you don’t understand a logical relationship, you can use the attempt to program it as a means to learn about it. (Chris Crawford) |
|
|
|
|
|
#3 | |
|
Senior Member
Iscritto dal: Nov 2004
Messaggi: 691
|
Quote:
|
|
|
|
|
|
|
#4 |
|
Junior Member
Iscritto dal: Jul 2007
Città: Benevento
Messaggi: 18
|
Se estendi URLClassLoader basta passare il classloader padre al costruttore
__________________
Pocket-MAL - MyAnimeList on iPhone - euks.wordpress.com PC1 - Thermaltake V4 | Corsair TX 650M | ASRock Z77 Extreme 6 | 3770K | 8GB G-Skill Sniper F3 PC2 - Cooler Master Centurion 534 | Enermax Liberty 500W | Gigabyte P35-DS4 | Intel Core2Duo E6600@3600 | 4GB Corsair Xms2 | Nvidia GeForce 8600GT | Seagate Barracuda 7200.10 320GB | LG H62N |
|
|
|
|
|
#5 |
|
Senior Member
Iscritto dal: Nov 2004
Messaggi: 691
|
Vero, ma ho comunque un problema, ossia il fatto che se una classe usata nella classe che carico dinamicamente è già stata caricata ottengo un classcastexception perchè la stessa classe caricata da due class loader diversi è diversa...
|
|
|
|
|
|
#6 |
|
Junior Member
Iscritto dal: Jul 2007
Città: Benevento
Messaggi: 18
|
Strano. Mi è capitato più volte di fare in questo modo e non ho avuto problemi.
Potresti postare il codice?
__________________
Pocket-MAL - MyAnimeList on iPhone - euks.wordpress.com PC1 - Thermaltake V4 | Corsair TX 650M | ASRock Z77 Extreme 6 | 3770K | 8GB G-Skill Sniper F3 PC2 - Cooler Master Centurion 534 | Enermax Liberty 500W | Gigabyte P35-DS4 | Intel Core2Duo E6600@3600 | 4GB Corsair Xms2 | Nvidia GeForce 8600GT | Seagate Barracuda 7200.10 320GB | LG H62N |
|
|
|
|
|
#7 | |
|
Senior Member
Iscritto dal: Oct 2007
Città: Padova
Messaggi: 4131
|
Quote:
Ed è giusto che sia così per via di implicazioni inerenti la sicurezza, quando si caricano classi remote.
__________________
As long as you are basically literate in programming, you should be able to express any logical relationship you understand. If you don’t understand a logical relationship, you can use the attempt to program it as a means to learn about it. (Chris Crawford) |
|
|
|
|
|
|
#8 | |
|
Senior Member
Iscritto dal: Nov 2004
Messaggi: 691
|
La main class
Codice:
public static void main(String[] args) throws ClassNotFoundException, IllegalAccessException, InstantiationException {
try {
ClassLoader parentClassLoader = MyClassLoader.class.getClassLoader();
MyClassLoader classLoader = new MyClassLoader(parentClassLoader);
Class myObjectClass = classLoader.loadClass("monitor.resource.Resource");
Resource object1 = (Resource) myObjectClass.getConstructor(String.class,String.class,String.class,boolean.class,String.class).newInstance("resourceFile","patternFile","resourceName",true,"parserClass");
...
}
Codice:
public class MyClassLoader extends URLClassLoader{
public MyClassLoader(ClassLoader parent) {
super(new URL[0],parent);
}
@Override
public Class loadClass(String name) throws ClassNotFoundException {
if(!"monitor.resource.Resource".equals(name) ){
System.out.println("DELEGATING LOADING OF "+name+" TO PARENT CLASSLOADER");
return super.loadClass(name);
}
try {
if("monitor.resource.Resource".equals(name)){
System.out.println("LOADING monitor.resource.Resource...");
String url = "jar:file:C:\\Documents and Settings\\User15\\Documenti\\NetBeansProjects\\monitorv4.0.2\\dist\\monitorv4.0.2.jar!/";
this.addURL(new URL(url));
return this.findClass("monitor.resource.Resource");
}
...
}
Quote:
Ultima modifica di tylerdurden83 : 24-11-2010 alle 13:15. |
|
|
|
|
|
|
#9 |
|
Senior Member
Iscritto dal: Nov 2004
Messaggi: 691
|
Precisazione, il classCast (dovrebbe) essere dovuto appunto al fatto che monitor.resource.Resource era già stata caricata in precedenza in quanto contenuta in un jar che va necessariamente incluso nel classpath
|
|
|
|
|
|
#10 | |
|
Senior Member
Iscritto dal: Oct 2007
Città: Padova
Messaggi: 4131
|
Quote:
Quindi in questo caso credo non sia sufficiente il fatto che la risorsa (file jar) in cui è contenuto il bytecode della classe in questione (monitor.resource.Resource) si trovi nel classpath per giustificarne il caricamento da parte del system class loader: deve essere già stato incontrato un riferimento nel codice eseguito prima dell'esecuzione di: Codice:
...
Class myObjectClass = classLoader.loadClass("monitor.resource.Resource");
...
In base al mio ragionamento, andando per esclusione, gli unici candidati papabili sembrano essere MyClassLoaderTest e MyClassLoader. @EDIT: Qua trovi un articolo focalizzato (2003) sul meccanismo del class loading. @RI-EDIT: Altre due risorse che contengono ulteriori info (in una si parla pure di class unloading) e mi sembrano più complete link1 link2
__________________
As long as you are basically literate in programming, you should be able to express any logical relationship you understand. If you don’t understand a logical relationship, you can use the attempt to program it as a means to learn about it. (Chris Crawford) Ultima modifica di banryu79 : 24-11-2010 alle 15:09. |
|
|
|
|
|
|
#11 | |
|
Senior Member
Iscritto dal: Nov 2004
Messaggi: 691
|
Quote:
Codice:
Resource object1 = ... Codice:
MyClassLoader classLoader = new MyClassLoader(parentClassLoader);
Class<?> myObjectClass = classLoader.loadClass("monitor.resource.Resource");
Codice:
Resource object1 = (Resource) myObjectClass.getConstructor(...); Ultima modifica di tylerdurden83 : 24-11-2010 alle 15:20. |
|
|
|
|
|
|
#12 |
|
Senior Member
Iscritto dal: Oct 2007
Città: Padova
Messaggi: 4131
|
Sì, stiamo dicendo la stessa cosa, la mia era una precisazione (circa il discorso relativo ad avere o no il jar nel classpath ai fini del class loading).
L'errore è quello: hai già caricato una classe monitor.resource.Resource con il system classloader, quando poi carichi monitor.resource.Resource anche con il tuo custom classloader, esso internamente dovrebbe delegare normalmente al parent classloader anche il caricamento di questa classe, e solo se non la trova già caricata dal parent classloader si occupa lui di caricarla. Negli esempi canonici di cui ho letto/ho visto io di estensione del class loader, di solito non si fa l'override del metodo loadClass, ma del metodo findClass (normalmente è il motivo principale per cui ci si scrive un proprio class loader, ti rimando alle risorse che ti ho linkato, in particolare l'estratto su google book). A questo punto forse è meglio se ci spieghi cosa stai cercando di ottenere?
__________________
As long as you are basically literate in programming, you should be able to express any logical relationship you understand. If you don’t understand a logical relationship, you can use the attempt to program it as a means to learn about it. (Chris Crawford) |
|
|
|
|
|
#13 | |
|
Senior Member
Iscritto dal: Nov 2004
Messaggi: 691
|
Quote:
Se faccio come dici e cioè caricare una classe solo se nn la trova gia caricata dal parent classloader posso ottenere il secondo target (caricare classi non ancora caricate), ma non ricaricare "a caldo" una classe modificata (la cui versione precedente era gia stata caricata) |
|
|
|
|
|
|
#14 | |
|
Senior Member
Iscritto dal: Oct 2007
Città: Padova
Messaggi: 4131
|
Quote:
Questo perchè una classe XYZ, una volta che è stata caricata viene tenuta in cache; se al class loader chiedi di "ricaricarla" esso non la ricarica, la pesca dalla cache. Bisogna quindi vedere se (e come) è possibile scaricare una classe già caricata da un class loader, in generale; e poi nello specifico (se nel tuo caso sarà per forza di cose il system class loader quello coinvolto con il caricamento delle classi in questione) se e come scaricare una classe dal system class loader.
__________________
As long as you are basically literate in programming, you should be able to express any logical relationship you understand. If you don’t understand a logical relationship, you can use the attempt to program it as a means to learn about it. (Chris Crawford) |
|
|
|
|
|
| Strumenti | |
|
|
Tutti gli orari sono GMT +1. Ora sono le: 20:47.




















