我的目标是在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++
?也许我的开始想法很糟糕。
答案 0 :(得分:3)
所以,抱歉花了那么久,但我解决了问题。基本上,这是造成破坏的一个小错字(至少在你的MWE中):
this->IDgetString = this->JNIEnvironment->GetStaticMethodID(this->javaClass
, "getString"
, "(()Ljava/lang/String;");
// ^ This parenthesis must be removed.
此外,我注意到你没有初始化对象。我使用instace方法而不是静态方法重新编写代码,对我来说,MWE运行正常。
Here是我使用的EXICodec.java
代码,here是JavaInterface.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
我确信这只是一个错字,但如果没有,可能是原始例子中问题的原因。