一、虚拟机中classloader的JNILibrary
调用JNI的时候,通常我们使用System.loadLibrary(String libname)来load JNI library, 同样也可以使用System.load(String fileName)来load JNI library,两者的区别是一个只需要设置库的名字,比如如果libA.so 只要输入A就可以了,而libA.so的位置可以同过设置 Java.library.path 或者 sun.boot.library.path,后者输入的是完整路经的文件名。
而不论用什么方法,最后JNI 库是通过classloader 来加载的。
- static void loadLibrary(Class fromClass, String name,
- boolean isAbsolute) {}
每个classloader 对象都有自己的nativeLibrary 数组,一个全局的systemNativeLibrary 数组,一个全局的已经加载过的loadLibraryNames数组,和一个正在加载过程中的记录栈nativeLibraryContext
对同一个classloader 对象可以重复加载相同的库,对不同的classloader只可以加载一次相同的库。
1. 这里定义的相同的库是指相同路经下的同一个文件
2. 这里同样指出的是同一个classloader对象,而不是同一种classloader类型,比如说如果一种classloader类型初始化成2个classloader对象,那么这两个对象就不能重复加载相同的库。
3. 重复加载,并不代表真的重复加载,而是代码中保护
- for (int i = 0; i < size; i++) {
- NativeLibrary lib = (NativeLibrary)libs.elementAt(i);
- if (name.equals(lib.name)) {
- return true;
- }
- }
4. 如果加载其他classloader已经加载过的库,会抛出 UnsatisfiedLinkError ERROR
在tomcat上,在不同的war包里,想加载相同的库文件,因为在 tomcat上是使用不同的classloader的对象去加载不同的war包,建议库文件放置在不同的路径通过System.load去加载。
二、
Linux 下如何 load JNILibrary
在博客java JNI (一)虚拟机中classloader的JNILibrary 中讨论了Java中的Library 是由classloader 来load的,那我们来看看 classloader是如何去load 一个library的
ClassLoader.c
- JNIEXPORT void JNICALL
- Java_java_lang_ClassLoader_00024NativeLibrary_load
- (JNIEnv *env, jobject this, jstring name)
- {
- const char *cname;
- jint jniVersion;
- jthrowable cause;
- void * handle;
-
- if (!initIDs(env))
- return;
-
- cname = JNU_GetStringPlatformChars(env, name, 0);
- if (cname == 0)
- return;
- handle = JVM_LoadLibrary(cname);
- if (handle) {
- const char *onLoadSymbols[] = JNI_ONLOAD_SYMBOLS;
- JNI_OnLoad_t JNI_OnLoad;
- int i;
- for (i = 0; i < sizeof(onLoadSymbols) / sizeof(char *); i++) {
- JNI_OnLoad = (JNI_OnLoad_t)
- JVM_FindLibraryEntry(handle, onLoadSymbols[i]);
- if (JNI_OnLoad) {
- break;
- }
- }
- if (JNI_OnLoad) {
- JavaVM *jvm;
- (*env)->GetJavaVM(env, &jvm);
- jniVersion = (*JNI_OnLoad)(jvm, NULL);
- } else {
- jniVersion = 0x00010001;
- }
-
- cause = (*env)->ExceptionOccurred(env);
- if (cause) {
- (*env)->ExceptionClear(env);
- (*env)->Throw(env, cause);
- JVM_UnloadLibrary(handle);
- goto done;
- }
-
- if (!JVM_IsSupportedJNIVersion(jniVersion)) {
- char msg[256];
- jio_snprintf(msg, sizeof(msg),
- "unsupported JNI version 0x%08X required by %s",
- jniVersion, cname);
- JNU_ThrowByName(