java JNI 实现原理 (一)

一、虚拟机中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 来加载的。

[java]  view plain   copy
  1. static void loadLibrary(Class fromClass, String name,  
  2.                 boolean isAbsolute) {}  

每个classloader 对象都有自己的nativeLibrary 数组,一个全局的systemNativeLibrary 数组,一个全局的已经加载过的loadLibraryNames数组,和一个正在加载过程中的记录栈nativeLibraryContext

对同一个classloader 对象可以重复加载相同的库,对不同的classloader只可以加载一次相同的库。

1. 这里定义的相同的库是指相同路经下的同一个文件

2.  这里同样指出的是同一个classloader对象,而不是同一种classloader类型,比如说如果一种classloader类型初始化成2个classloader对象,那么这两个对象就不能重复加载相同的库。

3. 重复加载,并不代表真的重复加载,而是代码中保护

[java]  view plain   copy
  1. for (int i = 0; i < size; i++) {  
  2.             NativeLibrary lib = (NativeLibrary)libs.elementAt(i);  
  3.         if (name.equals(lib.name)) {  
  4.             return true;  
  5.         }  
  6.         }  

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  

[cpp]  view plain   copy
  1. JNIEXPORT void JNICALL   
  2. Java_java_lang_ClassLoader_00024NativeLibrary_load  
  3.   (JNIEnv *env, jobject this, jstring name)  
  4. {  
  5.     const char *cname;  
  6.     jint jniVersion;  
  7.     jthrowable cause;  
  8.     void * handle;  
  9.   
  10.     if (!initIDs(env))  
  11.         return;  
  12.   
  13.     cname = JNU_GetStringPlatformChars(env, name, 0);  
  14.     if (cname == 0)  
  15.         return;  
  16.     handle = JVM_LoadLibrary(cname);  
  17.     if (handle) {  
  18.         const char *onLoadSymbols[] = JNI_ONLOAD_SYMBOLS;  
  19.         JNI_OnLoad_t JNI_OnLoad;  
  20.     int i;  
  21.     for (i = 0; i < sizeof(onLoadSymbols) / sizeof(char *); i++) {  
  22.         JNI_OnLoad = (JNI_OnLoad_t)   
  23.             JVM_FindLibraryEntry(handle, onLoadSymbols[i]);  
  24.         if (JNI_OnLoad) {  
  25.             break;  
  26.         }  
  27.     }  
  28.     if (JNI_OnLoad) {  
  29.         JavaVM *jvm;  
  30.         (*env)->GetJavaVM(env, &jvm);  
  31.         jniVersion = (*JNI_OnLoad)(jvm, NULL);  
  32.     } else {  
  33.         jniVersion = 0x00010001;  
  34.     }  
  35.   
  36.     cause = (*env)->ExceptionOccurred(env);  
  37.     if (cause) {  
  38.         (*env)->ExceptionClear(env);  
  39.         (*env)->Throw(env, cause);  
  40.         JVM_UnloadLibrary(handle);  
  41.         goto done;  
  42.     }  
  43.      
  44.     if (!JVM_IsSupportedJNIVersion(jniVersion)) {  
  45.         char msg[256];  
  46.         jio_snprintf(msg, sizeof(msg),  
  47.              "unsupported JNI version 0x%08X required by %s",  
  48.              jniVersion, cname);  
  49.         JNU_ThrowByName(
  • 1
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值