- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我无法从我的 Android 应用程序的 native 部分回调 Java(Kotlin)。这是一个音频应用程序,所以你会看到“很多音频”这个词,但我认为问题与此无关。
我已经阅读了很多文章并对此进行了几天的实验,但我无法解决我的问题。我可以从我的 native_lib.cpp
触发我的 java 回调。 ,所以我已经测试了所有这些,并且我的 JniBridge
之间的方法引用我的 native 代码应该是正确的。
但是我想从一个单独的类中完成它,而且我还必须能够从不同的线程中多次完成它。所以我想尽可能永久地存储我对 JVM 回调类的引用,但我没有做对。
我当前的代码因以下错误而失败:
A: java_vm_ext.cc:542] JNI DETECTED ERROR IN APPLICATION: JNI GetMethodID called with pending exception java.lang.ClassNotFoundException: Didn't find class "com.my.app.common.jni.JniBridge" on path: DexPathList[[directory "."],nativeLibraryDirectories=[/system/lib64, /system/vendor/lib64, /system/lib64, /system/vendor/lib64]]
A: java_vm_ext.cc:542] at java.lang.Class dalvik.system.BaseDexClassLoader.findClass(java.lang.String) (BaseDexClassLoader.java:134)
A: java_vm_ext.cc:542] at java.lang.Class java.lang.ClassLoader.loadClass(java.lang.String, boolean) (ClassLoader.java:379)
A: java_vm_ext.cc:542] at java.lang.Class java.lang.ClassLoader.loadClass(java.lang.String) (ClassLoader.java:312)
A: java_vm_ext.cc:542]
A: java_vm_ext.cc:542] in call to GetMethodID
A: java_vm_ext.cc:542] "Thread-6" prio=10 tid=17 Runnable
A: java_vm_ext.cc:542] | group="main" sCount=0 dsCount=0 flags=0 obj=0x13080000 self=0x7ca400e800
A: java_vm_ext.cc:542] | sysTid=30175 nice=-16 cgrp=default sched=1073741825/2 handle=0x7c911014f0
A: java_vm_ext.cc:542] | state=R schedstat=( 5738847 149269 9 ) utm=0 stm=0 core=1 HZ=100
A: java_vm_ext.cc:542] | stack=0x7c91006000-0x7c91008000 stackSize=1009KB
A: java_vm_ext.cc:542] | held mutexes= "mutator lock"(shared held)
A: java_vm_ext.cc:542] native: #00 pc 00000000003cb654 /system/lib64/libart.so (art::DumpNativeStack(std::__1::basic_ostream<char, std::__1::char_traits<char>>&, int, BacktraceMap*, char const*, art::ArtMethod*, void*, bool)+220)
#include <jni.h>
class AudioCallback {
public:
explicit AudioCallback(JavaVM&, jobject&);
void playBackProgress(int progressPercentage);
private:
JavaVM& mJvm;
jobject& mObject;
};
#include <jni.h>>
#include <utils/logging.h>
#include "AudioCallback.h"
AudioCallback::AudioCallback(JavaVM &jvm, jobject &object) : mJvm(jvm), mObject(object) {
}
void AudioCallback::playBackProgress(int progressPercentage) {
JNIEnv *g_env = NULL;
int getEnvStat = mJvm.GetEnv((void **) &g_env, JNI_VERSION_1_6);
JavaVMAttachArgs vmAttachArgs;
vmAttachArgs.version = JNI_VERSION_1_6;
vmAttachArgs.name = NULL;
vmAttachArgs.group = NULL;
if (getEnvStat == JNI_EDETACHED) {
LOGD("GetEnv: not attached - attaching");
if (mJvm.AttachCurrentThread(&g_env, &vmAttachArgs) != 0) {
LOGD("GetEnv: Failed to attach");
}
} else if (getEnvStat == JNI_OK) {
LOGD("GetEnv: JNI_OK");
} else if (getEnvStat == JNI_EVERSION) {
LOGD("GetEnv: version not supported");
}
if (g_env != NULL) {
jclass target = g_env->FindClass("com/my/app/common/jni/JniBridge");
jmethodID id = g_env->GetMethodID(target, "integerCallback", "(I)V");
g_env->CallVoidMethod(mObject, id, (jint) progressPercentage);
} else {
LOGE("JNIEnv is null!");
mJvm.DetachCurrentThread();
}
}
native_lib.cpp
,我在
JNI_OnLoad
中获得了 JavaVM像这样:
JavaVM *jvm;
jint JNI_OnLoad(JavaVM *vm, void *reserved) {
JNIEnv jvm_env;
int getEnvStatus = vm->GetEnv((void **) &jvm_env, JNI_VERSION_1_6);
if (getEnvStatus != JNI_OK) {
LOGE("JNI_ONLOAD Failed to get the environment using GetEnv()");
return -1;
}
jvm = vm;
if (jvm == NULL) {
LOGE("JNI_ONLOAD: globabl jvm is NULL");
} else {
LOGD("JNI_ONLOAD: global jvm is NOT NULL");
}
LOGD("Onload done");
return JNI_VERSION_1_6;
}
AudioEngine
使用它的类:
JNIEXPORT void JNICALL
Java_com_my_app_common_jni_JniBridge_playFromJNI(JNIEnv *env, jobject instance,jstring URI) {
// Here I instantiate my callback
callback = std::make_unique<AudioCallback>(*jvm, instance);
// below code sets up my audio engine class, which calls the AudioCallback during playback.
// This works without problems.
const char *uri = env->GetStringUTFChars(URI, NULL);
AMediaExtractor *extractor = AMediaExtractor_new();
if (extractor == nullptr) {
LOGE("Could not obtain AMediaExtractor");
return;
}
media_status_t amresult = AMediaExtractor_setDataSource(extractor, uri);
if (amresult != AMEDIA_OK) {
LOGE("Error setting extractor data source, error %d", amresult);
}
audioEngine = std::make_unique<AudioEngine>(*extractor, *callback);
audioEngine->setFileName(uri);
audioEngine->start();
}
audioEngine
使用
oboe用于音频处理的库,因此我无法控制其中可能的线程创建,因此
AttachCurrentThread()
在我的回调类中。附加和分离工作,唯一失败的是对 Java 的回调。
最佳答案
我主要自己解决了这个问题(阅读评论中@Michael的建议以及那里提到的其他一些地方肯定有帮助!):
首先,我将“设置”部分与实际的回调函数分开(这一直是计划,但不这样做会妨碍看到问题)。
二、在我的native_lib.cpp
我创建了一个对我的 jobject
的全局引用(问题示例中 playFromJNI
中的“实例”对象),然后将其传递给 AudioCallback's
像这样的构造函数:
myJNIClass = env->NewGlobalRef(instance);
callback = std::make_unique<AudioCallback>(*g_jvm, myJNIClass);
playbackProgress
我只在必要时处理 ÀttachToCurrentThread ,并使用我事先用正确数据初始化的类成员。
AudioCallback
实现现在看起来像:
#include <jni.h>>
#include <utils/logging.h>
#include "AudioCallback.h"
jclass target;
jmethodID id;
AudioCallback::AudioCallback(JavaVM &jvm, jobject object) : g_jvm(jvm), g_object(object) {
JNIEnv *g_env;
int getEnvStat = g_jvm.GetEnv((void **) &g_env, JNI_VERSION_1_6);
LOGD("Env Stat: %d", getEnvStat);
if (g_env != NULL) {
target = g_env->GetObjectClass(g_object);
id = g_env->GetMethodID(target, "integerCallback", "(I)V");
}
}
void AudioCallback::playBackProgress(int progressPercentage) {
JNIEnv *g_env;
int getEnvStat = g_jvm.GetEnv((void **) &g_env, JNI_VERSION_1_6);
if (getEnvStat == JNI_EDETACHED) {
LOGD("GetEnv: not attached - attaching");
if (g_jvm.AttachCurrentThread(&g_env, NULL) != 0) {
LOGD("GetEnv: Failed to attach");
}
} else if (getEnvStat == JNI_OK) {
LOGD("GetEnv: JNI_OK");
} else if (getEnvStat == JNI_EVERSION) {
LOGD("GetEnv: version not supported");
}
g_env->CallVoidMethod(g_object, id, (jint) progressPercentage);
}
关于c++ - 如何解决 "Didn' t 在路径 : DexPathList in my native callback to Java 上找到类 "...",我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/61168825/
E/AndroidRuntime(4999): Caused by: java.lang.ClassNotFoundException: Didn't find class "com.android.
在运行并收到短信后,我在此代码中使用了接收器,显示此错误。我尝试一些事情。我创建新项目并再次添加文件。删除bin文件。清理并重新启动项目。但它不能解决错误。有什么可以解决这个错误。我真的解决不了。
我试图将我的项目更新到 targetSdk 23,但鉴于所有的弃用,我决定我不适合它。我不想经历它,所以我在 mercurial 上恢复到旧版本,现在我得到的只是这个错误,我无法让该死的应用程序再次运
我的 Samsung S3(4.4 SDK 版本)有时会出现此异常: Caused by java.lang.ClassNotFoundException: Didn't find class "an
说真的,我不知道该怎么做才能解决这个问题。我的 android 项目运行良好,直到我需要使用 Maven 导入一个库。从那以后,一切都开始崩溃。 发生了什么:当我启动我的应用程序时,LogCat 上出
当我在 android studio 中使用 USB 在真实设备中测试时,我的应用程序运行良好。但是当我将 apk 发送到另一台设备并在那里安装时,它无法正常工作。应用程序崩溃了。我没有找到任何正确的
我在 Playstore 上更新了我的应用后遇到了问题。自本次更新以来,引发了异常,但我没有更改与此异常相关的任何内容。 堆栈跟踪: java.lang.RuntimeException: Unabl
我想按照本指南在 Ubuntu x64 上构建和运行 Dalvik VM:PATH/TO/AOSP/dalvik/docs/hello-world.html . 我的代码分支和主机是这样的: ====
谁能给我解释一下这个错误的原因吗?我知道 android 中的 dexpath 和 dex 类是什么以及如何修复此错误。但不知道时不时弹出这个错误的原因是什么。 此错误发生在以下情况 我将项目从一台电
我正在 Android 上构建我的第一个应用程序。我收到错误“在路径 dexpathlist 上找不到类 scrollview”。你能帮帮我吗? 这是我的 main.xml:
我正在使用 KeyGenParameterSpec 来定义我的 key 规范。当我运行应用程序时,我在日志中收到以下错误: Caused by: java.lang.ClassNotFoundExce
我的应用程序最近崩溃了,无论我做什么都无法工作。它之前工作得很好,但是我一定在重构一些变量/类时犯了一个错误。从 LOGCAT 返回的错误提到了一个我从未在项目中使用或实现过的类 NotifyStar
我在我的应用程序中使用 Multidex,这导致了一些问题,导致我的应用程序在安装后立即打开它时意外崩溃,当我通过我的 android studio 运行它时它工作正常,但当我尝试使用安装应用程序时我
直到昨天我的应用程序运行良好,今天突然开始崩溃并出现 Retrofit2 错误所以我通过升级依赖项并将 multidex 添加到我的应用程序来解决它,问题是现在当我尝试启动应用程序时它没有找到任何 A
我正在开发一个 Android 库来对我的应用程序中的一些工作进行模块化,我使用改造来使用一些网络 API。 当我在同一个 Android Studio 项目中的示例应用程序(编译库模块)中编译和使用
将 Gradle 版本迁移到 7.2 后,运行应用程序时出现此错误。 我注意到它指向 Java 8 日期/时间功能,LocalDate . 我的项目已经有 coreLibraryDesugaringE
我是 Xamarin 的新手,正在尝试让 OneSignal 推送通知在 Android 中运行。当运行模拟器并且我从 OneSignal 发出推送通知时,它在模拟器上成功通过。但是,当我通过 Pla
我从 jni 开始,尝试让一个简单的 HelloWorld 示例正常工作。我遇到了链接错误,我猜这与我的 native 库没有指向正确的文件路径有关。 我必须使用命令:java -Djava.libr
我遇到了以下错误,并且我已经尝试了 Stackoverflow 中的几乎所有解决方案。我是android新手,所以可能我不明白原因。我正在按照说明测试 libgdx 项目:https://github
这是我从导师那里得到的项目,在我的测试手机上运行后出现错误。我尝试了 stackovetflow 上的答案,但没有任何帮助。 logcat和manifest.xml如下
我是一名优秀的程序员,十分优秀!