CallObjectMethod不返回字符串

时间:2015-05-29 08:23:52

标签: java c++ java-native-interface

我的目标是在Java中创建C++类的实例,然后调用Java类中定义的方法。

以下是我的Java课程摘要:

EXICodec.java

public class EXICodec {
    ...
    private static String inputXML;
    ...
    public EXICodec()
    {
        System.out.println("Constructor");  
        this.inputXML = "string";
        ...
    }
    ...
    public static void setInputXML(String inXML)
    {
        inputXML = inXML;
    }
    ...
    public static  String getInputXML()
    {
        return inputXML;
    }

以下是用于与该类进行交互的C++代码的摘要:

JavaInterface.hh

class JavaInterface
{
    private:
        JNIEnv*     JNIEnvironment;
        JavaVM*     javaVM;
        jclass      javaClass;
        jobject     javaClassInstance;

        jmethodID   IDconstructor;
        jmethodID   IDsetInputXML;
        jmethodID   IDgetInputXML;

    public:
        JavaInterface();                    
        void init_context( void);               
        void init_class( void);
        void init_methods( void);                   

        void setInputXML( std::string);
        std::string getInputXML( void);
}

JavaInterface.cc

void JavaInterface::init_context(){..}    //initiate the JNIEnvironment & javaVM attributes
void JavaInterface::init_class(){..}      //initiate the javaClass attribute
void JavaInterface::init_methods()
{
     this->IDconstructor = this->JNIEnvironment->GetMethodID(this->javaClass, "<init>", "()V");
     if (this->IDconstructor == NULL) 
     {
         throw std::runtime_error("JAVA_INIT_METHOD_EXCEPTION");
     }

     this->IDsetInputXML = this->JNIEnvironment->GetStaticMethodID(this->javaClass, "setInputXML", "(Ljava/lang/String;)V");
     if (this->IDsetInputXML == NULL) 
     {
         throw std::runtime_error("JAVA_INIT_METHOD_EXCEPTION");
     }

     this->IDgetInputXML = this->JNIEnvironment->GetStaticMethodID(this->javaClass, "getInputXML", "()Ljava/lang/String;");

     if (this->IDgetInputXML == NULL) 
     {
         throw std::runtime_error("JAVA_INIT_METHOD_EXCEPTION");
     }
}      
void JavaInterface::j_constructor()
{
    this->javaClassInstance = this->JNIEnvironment->NewObject(this->javaClass, this->IDconstructor);      
    if (this->javaClassInstance == NULL) 
    {
        throw std::runtime_error("JAVA_CONSTRUCTOR_CALL_EXCEPTION");
    }
}

void JavaInterface::j_setInputXML( std::string str)
{   
    jstring argument = this->JNIEnvironment->NewStringUTF(str.c_str());
    this->JNIEnvironment->CallVoidMethod(this->javaClassInstance, this->IDsetInputXML, argument);
}

std::string JavaInterface::j_getInputXML()
{
    jstring javaString = NULL;

    // ---> PROBLEM HERE : javaString is still NULL after the line below : <---
    javaString = (jstring)this->JNIEnvironment->CallObjectMethod(this->javaClassInstance, this->IDgetInputXML, 0);  

    // jString to char*
    const char *nativeString = this->JNIEnvironment->GetStringUTFChars(javaString, JNI_FALSE);
    // char* to std::string
    std::string str(nativeString);

    this->JNIEnvironment->ReleaseStringUTFChars(javaString, nativeString);

    return str;
}   

所以我的问题是CallObjectMethod的调用不会返回任何内容。 我试图将Java方法更改为static,但它没有改变。

更一般地说,如何将字符串从Java返回到C++?也许我的开始想法很糟糕。

1 个答案:

答案 0 :(得分:3)

所以,抱歉花了那么久,但我解决了问题。基本上,这是造成破坏的一个小错字(至少在你的MWE中):

 this->IDgetString = this->JNIEnvironment->GetStaticMethodID(this->javaClass
                                    , "getString"
                                    , "(()Ljava/lang/String;");
                                    // ^ This parenthesis must be removed.

此外,我注意到你没有初始化对象。我使用instace方法而不是静态方法重新编写代码,对我来说,MWE运行正常。

Here是我使用的EXICodec.java代码,hereJavaInterface.cpp的代码。

我再次看了你原始代码,发现了别的东西。

void JavaInterface::init_methods()
{
    this->IDconstructor = this->JNIEnvironment->GetMethodID(this->javaClass, "<init>", "()V");
    if (this->IDconstructor == NULL) 
    {
        throw std::runtime_error("JAVA_INIT_METHOD_EXCEPTION");
    }

    this->IDsetInputXML = this->JNIEnvironment->GetStaticMethodID(this->javaClass, "setInputXML", "(Ljava/lang/String;)V");
    if (this->IDsetInputXML == NULL) 
    {
        throw std::runtime_error("JAVA_INIT_METHOD_EXCEPTION");
    }

    this->IDgetInputXML = this->JNIEnvironment->GetStaticMethodID(this->javaClass, "getInputXML", "()Ljava/lang/String;");
//} <- This brace must be further down
    if (this->IDgetInputXML == NULL) 
    {
        throw std::runtime_error("JAVA_INIT_METHOD_EXCEPTION");
    }
} // <- To here

我确信这只是一个错字,但如果没有,可能是原始例子中问题的原因。