вызов нестатических методов из java в cpp с использованием JNI

Я пытаюсь вызвать нестатический метод из java в C++ с помощью JNI. Мой Java-код находится здесь:

public class hellojava
{
  public static void main(String args[]) 
  {
     System.out.println("Hello World!");
     System.out.println("This is the main function from the HelloWorld java class.");
  }

public void message()
{
    System.out.println("call from object");

}

}

И мой код C++ здесь:

#include <stdio.h>
#include <jni.h>


JNIEnv* create_vm(JavaVM ** jvm) {

JNIEnv *env;
JavaVMInitArgs vm_args;
JavaVMOption options;

options.optionString = "-Djava.class.path=/home/../nonstaticJavaMethods/";     
//Path to the java source code

vm_args.version = JNI_VERSION_1_6; //JDK version. This indicates version 1.6
vm_args.nOptions = 1;
vm_args.options = &options;
vm_args.ignoreUnrecognized = 0;


int ret = JNI_CreateJavaVM(jvm, (void**)&env, &vm_args);
if(ret < 0)
    printf("\nUnable to Launch JVM\n");     
return env;
}

int main(int argc, char* argv[])
{
JNIEnv *env;
JavaVM * jvm;
env = create_vm(&jvm);
if (env == NULL)
    return 1;   


//jclass clsH=NULL;
jmethodID midMain = NULL;
jstring square;
jclass clsH = env->FindClass("helloWorld");
jmethodID constructor = env->GetMethodID(clsH, "<init>", "void(V)");
jobject object = env->NewObject(clsH, constructor);



//Obtaining Method IDs
if (clsH != NULL)
{          midMain = env->GetMethodID(clsH, "message", "void(V)");
           env->CallVoidMethod(clsH, midMain, object,NULL); 
}
else
{
    printf("\nUnable to find the requested class\n");       
}




//Release resources.
int n = jvm->DestroyJavaVM();
return 0;

}

My code compiles but it is giving me runtime error. Following is the error

Среда выполнения Java обнаружила фатальную ошибку:

 SIGSEGV (0xb) at pc=0x00007fdf3f5126bb, pid=11302, tid=140596827092800

JRE version: OpenJDK Runtime Environment (7.0_55-b14) (build 1.7.0_55-b14)
Java VM: OpenJDK 64-Bit Server VM (24.51-b03 mixed mode linux-amd64 compressed oops)
Problematic frame:
V  [libjvm.so+0x5c46bb]  alloc_object(_jclass*, Thread*)+0x1b

 Failed to write core dump. Core dumps have been disabled. To enable core dumping,    
 try    
 "ulimit -c unlimited" before starting Java again

 An error report file with more information is saved    as  

 /home/../nonstaticJavaMethods/hs_err_pid11302.log      
 Aborted!

person user3597719    schedule 03.06.2014    source источник


Ответы (2)


В дополнение к ответу immibis' я также думаю, что вызов

jclass clsH = env->FindClass("helloWorld");

ничего не возвращает, так как ваш класс называется

public class hellojava

Таким образом, ваше приложение, вероятно, имеет сбой сегментации в GetMethodID() или NewObject().

person Naytzyrhc    schedule 05.06.2014

void(V) не является допустимым дескриптором метода.

Поскольку не существует метода с именем <init> с дескриптором void(V) (которого не может быть, поскольку он недействителен), GetMethodID возвращает 0. Затем вы пытаетесь создать новый объект, используя этот недопустимый идентификатор метода.

Дескриптор метода, который не принимает аргументов и возвращает void (что является конструктором), равен ()V.

person user253751    schedule 03.06.2014
comment
@TomBlodget нет метода с именем <init> с дескриптором void(V). - person user253751; 05.06.2014
comment
О, да, не так просто для конструкторов. В любом случае, запрашивайте по имени класса, но используйте ‹init› в JNI. Вот лучший пример: javap -s -public java.lang.String | egrep -A 2 "String\(" - person Tom Blodget; 05.06.2014
comment
Конструкторы @TomBlodget такие же, как и другие методы, за исключением того, что имя всегда <init>, а тип возвращаемого значения всегда V (соответствует void в Java). void(V) не является допустимым дескриптором для любого метода — он должен быть ()V для метода, который не принимает параметров и возвращает void. - person user253751; 06.06.2014