多次加载本机库

时间:2013-06-01 19:44:28

标签: android android-ndk android-library

我创建了一个简单的本机库,可以存储一个整数并返回它。

#include <string.h>
#include <jni.h>
static int a;

void Java_com_example_testnativelibs_TestClass_setA(JNIEnv* env, jobject javaThis, jint val){
    a = val;
}
jint Java_com_example_testnativelibs_TestClass_getA(JNIEnv* env, jobject javaThis) {
      return a;
}

这是TestClass代码:

public class TestClass {
    public TestClass() {
        System.loadLibrary("ndkfoo2");
    }
public native void setA(int val);
public native int getA();
}

然后是我的MainActivity的代码:

TestClass a = new TestClass();
TestClass b = new TestClass();
a.setA(5);
b.setA(2);
Log.i("A VALUE",""+a.getA());
Log.i("B VALUE",""+b.getA());

值2在日志中显示两次,这意味着该库仅加载一次,并且由同一类的所有实例“共享”。是否可以多次加载,每个类实例一个?

1 个答案:

答案 0 :(得分:2)

没有。 Linux(Android)上的共享库只加载一次进程。这就是为什么你应该非常非常少地在共享库中使用全局数据 - 数据在整个过程中是全局的。

相反,您的库应该生成并使用某种“状态”变量(结构,指针等),以便在其函数的调用之间跟踪数据。


不幸的是我从未使用过JNI,因此我不知道相关的API调用来实现这一点。

在普通的Linux中,你可能会有类似的东西:

公共标题文件

typedef void* PublicState;  // Don't let consumers know anything about the
                            // state that we're keeping.

PublicState MyLib_init();
void        MyLib_free(PublicState state)
int         MyLib_getVal(PublicState state);

私有C实施文件

// This is the actual "state" object we keep track of information with.
typedef struct _PrivateState {
    int a;
} PrivateState;


PublicState MyLib_init() {
    PrivateState* state = malloc( sizeof(PrivateState) );
    // check for errors

    state->a = 42;

    return (PublicState)state;
}

int MyLib_getVal(PublicState state) {
    PrivateState* ps = (PrivateState*)state;

    return ps->a;
}

void MyLib_free(PublicState state) {
    // any other cleanup
    free(state);
}

我甚至不知道这是否是你在JNI中“应该”这样做的方式。

另见: