我真的很难弄清楚如何使用JNI从Objective-C调用Java函数。
我应该先说我对Java一无所知,但对Obj-C非常熟悉。我有一个Java类,我需要从我的应用程序包中调用一个方法。 jar位于bundle中的Resources文件夹内,我将NSJavaRoot
设置为Content/Resources
,NSJavaNeeded
被检查,NSJavaPath
包含2个jar文件的名称(主要文件名为1)一个依赖)。
我正在使用对JNI_CreateJavaVM
的调用启动虚拟机,然后尝试使用NSClassFromString
找到该类感觉错误,但这是我在搜索中找到的唯一方法。我相信这个方法在使用已弃用的Java桥时是正确的,但我找不到任何使用JNI的示例或引用。
我的Java类看起来像这样:
package foo;
public class bar {
public function dostuff() {
}
}
我需要在应用流程中调用dostuff()
一次。有没有人有任何想法如何做到这一点?
谢谢, Ĵ
答案 0 :(得分:12)
由于JavaBridge已被弃用,您将不得不使用JNI完成所有操作。在MacOS上,您需要将JavaVM.framework添加到项目中,并且从源文件中,您要查找的标题是:
#import <JavaVM/jni.h>
接下来,您需要设置JavaVM。这可能是非平凡的,具体取决于您的类路径和其他要求,但您是在正确的轨道上,因为这是您用于创建JVM的函数:
JNI_CreateJavaVM(JavaVM **pJVM, void **pJNIEnv, void *args);
接下来,您将希望获得对您的类foo.bar的引用。这可以使用从JNI_CreateJavaVM传递的pJNIEnv来完成。你想打电话:
jclass myClass = JNIEnv->FindClass(JNIEnv, "foo/bar"); // Note the slashes replacing the dots...
假设一切设置正确,您将获得对您班级的引用。假设foo.bar目前有一个默认的无参数构造函数,你可以得到它的一个实例,如下所示:
jobject myFooBar = JNIEnv->NewObject(JNIEnv, myClass);
现在您需要获取doStuff方法的methodID。要做到这一点,你需要它的方法签名,你可以通过调用javap来获得,如下所示:
% javap -s foo.bar
应该产生这样的输出:
Compiled from "bar.java"
public class foo.bar extends java.lang.Object{
public foo.bar();
Signature: ()V
public void doStuff();
Signature: ()V
}
然后你可以用它来获取你需要调用它的methodID。像这样:
jmethodID mid = JNIEnv->GetMethodID(JNIEnv, myClass, "doStuff", "()V");
假设所有这些事情都正确,那么你可以这样调用这个方法:
JNIEnv->CallVoidMethod(JNIEnv, myFooBar, mid);
并且应该调用该方法。在任何这些阶段之后,您可能希望检查VM以查看是否存在异常。您可以通过执行以下操作来检查是否发生了异常:
if (JNIEnv->ExceptionCheck(JNIEnv))
{
// Handle the exception...
}
如果你想要实际的throwable,你可以使用ExceptionOccurred JNI方法获得它。
所以,它最让人失望的是:如何使用JNI从Cocoa调用Java方法。您将需要阅读JNI文档,特别是有关理解全局和本地引用之间差异的部分。 (即使你的Cocoa端引用的持续时间足以在创建它们的范围之外调用。)事实上,有一整章关于常见的错误和陷阱,其中很多都是你会遇到的。
祝你好运!