遇到的问题,Native层本地多线程回调Java函数时env->findClass()失败。

前面的代码是这样的在 JNI_OnLoad记录全局变量g_vm static JavaVM* g_vm = NULL;

 JNIEXPORT jint JNICALL JNI_OnLoad(JavaVM* vm, void* reserved)

 {

     JNIEnv * env = NULL;

     if (g_vm == NULL)
{
g_vm = vm;
} if (g_vm->GetEnv((void**) &env, JNI_VERSION_1_4) != JNI_OK)
{ ALOGD("connot get g_Env is OK "); return JNI_ERR; } return JNI_VERSION_1_4; }

在子线程回调函数中去g_vm->AttachCurrentThread获取env,通过env去findClass代码如下这里发现cls == NULL

 int nativeCallBackJava(int id)

 {

     jint result;

     JNIEnv * env = NULL;

     if (g_vm == NULL)
{
return FALSE;
} result = (g_vm)->AttachCurrentThread(&env, NULL); if (result != JNI_OK)
{
return FALSE;
} cls = (env)->FindClass("com/lipeng/NativeCallJava"); if (cls == NULL)
{
return FALSE;
} mid = (env)->GetStaticMethodID(cls, "nativeNotifyJava", "(I)I"); if (mid == NULL)
{
return FALSE;
} (env)->CallStaticIntMethod(cls, mid, id); return TRUE; }

为什么会这样呢?我并没有看源码,觉得子线程AttachCurrentThread得到的env其类的加载器中并没有去加载自定义的类,所有这里你无法去findClass你自己的类,有人说这里find 系统类env->FindClass("java/lang/String");
这样是可以的,我没有做尝试。

解决方案如下,

Java

1. 需要的类里面有个public native void setClsRef(void);

2. 在初始的时候调用本地函数来设计该类的全局引用

本地Native定义全局引用static jobject g_ObjCall = NULL;

实现setClsRef函数

