- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我有这个代码:
JNI 调用的包装器:
/// <summary>
/// Simple JNI class wrapper
/// </summary>
struct SimpleJNIClass{
JavaVM* vm = nullptr;
jobject instanceRef = nullptr;
JNIEnv *lockedEnv = nullptr;
SimpleJNIClass(JNIEnv *env, jobject instanceRef){
//Returns the Java VM interface (used in the Invocation API) associated with the current thread.
//The result is placed at the location pointed to by the second argument, vm.
env->GetJavaVM(&this->vm);
this->instanceRef = (jobject)env->NewGlobalRef(instanceRef);
};
SimpleJNIClass(JNIEnv * env){
env->GetJavaVM(&this->vm);
};
virtual ~SimpleJNIClass() {
if (this->vm == nullptr) {
return;
}
if (JNIEnv * env = GetEnv()){
env->DeleteGlobalRef(instanceRef);
instanceRef = nullptr;
FreeEnv(env);
}
};
JNIEnv * LockEnv() {
this->lockedEnv = this->GetEnv();
return this->lockedEnv;
}
void UnLockEnv() {
this->lockedEnv = nullptr;
this->FreeEnv(this->lockedEnv);
}
JNIEnv * GetEnv(){
if (this->lockedEnv != nullptr) {
return this->lockedEnv;
}
JNIEnv * env = nullptr;
int getEnvStat = vm->GetEnv((void **)&env, JNI_VERSION_1_4);
if (getEnvStat == JNI_EDETACHED){
if (vm->AttachCurrentThread(&env, NULL) != 0){
MY_LOG_ERROR("AttachCurrentThread was not successful. "
"This may be due to the thread being attached already to another JVM instance.");
}
else{
attachCurrentThread = true;
}
}
else if (getEnvStat == JNI_OK){
//no need to attach, already attached
}
else if (getEnvStat == JNI_EVERSION){
MY_LOG_ERROR("GetEnv: version not supported");
}
return env;
};
void FreeEnv(JNIEnv * env) {
if (this->lockedEnv != nullptr) {
return;
}
if (attachCurrentThread) {
vm->DetachCurrentThread();
attachCurrentThread = false;
}
}
protected:
bool attachCurrentThread = false;
};
struct JNICallback : public SimpleJNIClass
{
jmethodID callbackMethod = nullptr;
JNICallback(JNIEnv *env, jobject classRef,
const std::string & methodName, const std::string & methodSignature)
: SimpleJNIClass(env, classRef) {
jclass callbackClass = (jclass)env->GetObjectClass(this->instanceRef);
this->callbackMethod = env->GetMethodID(callbackClass, methodName.c_str(), methodSignature.c_str());
};
virtual ~JNICallback() {
callbackMethod = nullptr;
}
template<typename... Args>
void RunVoid(Args... args) {
JNIEnv *env = this->GetEnv();
env->CallVoidMethod(this->instanceRef, this->callbackMethod, args...);
FreeEnv(env);
}
};
struct JNIClass : public SimpleJNIClass {
jclass callbackClass;
std::unordered_map<std::string, jmethodID> callbacks;
JNIClass(JNIEnv *env, jobject classRef)
: SimpleJNIClass(env, classRef) {
this->callbackClass = (jclass)env->NewGlobalRef((jclass)env->GetObjectClass(this->instanceRef));
};
JNIClass(JNIEnv *env, const std::string & classRefName)
: SimpleJNIClass(env) {
this->callbackClass = (jclass)env->NewGlobalRef(env->FindClass(classRefName.c_str()));
};
virtual ~JNIClass() {
if (JNIEnv * env = GetEnv()) {
env->DeleteGlobalRef(callbackClass);
callbackClass = nullptr;
FreeEnv(env);
}
callbacks.clear();
};
void AddMethod(const std::string & methodName, const std::string & methodSignature) {
JNIEnv *env = this->GetEnv();
this->callbacks[methodName] = env->GetMethodID(this->callbackClass, methodName.c_str(), methodSignature.c_str());
FreeEnv(env);
};
template<typename... Args>
void InitNewInstance(Args... args) {
JNIEnv *env = this->GetEnv();
this->SetInstance(env->NewObject(this->callbackClass, this->callbacks["<init>"], args...));
FreeEnv(env);
}
void SetInstance(jobject instRef) {
JNIEnv *env = this->GetEnv();
env->DeleteGlobalRef(this->instanceRef);
this->instanceRef = env->NewGlobalRef(instRef);
FreeEnv(env);
}
template<typename... Args>
void RunVoid(const std::string & name, Args... args) {
JNIEnv *env = this->GetEnv();
env->CallVoidMethod(this->instanceRef, this->callbacks[name], args...);
FreeEnv(env);
};
};
在 JNI_OnLoad 中我设置了全局变量 JNIClass * fcDataClass
:
fcDataClass = new JNIClass(env, "com/example/MyClass");
然后在来自 Android 的一些 JNI 回调中,我有:
int resCount = 3;
JNIEnv *lockedEnv = fcDataClass->LockEnv();
jobjectArray dataArray = lockedEnv->NewObjectArray(resCount, fcDataClass->callbackClass, NULL);
//do something with data
fcDataClass->UnLockEnv();
问题是,NewObjectArray
使整个应用崩溃:SIGSEGV(信号 SIGSEGV:无效地址(故障地址:0x9c))
使用调用栈:
art::JavaVMExt::JniAbort(char const*, char const*) 0x00000000ef74815c
art::JavaVMExt::JniAbortV(char const*, char const*, std::__va_list) 0x00000000ef7495ba
art::ScopedCheck::AbortF(char const*, ...) 0x00000000ef5d8eee
art::ScopedCheck::CheckThread(_JNIEnv*) 0x00000000ef5d8a0a
art::ScopedCheck::CheckPossibleHeapValue(art::ScopedObjectAccess&, char, art::JniValueType) 0x00000000ef5d7af2
art::ScopedCheck::Check(art::ScopedObjectAccess&, bool, char const*, art::JniValueType*) 0x00000000ef5d6fce
art::CheckJNI::NewObjectArray(_JNIEnv*, int, _jclass*, _jobject*) 0x00000000ef5d3808
_JNIEnv::NewObjectArray(int, _jclass*, _jobject*) jni.h:858
最佳答案
参见 CheckThread()来源。如果当前线程未附加,或者如果您尝试使用属于不同 trhead 的 JNIEnv*
,它将中止。
您的fcDataClass->LockEnv()
似乎不依赖于线程。获取正确的 JNIEnv*
的合法方法是调用 AttachCurrentThread()
( see how WebRTC does this )。不要忘记在处理此类线程之前使用 DetachCurrentThread()
。
关于android - JNI - 在 NewObjectArray 上崩溃,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/49456677/
我刚开始使用 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这样的更高级别,但仍然能够侵入进程内存。 有什么线索吗? []的 最佳答案 这
我是一名优秀的程序员,十分优秀!