从C++调用Java

16
我将尝试按照这里的示例,从C++中调用Java虚拟机(JVM)。
基本上,我有一个小的Java程序:
public class TestJNIInvoke
{
    public static void main(String[] args)
    {
    System.out.println(args[0]);
    }
}

我有一个C++程序,想要创建一个JVM并调用TestJNIInvoke类:

#include <jni.h>
#include <cstdlib>
#define PATH_SEPARATOR ';' /* define it to be ':' on Solaris */
#define USER_CLASSPATH "." /* where Prog.class is */

using namespace std;

int main() {
     JNIEnv *env;
     JavaVM *jvm;
     jint res;
     jclass cls;
     jmethodID mid;
     jstring jstr;
     jclass stringClass;
     jobjectArray args;

 #ifdef JNI_VERSION_1_2
     JavaVMInitArgs vm_args;
     JavaVMOption options[1];
     options[0].optionString =
         "-Djava.class.path=" USER_CLASSPATH;
     vm_args.version = 0x00010002;
     vm_args.options = options;
     vm_args.nOptions = 1;
     vm_args.ignoreUnrecognized = JNI_TRUE;
     /* Create the Java VM */
     res = JNI_CreateJavaVM(&jvm, (void**)&env, &vm_args);
 #else
     JDK1_1InitArgs vm_args;
     char classpath[1024];
     vm_args.version = 0x00010001;
     JNI_GetDefaultJavaVMInitArgs(&vm_args);
     /* Append USER_CLASSPATH to the default system class path */
     sprintf(classpath, "%s%c%s",
             vm_args.classpath, PATH_SEPARATOR, USER_CLASSPATH);
     vm_args.classpath = classpath;
     /* Create the Java VM */
     res = JNI_CreateJavaVM(&jvm, &env, &vm_args);
 #endif /* JNI_VERSION_1_2 */

     if (res < 0) {
         fprintf(stderr, "Can't create Java VM\n");
         exit(1);
     }
     cls = (*env)->FindClass(env, "TestJNIInvoke");
     if (cls == NULL) {
         goto destroy;
     }

     mid = (*env)->GetStaticMethodID(env, cls, "main",
                                     "([Ljava/lang/String;)V");
     if (mid == NULL) {
         goto destroy;
     }
     jstr = (*env)->NewStringUTF(env, " from CPP!");
     if (jstr == NULL) {
         goto destroy;
     }
     stringClass = (*env)->FindClass(env, "java/lang/String");
     args = (*env)->NewObjectArray(env, 1, stringClass, jstr);
     if (args == NULL) {
         goto destroy;
     }
     (*env)->CallStaticVoidMethod(env, cls, mid, args);

 destroy:
     if ((*env)->ExceptionOccurred(env)) {
         (*env)->ExceptionDescribe(env);
     }
     (*jvm)->DestroyJavaVM(jvm);
 }

但是当我尝试编译C++程序时,出现了以下错误:
c:\java\JNI> g++ -I"c:\Program Files\Java\jdk1.7.0\include"-I"c:\ProgramFiles\Java\jdk1.7.0\include\win32" -c TestJNIInvoke.cpp


TestJNIInvoke.cpp: In function 'int main()':
TestJNIInvoke.cpp:20:31: warning: deprecated conversion from string constant to
'char*'
TestJNIInvoke.cpp:44:18: error: base operand of '->' has non-pointer type 'JNIEn
v'
TestJNIInvoke.cpp:49:18: error: base operand of '->' has non-pointer type 'JNIEn
v'
TestJNIInvoke.cpp:54:19: error: base operand of '->' has non-pointer type 'JNIEn
v'
TestJNIInvoke.cpp:58:26: error: base operand of '->' has non-pointer type 'JNIEn
v'
TestJNIInvoke.cpp:59:19: error: base operand of '->' has non-pointer type 'JNIEn
v'
TestJNIInvoke.cpp:63:12: error: base operand of '->' has non-pointer type 'JNIEn
v'
TestJNIInvoke.cpp:66:16: error: base operand of '->' has non-pointer type 'JNIEn
v'
TestJNIInvoke.cpp:67:16: error: base operand of '->' has non-pointer type 'JNIEn
v'
TestJNIInvoke.cpp:69:12: error: base operand of '->' has non-pointer type 'JavaVM'

有什么想法吗?

谢谢


你应该包含一条关于你的代码有什么问题的注释。你是收到了错误还是它只是无法正常工作? - Kiley Naro
我不知道如何从C++中调用Java,但是你声明了env作为一个指针(JNIEnv *env),所以当你执行(*env)->anything时,它会抱怨因为*env不是一个指针。尝试使用env->anything - dantswain
或者,您可以使用点(.)运算符代替箭头(->)。JVM对象也存在同样的问题:(*jvm)。此外,在某些地方,字符串字面值被隐式转换为char *,因此您可能希望将此转换_显式_化。 (这将消除警告)。 - user268396
我看到你的代码本质上与Sun在他们的网站上发布的代码相同。你使用的是与示例相同版本的Java吗?如果你找到了一个不同的使用JNI的示例,会怎么样? - Greg Hewgill
2个回答

38
即使你包含了相同的头文件,Java Native Interface(JNI)也会为C和C++分别使用两个不同的接口。
在C++中,它是这样的:
jclass cls = env->FindClass("java/lang/String");

使用(适用于C语言)的替代方法:

jclass cls = (*env)->FindClass(env, "java/lang/String");

因此,在C语言中需要在两个位置使用 env 的函数调用在C++中变成了一个方便的成员函数调用。

这在Java Native Interface 6.0规范本地方法参数部分中有提到。


非常感谢,我不知道有不同的接口。这将使我的生活变得更加轻松。 :) - Hunter McMillen

1

我的猜测是你试图使用你使用的命令行编译win32头文件。你尝试过使用-I"c:\ProgramFiles\Java\jdk1.7.0\include\solaris"吗?(假设基于源代码中更高层次的评论,这是你的平台)


这不是头文件的问题,而是语法不正确。通过指针(->)访问对象的成员/方法的语法被误用,应该使用通过指针解引用(* env)提供值的语法。 - user268396
啊,对不起,我同意。我没有注意到滚动条,只看了代码片段顶部的代码,其中没有包含任何问题代码。 - russw_uk

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接