3.

 JNIEXPORT void JNICALL Java_com_lipeng_NativeCallJava_setClsRef(JNIEnv* env, jobject thiz)

 {

     if (g_ObjCall == NULL)
{ g_ObjCall = env->NewGlobalRef(thiz); //获取全局引用 if (g_ObjCall == NULL) { ALOGD("get g_ObjCall == NULL) "); } if (thiz != NULL)
{ env->DeleteLocalRef(thiz);
}//释放局部对象.这里可不要,调用结束后虚拟机会释放 }

4.调用

 int nativeCallBackJava(int id)

     {

         jint result;

         JNIEnv * env = NULL;

         if (g_vm == NULL)
{
return FALSE;
} result = (g_vm)->AttachCurrentThread(&env, NULL); if (result != JNI_OK)
{
return FALSE;
} cls = (env)->GetObjectClass(g_ObjAd); //从全局引用获取局部对象 if (cls == NULL)
{
return FALSE;
} mid = (env)->GetStaticMethodID(cls, "nativeNotifyJava", "(I)I"); if (mid == NULL)
{
return FALSE;
} (env)->CallStaticIntMethod(cls, mid, id); env->DeleteLocalRef(cls); //调用完后释放局部对象,这里是需要的,子线程调用该局部对象不会被虚拟机销毁,需显示调用Delete return TRUE; }

这样就可以在本地多线程随意使用了,回调Java层了,子线程结束后别忘了 g_vm->DetachCurrentThread()

Jni本地多线程回调Java函数,env->findClass()失败。的更多相关文章

  1. JNI通过线程c回调java层的函数

    1.参看博客:http://www.jianshu.com/p/e576c7e1c403 Android JNI 篇 - JNI回调的三种方法(精华篇) 2.参看博客: JNI层线程回调Java函数关 ...

  2. javap 可以打印出用于jni调用的java函数的签名信息

    javap可以打印出java的字节码: -c     Prints out disassembled code, i.e., the instructions that comprise the Ja ...

  3. jni不通过线程c回调java的函数

    整个工程的项目如下: 1.项目的思路是在activity中启动MyService这个服务,在服务中调用 JniScsManger类中的本地方法startNativeScsService,在 start ...

  4. jni不通过线程c回调java的函数 --总结

    1.JNIEnv类型是一个指向全部JNI方法的指针.该指针只在创建它的线程有效,不能跨线程传递 2.JavaVM是虚拟机在JNI中的表示,一个JVM中只有一个JavaVM对象,这个对象是线程共享的. ...

  5. Android jni中回调java的方法

    在上一篇的基础上,添加在C++代码中回调java方法. 代码如下: Demo.java 中添加callback函数, 打印一条log. package com.example.scarecrow.dy ...

  6. Android JNI编程(八)——体验AS2.2.2编写Jni程序、Java调C、C调Java函数、将C代码中的Log打印至Logcat

    版权声明:本文出自阿钟的博客,转载请注明出处:http://blog.csdn.net/a_zhon/. 目录(?)[+] 不得不说在AS2.2以上的版本进行开发就一个字——爽,在2.0上使用jni出 ...

  7. Jni层回调java代码【转】

    本文转载自:http://www.linuxidc.com/Linux/2014-03/97562.htm JNI是Java Native Interface的缩写,是Java平台的重要特性,使得Ja ...

  8. Android jni c/c++线程通过CallVoidMethod调用java函数出现奔溃问题

    最近在移植网络摄像机里的p2p库到android平台,需要用到jni,最近在c线程了调用java函数的时候 出现一个问题,假如在同一个线程调用java函数是没问题的,但在一个c线程了调用java函数就 ...

  9. Python 3.X 调用多线程C模块,并在C模块中回调python函数的示例

    由于最近在做一个C++面向Python的API封装项目,因此需要用到C扩展Python的相关知识.在此进行简要的总结. 此篇示例分为三部分.第一部分展示了如何用C在Windows中进行多线程编程:第二 ...

随机推荐

  1. 大理石在哪儿(UVa10474)

    题目具体描述见:https://uva.onlinejudge.org/index.php?option=com_onlinejudge&Itemid=8&category=835&a ...

  2. Solr本地服务器搭建及查询

    0.安装solr之前,确保已安装好java8,  java -version 查看是否安装 1.新建本地目录solr1 并 解压两个压缩包文件 .tar.gz .tgz tomcat7 2.将CATA ...

  3. vuejs学习——vue+vuex+vue-router项目搭建(二)

    前言 最近比较忙,所有第二章发布晚了,不好意思各位. vuejs学习——vue+vuex+vue-router项目搭建(一) 中我们搭建好了vue项目,我相信大家已经体验了vue其中的奥妙了,接下来我 ...

  4. 洛谷P2704 [NOI2001]炮兵阵地 [状压DP]

    题目传送门 炮兵阵地 题目描述 司令部的将军们打算在N*M的网格地图上部署他们的炮兵部队.一个N*M的地图由N行M列组成,地图的每一格可能是山地(用“H” 表示),也可能是平原(用“P”表示),如下图 ...

  5. Python并发编程-SocketServer多线程版

    #server.py import socket from threading import Thread def chat(conn): conn.send(b'hello') msg = conn ...

  6. JavaWeb 之 AJAX

    Ajax ajax:AJAX 是与服务器交换数据的艺术,它在不重载全部页面的情况下,实现了对部分网页的更新 AJAX:Asynchronous JavaScript and XML,异步 javasc ...

  7. Oracle意外赢官司,程序员或过苦日子

    关于“Google在Android平台使用Java侵犯知识产权”一案,2014年5月,联邦法院判定Oracle获胜,这个结果完全出人意料,因为这样一来无异于打开了软件开发领域中API使用方式的潘多拉之 ...

  8. redis_NoSql入门概述数据模型简介

    以下面的背景去对比关系型数据库和非关系型数据库的差异(一个电商客户.订单.订购.地址模型来对比以下关系型数据库和非关系型数据库) 传统数据库一般设计会使用ER图(1:1/1:N/N:N,主键等) 而N ...

  9. 【BZOJ 3443】 3443: 装备合成 (离线+线段树)

    3443: 装备合成 Time Limit: 15 Sec  Memory Limit: 128 MBSubmit: 63  Solved: 31 Description [背景]     lll69 ...

  10. 【BZOJ 4568】 4568: [Scoi2016]幸运数字 (线性基+树链剖分+线段树)

    4568: [Scoi2016]幸运数字 Description A 国共有 n 座城市,这些城市由 n-1 条道路相连,使得任意两座城市可以互达,且路径唯一.每座城市都有一个 幸运数字,以纪念碑的形 ...