- iOS/Objective-C 元类和类别
- objective-c - -1001 错误,当 NSURLSession 通过 httpproxy 和/etc/hosts
- java - 使用网络类获取 url 地址
- ios - 推送通知中不播放声音
我正在使用 JNI 调用一个静态 java 方法,该方法又创建一个 Swing JFrame 并显示它。代码相当简单,Java 代码独立运行(即 java StartAWT
做它应该做的事),而当使用 JNI 从 C 调用时,进程挂起。
我在 Mac OS X 10.8 Mountain Lion 上使用 JDK 1.7.0_09。
这是我用来调用静态方法的 C 代码:
JavaVM* jvm;
JNIEnv* env = create_vm(&jvm);
jclass class = (*env)->FindClass(env, "StartAWT");
jmethodID method = (*env)->GetStaticMethodID(env, class, "run", "()V");
(*env)->CallStaticVoidMethod(env, class, method);
(*jvm)->DestroyJavaVM(jvm);
StartAWT
类如下所示:
public class StartAWT {
public static class Starter implements Runnable {
public void run() {
System.out.println("Runnning on AWT Queue.");
JFrame.setDefaultLookAndFeelDecorated(true);
JFrame frame = new JFrame("That's a frame!");
JLabel label = new JLabel("A Label");
frame.getContentPane().add(label);
frame.pack();
frame.setVisible(true);
}
}
public static class GUI implements Runnable {
public void run() {
try {
System.out.println("Going to put something on the AWT queue.");
SwingUtilities.invokeAndWait(new Starter());
} catch (Exception exc) {
throw new RuntimeException(exc);
}
}
}
public static void run() {
Thread gui = new Thread(new GUI());
gui.start();
}
}
当我启动应用程序时,我确实看到了Going to put something on the AWT queue
但没有看到 Running on AWT Queue
。
我相信我的 C 进程中的虚拟机没有 AWT 事件队列,但我不知道如何设置它以拥有一个(我也不确定这是原因)。
要使用 JNI 显示基于 AWT 的 GUI,需要做什么?
--
编辑: 我插入了循环以查看哪些线程处于 Activity 状态,哪些线程不活动(可以在 this gist 中看到)。在此版本中,我在另一个线程中调用了 SwingUtilities.invokeAndWait
。结果:主线程处于 Activity 状态 (C)。 Java 派发的第一个线程(不是主线程)是存活的;执行 Call invokeAndWait
的线程被阻塞(我认为 invokeAndWait 甚至没有返回),甚至没有输入应该在 EventQueue 上运行的函数。
我也尝试过直接调用 SwingUtilities.invokeAndWait
,这将给出以下消息:
2013-02-02 13:50:23.629 swing[1883:707] Cocoa AWT: Apple AWT Java VM was loaded on first thread -- can't start AWT. (
0 liblwawt.dylib 0x0000000117e87ad0 JNI_OnLoad + 468
1 libjava.dylib 0x00000001026076f1 Java_java_lang_ClassLoader_00024NativeLibrary_load + 207
2 ??? 0x000000010265af90 0x0 + 4335185808
)
这也是我在 StackOverflow 上的其他问题中读到的内容,例如下面评论中建议的问题。但是,我找不到解决原始问题的方法。也许值得注意的是,在出现上述消息后,主线程仍然存在,即进程既没有死锁也没有崩溃。
--
编辑: 我在 Linux 上测试了代码,它按预期工作。所以我相信这是 Cocoa AWT 的 Mac OS X 问题,但我不知道如何规避它。
--
编辑:我还尝试将 JVM 的整个调用移动到一个新的 native 线程上。这适用于带有 Apples Java 32 位 (1.6.0_37) 的 Mac OS X 10.6,但会导致与上述相同的死锁。在 Mac OS X 10.8 上,情况更糟,应用程序崩溃并显示唯一消息“Trace/BPT trap: 5”(seems to be related to loading dynamic libraries)。
我还尝试按照描述捆绑二进制文件 in this Q&A ,但启动失败并显示消息 lsopenurlswithrole() failed with the message -10810
,根据 Apples Launch Services Reference,这是一个未知错误.后者也在没有尝试使用 AWT 的情况下发生(仅仅 JVM 调用失败)。
最佳答案
终于找到了解决办法。
问题不在于在哪个线程上创建虚拟机,问题在于在哪个线程上初始化 AWT 事件队列。换句话说:第一次加载 AWT 类时,它可能不会在主线程上加载。因此第 1 步:加载(例如)java.awt.Component
在另一个线程上。
但现在 EventQueue 将阻塞,因为它将工作委托(delegate)给未运行的 Cocoa Main Event Queue - 果然如此,因为它只会在主线程上运行,而主线程是我的应用程序。因此主运行循环需要在主线程上启动:
void
runCocoaMain()
{
void* clazz = objc_getClass("NSApplication");
void* app = objc_msgSend(clazz, sel_registerName("sharedApplication"));
objc_msgSend(app, sel_registerName("run"));
}
我必须将我的应用程序与 Cocoa 框架链接起来并包含 <objc/objc-runtime.h>
.主线程在调用 runCocoaMain 后被阻塞(因为事件循环在那里运行),因此需要为应用程序本身求助于另一个线程。
使用上面的代码片段运行 EventQueue 后,另一个线程上的 AWT 类加载将成功,您可以在那里继续。
关于Java JNI : Creating a Swing Window using JNI from C,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/14661249/
我刚开始使用 JNI,但遇到以下问题。 我有一个包含简单类的 C++ 库。我从 Java Android 项目中调用了三个 JNI 方法,分别实例化所述类、调用实例化类的方法并销毁它。我保留了对该对象
背景 我有一个 android 项目,它使用 JNI(使用 NDK)以 Java 和 C/C++ 进行编码。 我在java端创建了一个Jni java包装器,它将自己完成所有Jni操作,而除了这个包装
我想传递一个java对象的地址,JNI方法应该填充所传递对象的字段。1. java代码传递给定的对象引用。 JNI 方法应该能够缓存这个对象。这是一次性通话。2. 然后,java 对象使用不同的参数重
我本质上是在尝试遵循描述的 SO 解决方案 here , 但我遇到了问题。 这基本上就是我在 JNI 库中所做的事情: handle = dlopen("/data/data/lib/my.packa
我试图实现某种异常处理 一开始我打电话 jni::ExceptionDescribe() 之后我使用了的实现 How to obtain a description of a Java excepti
是否可以从不同 jni 库的另一个方法调用 1 个库的 jni 方法?例如:我有 2 个库 lib_1.so 和 lib_2.so。 我想从 lib_2.so 调用 lib_1.so 的方法 get_
我想在native方法中使用动态注册,所以我需要设置JNI_onLoad功能。我只是写了一个函数来获取两个数字的总和。但是,它无法正确构建。我该如何更正错误? 这是我的 *.cpp 文件,我将此文件命
我已经为 C 头文件制作了一个 make 文件,它工作正常,但是说 JNICALL 和 JNIEnv 存在语法错误,但我已经弄明白了这是因为头文件中的类型。 Image of the failure
我需要实现一个本地方法,比方说“public native void someFunc();”。我有两个库,libabc.so 和 libdef.so。 Java 使用 System.loadLibr
背景 我正在 eclipse 中为 android 开发一个应用程序,现在我遇到了一个问题,我需要你的帮助。所以我必须从 JAVA 应用程序调用用 C 编写的函数。但是在我编写代码的过程中,我有一些问
我正在使用 Android 上的 Java native 接口(interface)将当前 Activity 传递给 native 方法。但我没有使用类似 JNI 的函数名称来执行此操作。我正在手动注
我正在使用 JNI,我想知道是否可以通过 delegate 进行通信。 例如: Kotlin typealias MessageReceived = (msg: String) -> Unit ext
看来我对 JNI 的运气并不好。我正在等待我买的书到货,但现在是试错法。 我正在使用 JNI 来实现 Lua 求值器。 evaluatorNew() 只是创建一个新的 Evaluator() 对象,创
我有下面的代码,我想调用在同一个源文件中实现的函数,在本例中使用 C 语言: JNIEXPORT jstring JNICALL MyClass_get_1Uname__C (JNIEnv *env,
硬件手机和平板电脑内存太少,但 HAXM 工作正常? 我的基本问题是我的应用程序(很可能是我的 Java 应用程序加载的 JNI 动态库)太大。如果未使用硬件电话和模板调用 JNI 指令,Java 应
我正面临崩溃 JNI WARNING : 0x44f81e80 is not a valid JNI reference, in Ldalvik/system/NativeStart;. run()v
我移植了很多数学知识。我正在使用 over to c++ from java 并看到这样做有很大的性能提升,但我无法弄清楚要使用什么 jni 函数来摆脱我不再需要的变量。例如,我知道当您的 jni 方
我正在使用 JNI 调用一个静态 java 方法,该方法又创建一个 Swing JFrame 并显示它。代码相当简单,Java 代码独立运行(即 java StartAWT 做它应该做的事),而当使用
我正在尝试创建一个新线程,因此我将 VM 从我的方法初始化(从 Java 调用)传递到我的新线程。在线程中,我调用 AttachCurrentThread 并获取 JNIEnv* env。 稍后,我尝
我想知道是否有可能从java调用C++方法。 我非常希望能够从 java 读取内存进程。 我懂c++,但我想使用像java这样的更高级别,但仍然能够侵入进程内存。 有什么线索吗? []的 最佳答案 这
我是一名优秀的程序员,十分优秀!