CallObjectMethod 不是 return 字符串

CallObjectMethod does not return a string

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

这是我的总结 Java class :

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;
    }

以下是用于与 class 交互的 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 的调用没有 return 任何东西。 我试图将 Java 方法更改为 static 但它没有改变。

更一般地说,您如何 return 从 JavaC++ 的字符串?可能我的初衷不好。

很抱歉让您等了这么久,但我已经解决了问题。基本上,这是一个造成严重破坏的小错字(至少在你的 MWE 中):

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

此外,我注意到您没有初始化对象。我重构了你的代码以使用 instace 方法而不是静态方法,对我来说,MWE 运行 很好。

Here is the code for EXICodec.java I used and 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

我确定这只是一个拼写错误,但如果不是,则可能是您原始示例中出现问题的原因。