JNI是否要求基于参数类型对参数进行排序

时间:2016-08-23 10:22:49

标签: java c++ android-ndk java-native-interface jnienv

我正在尝试将一些JNI代码集成到我的应用程序中,该应用程序从Web服务接收数据并在JNI层中解析它,我尝试通过传递从Web服务接收的一些参数来调用Java层中定义的方法。以下是功能和方法的流程:

为jclass和jmethodID定义的变量:

    static jclass userInfo                      = NULL ;
    static jmethodID create_user_info = NULL ;

在c层定义的struct以保存值

    struct User_Info {            
        long userId; // 1
        std::string firstName; // 2
        std::string lastName; // 3
        bool follow; // 4
    };

此功能有问题。在从webservice接收数据,解析它并创建struct User_Info

之后调用此函数
    static jobject createJavaObject(JNIEnv *env, const User_Info& info) {
        if(!userInfo){
            userInfo = (jclass)env->NewGlobalRef(env->FindClass("com/model/UserInfo"));
            assert(userInfo != NULL);
            create_user_info = env->GetStaticMethodID(userInfo, "createUserInfo", "("
                "I" // userId // 1
                "Ljava/lang/String;" // firstName // 2
                "Ljava/lang/String;" // lastName // 3
                "Z" // follow // 4
                ")Lcom/model/UserInfo;");
            assert(create_user_info != 0);
        }
        jstring firstName = cast<jstring>(env, info.first_name); // cast function just converts string to jstring 
        assert(firstName != NULL);
        jstring lastName = cast<jstring>(env, info.last_name);  // cast function just converts string to jstring 
        assert(lastName != NULL);

        // ISSUE IS WITH FOLLOWING CALL
        jobject res = env->CallStaticObjectMethod(userInfo, create_user_info,
                                                                  info.userId, // 1
                                                                  firstName, // 2
                                                                  lastName, // 3
                                                                  info.follow // 4 
                                                                  );

        env->DeleteLocalRef(firstName);
        env->DeleteLocalRef(lastName);
        return res;
    }


    // cast function:

    jstring cast<jstring, char const *>()(JNIEnv *env, char const *s)         {
        if (!s) return 0;
        jbyteArray bytes = env->NewByteArray(strlen(s));
        env->SetByteArrayRegion(bytes, 0, strlen(s), (jbyte*)s);
        jmethodID midStringCtor = env->GetMethodID(clsString, "<init>", "([BLjava/lang/String;)V");
        jstring ret = (jstring)env->NewObject(clsString, midStringCtor, bytes, encoding);
        env->DeleteLocalRef(bytes);
        return ret;
    }

Java方法:

@Native
private static UserInfo createUserInfo(
                            int userId, // 1
                            String firstname, // 2
                            String lastname, // 3
                            boolean follow // 4
                        ) {
    UserInfo info = new UserInfo();
    info.userId = userId;
    info.firstname = firstname;
    info.lastname = lastname;
    info.follow = follow;
    return info;
}   

问题是,当我运行代码时,我收到以下错误:

  
    

JNI ERROR(应用程序错误):预期的jboolean(0/1)但得到值1048605作为参数4到com.model.UserInfo com.model.UserInfo.createUserInfo(int,java.lang.String,java.lang .String,boolean)

  

当我更改java方法和jni函数中的参数顺序并在int'userId'之后保持boolean'follow'时,则不会发生问题。

我的问题是:

JNI是否需要根据类型对参数进行排序 例如long参数之前的int参数或对象之前的原始参数等。

如果是,是否有任何建议它的参考/文档,如果没有,那么为什么我的代码在更改参数顺序后工作?

1 个答案:

答案 0 :(得分:2)

  

是否需要在JNI函数参数

中保留一些基元和对象的顺序

这个问题几乎无法理解。所需要的是使JNI方法方法声明与在包含本机方法的类上运行javah所产生的一致。

static jclass userInfo                      = NULL ;

这是一个错误。保留对jobjectsjclasses的静态引用是无效的,除非它们是全局或弱引用。