java本身建议创建ClassLoader子类,并完成类似addURL方法已达到动态加载类路径。但在第三方包运行时,对方已经实现了ClassLoader子类且未提供此方法。则可以利用JVMPI技术完成这一操作。
java 部分:
<com.myCom.myproj.myclass>
extClassLoader = Thread.currentThread().getContextClassLoader();
File extDir = new File( System.getProperty( "cyanea.home" )+"/lib/ext/" );
File extFiles[] = extDir.listFiles();
ArrayList loaders = null;
if( extFiles != null )
{
for( int i = 0 ; i < extFiles.length ; i ++ )
{
try
{
loaders = addClassPath( extClassLoader, extFiles[i].toURL() );
}catch( Exception e )
{
System.out.println( "exception in addClassPath" );
}
}
C 部分:
#include "classloader.h"
#pragma convlit(resume)
//it is a helper function to get the URL of a loader
//addClassPath will return an arraylist of loader, iterate the arraylist and call this function
//will print out the class loader loading sequence
JNIEXPORT jobject JNICALL Java_com_myCom_myproj_myclass_addClassPath( JNIEnv *env, jobject thisObject, jobject extclassloader, jobject url )
{
jobject loaders = NULL;
try{
if( extclassloader == NULL || url == NULL ) throw 0;
jclass extclassloaderclass = env->GetObjectClass( extclassloader );
//add code to deal with the ClassLoader Implement Class here.
//load
jfieldID ucpid = env->GetFieldID( extclassloaderclass, "ucp", "Lsun/misc/URLClassPath;" );
if( ucpid == NULL ) throw 5;
jobject ucp = env->GetObjectField( extclassloader, ucpid );
if( ucp == NULL ) throw 6;
jclass ucpclass = env->GetObjectClass( ucp );
jfieldID pathid = env->GetFieldID( ucpclass, "path", "Ljava/util/ArrayList;" );
if( pathid == NULL ) throw 7;
jobject path = env->GetObjectField( ucp, pathid );
if( path == NULL ) throw 8;
jclass pathclass = env->GetObjectClass( path );
jmethodID add2path = env->GetMethodID( pathclass, "add", "(ILjava/lang/Object;)V" );
if( add2path == NULL ) throw 9;
env->CallVoidMethod( path, add2path, 0, url );
if( env->ExceptionOccurred() ) throw 10;
jmethodID check = env->GetStaticMethodID( ucpclass, "check", "(Ljava/net/URL;)V" );
if( check == NULL ) throw 11;
env->CallStaticVoidMethod( ucpclass, check, url );
if( env->ExceptionOccurred() ) throw 12;
//create loader and add to loaders and lmap
jfieldID loadersID = env->GetFieldID( ucpclass, "loaders", "Ljava/util/ArrayList;" );
jfieldID lmapID = env->GetFieldID( ucpclass, "lmap", "Ljava/util/HashMap;" );
if( loadersID == NULL || lmapID == NULL ) throw 13;
loaders = env->GetObjectField( ucp, loadersID );
jobject lmap = env->GetObjectField( ucp, lmapID );
if( loaders == NULL || lmap == NULL )
{
throw 14;
}
jfieldID jarHandlerID = env->GetFieldID( ucpclass, "jarHandler", "Ljava/net/URLStreamHandler;" );
if( jarHandlerID == NULL ) throw 15;
jobject jarHandler = env->GetObjectField( ucp, jarHandlerID );
if( jarHandler == NULL ) printf( "jar handler is null" );
jclass jarloaderclass = env->FindClass( "sun/misc/URLClassPath$JarLoader" );
if( jarloaderclass == NULL )
{
throw 16;
}
jmethodID jarloaderconstructor = env->GetMethodID( jarloaderclass, "<init>", "(Ljava/net/URL;Ljava/net/URLStreamHandler;Ljava/util/HashMap;)V" );
if( jarloaderconstructor == NULL ) throw 17;
//create a new instance of JarLoader
jobject jarloader = env->NewObject( jarloaderclass, jarloaderconstructor, url, jarHandler, lmap );
if( jarloader == NULL ) throw 18;
jclass loadersclass = env->GetObjectClass( loaders );
jclass lmapclass = env->GetObjectClass( lmap );
//add jarloader to loaders
jmethodID add2loaders = env->GetMethodID( loadersclass, "add", "(ILjava/lang/Object;)V" );
if( add2loaders == NULL ) throw 19;
env->CallVoidMethod( loaders, add2loaders, 0, jarloader);
if( env->ExceptionOccurred() ) throw 20;
//put (url,jarloader) to lmap
jmethodID put2lmap = env->GetMethodID( lmapclass, "put", "(Ljava/lang/Object;Ljava/lang/Object;)Ljava/lang/Object;" );
if( put2lmap == NULL ) throw 21;
env->CallObjectMethod( lmap, put2lmap, url, jarloader );
if( env->ExceptionOccurred() ) throw 22;
}catch( int errcode )
{
loaders = NULL;
probelog( ERROR_LOG , "PROBE_DEBUG", "Error" );
}
return loaders;
}
#pragma convlit(suspend)