Access android context in ndk application

前端 未结 2 1121
甜味超标
甜味超标 2020-12-03 08:51

Is there any way in which I can pass/get an object of android context in my ndk appliation. I want to use SharedPreferences in my ndk application via jni interf

相关标签:
2条回答
  • 2020-12-03 09:30

    Looks like things have changes recently and the solution above and few others posted on other SO posts didn't work for me. After few tries I was able to make following solution work. My goal was to pass Context Object to JNI and get absolute storage path.

    void Java_com_path_to_my_class_jniInit(JNIEnv* env, jobject thiz, jobject contextObject) {
    
        try {
             //Get Context Class descriptor
             jclass contextClass = env->FindClass("android/content/Context");
             //Get methodId from Context class
             jmethodID getFilesDirMethodId = env->GetMethodID(contextClass,"getFilesDir","()Ljava/io/File;");
    
             //Call method on Context object which is passed in
             jobject fileObject = env->CallObjectMethod(contextObject,getFilesDirMethodId);
    
             //Get File class descriptor
             jclass fileClass = env->FindClass("java/io/File");
             //Get handle to the method that is to be called
             jmethodID absolutePathMethodId = env->GetMethodID(fileClass,"getAbsolutePath","()Ljava/lang/String;");
             //Call the method using fileObject
             jstring stringObject = (jstring)env->CallObjectMethod(fileObject,absolutePathMethodId);
          }
          catch(exception& ex){
                JNIExceptionHelper::throwException(env, ex.what());
                return;
          }
    }
    
    0 讨论(0)
  • 2020-12-03 09:31

    There is nothing special you have to do, it is just like regular JNI mechanism. You need to get a pointer to the context object, then retrieve the method ID you want to call and then invoke it with the args you want.

    Of course in words it sounds super straightforward, but in code it gets really messy since the all the checks and JNI calls.

    So in my opinion i will not try to implement the whole thing from native/JNI code, instead i will implement a helper method in Java that makes all the stuff and just receives the needed data to read/write the preference.

    That will simplify a lot your native code and it will make it easier to maintain.

    eg:

    //Somewhere inside a function in your native code
    void Java_com_example_native_MainActivity_nativeFunction(JNIEnv* env, jobject thiz)
    {
        jclass cls = (*env)->FindClass(env,"PreferenceHelper");
        if (cls == 0) printf("Sorry, I can't find the class");
    
        jmethodID set_preference_method_id;
    
        if(cls != NULL)
        {
            set_preference_method_id = (*env)->GetStaticMethodID(env, cls, "setPreference", "(Ljava/lang/String;Ljava/lang/StringV");
    
            if(set_preference_method_id != NULL )
            {
                jstring preference_name = (*env)->NewStringUTF(env, "some_preference_name");
                jstring value = (*env)->NewStringUTF(env, "value_for_preference");
    
                (*env)->CallStaticVoidMethod(env, cls, get_main_id, preference_name, value);
            }
        }
    }
    

    Note that i just wrote the code from memory so expect not to work out of the box.

    0 讨论(0)
提交回复
热议问题