在native线程利用JNI 反射自定义类--ndk开发参考2

在native线程利用JNI 反射自定义类

zxqianrong   

 

NDK编程中遇到的一些细节问题,希望对大家有帮助

                                                                                                   -----题记

在JNI中,有时候出于业务要求需要实现异步事件机制,例如网络通讯的收发

这时就会在C++中回调java类的方法,于是就会用到java反射机制

在JNI中,实现类反射主要用到以下几个方法:(本例以反射静态方法为例)

JavaVM             jint GetEnv(void **penv, jint version)

JavaVM             jint AttachCurrentThread(void **penv, void *args)

JNIEnv              jclass FindClass(const char *name)

JNIEnv              jmethodID GetStaticMethodID(jclass clazz, const char *name,  const char *sig)

JNIEnv            void CallStaticVoidMethod(jclass cls, jmethodID methodID, ...)

 JavaVM           jint DetachCurrentThread()

假设我们要反射的类是:

package com.genius.test; public class InflectClass {  public static void test(int cmd,String value,String data){  }  }

那么C++中回调的JNI代码就是:

void testInflect(int cmd, const char* value, const char* data) { if (g_vm == NULL) { return ; } int status; JNIEnv *env = NULL; bool isAttach = false; status = g_vm->GetEnv((void **) &env, JNI_VERSION_1_4); if(status != JNI_OK)  { status = g_vm->AttachCurrentThread(&env, NULL); if(status < 0) { return; } isAttach = true; }	  jstring valueString = NULL; jstring dataString = NULL; jclass inflectClass = NULL; jmethodID inflectMethod = NULL;  jclass inflectClass = env->FindClass("com/genius/test/InflectClass"); if (inflectClass == NULL) { return; }  jmethodID inflectMethod= env->GetStaticMethodID(inflectClass, "test", "(ILjava/lang/String;Ljava/lang/String;)V"); if (inflectMethod == NULL) { return ; }  valueString = env->NewStringUTF(value); dataString = env->NewStringUTF(data);  env->CallStaticVoidMethod(inflectClass, inflectMethod, cmd, valueString, dataString);  end: if (env->ExceptionOccurred()) { env->ExceptionDescribe(); env->ExceptionClear(); } if (isAttach) { g_vm->DetachCurrentThread(); }  env->DeleteLocalRef(dataString);  }

其中g_vm是全局的虚拟机实例,可以在

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

函数中保存该实例

 

jint GetEnv(void **penv, jint version)是获取当前线程对应的JNI环境指针

在JNI中,多线程间JNIEnv 是不可以共享的,所以不能全局保存使用

获取是有可能失败的,比如当前是native线程

何为native线程?即在jni中开启的线程。

而java线程则是调用native方法的宿主线程,可以是主线程也可以是子线程

在获取失败时,调用jint AttachCurrentThread(void **penv, void *args)来将当前线程附加到虚拟机并获取JNI环境指针

 

之后就是通过

jclass FindClass(const char *name)

GetStaticMethodID(jclass clazz, const char *name,  const char *sig)

void CallStaticVoidMethod(jclass cls, jmethodID methodID, ...)

来获取类方法并调用

 

GetStaticMethodID方法的第三个参数是函数签名

主要是c++中重载函数的存在使得函数名不能作为区别函数方法的唯一标示

这时就需要区别参数列表,获取函数签名的命令是javap

具体使用方法大家百度一下就知道了

 

最后如果有附加当前线程到虚拟机的话

需要调用DetachCurrentThread方法来释放线程

否则线程不能正常结束

 

温馨提示:

理论上通过上面几个步骤就可以反射到java层了

但实际操作起来会发现在native线程里执行FindClass的时候会找不到该自定义类(系统类可以)

具体原因不详,解决方案如下:

JNI_OnLoad里获取该类对象并保存一个全局引用

JavaVM *g_vm = NULL; jclass g_inflectClass = NULL; jmethodID g_methodID = NULL;
void InitInflectClass(JavaVM* vm) { g_vm = vm;  JNIEnv *env = NULL; int status = g_vm->GetEnv((void **) &env, JNI_VERSION_1_4); if(status != JNI_OK)  { return ; }   jclass inflectClass = env->FindClass("com/genius/test/InflectClass"); if (inflectClass == NULL) { return ; }  g_inflectClass = inflectClass; g_methodID = env->GetStaticMethodID(inflectClass, "test", "(ILjava/lang/String;Ljava/lang/String;)V"); if (g_methodID == NULL) { return ;
} }  直接把函数方法保存下来也可以

然后在C++回调的时候直接使用该全局类或函数方法即可

这样类就反射出来了得意

参考1:http://www.cfanz.cn/?c=article&a=read&id=74827

----------------------------------------------------

参考2:

Android培训班(74)Dalvik虚拟机的FindClass函数

分类: Android培训班   3405人阅读  评论(0)  收藏  举报
从前面我们知道,在虚拟机初始化后,运行Java代码的方法时,要先查找到类,也就是调用函数FindClass。接着后面分析怎么样从dex文件加载类数据到内存,现在开始对查找函数FindClass进行分析,就很好理解了,因为前面介绍加载类到内存的流程已经很清楚。函数FindClass代码如下:
static jclass FindClass(JNIEnv* env, const char* name)
{
这个函数传入两个参数,第一个参数是jni的环境参数;第二个参数name是需要查找的类描述名称。


    JNI_ENTER();


    const Method* thisMethod;
    ClassObject* clazz;
    jclass jclazz = NULL;
    Object* loader;
    char* descriptor = NULL;


    thisMethod = dvmGetCurrentJNIMethod();
    assert(thisMethod != NULL);
这段代码是获取当前正在运行的JNI加载方法。


    descriptor = dvmNameToDescriptor(name);
    if (descriptor == NULL) {
        clazz = NULL;
        goto bail;
    }
这段代码是把查找类的名称转换类描述的方式,以便从类索引表里查找到。




    //Thread* self = dvmThreadSelf();
    if (_self->classLoaderOverride != NULL) {
        /* hack for JNI_OnLoad */
        assert(strcmp(thisMethod->name, "nativeLoad") == 0);
        loader = _self->classLoaderOverride;
这段代码是使用本线程重载加载方法,也就是使用 nativeLoad的本地方法。


    } else if (thisMethod == gDvm.methFakeNativeEntry) {
        /* start point of invocation interface */
        if (!gDvm.initializing)
            loader = dvmGetSystemClassLoader();
        else
            loader = NULL;
这段代码是使用系统里的类加载器。


    } else {
        loader = thisMethod->clazz->classLoader;
这行代码是使用本类里的加载器。


    }


    clazz = dvmFindClassNoInit(descriptor, loader);
这行代码是使用指定类加载器来查找给出描述名称的类。


    jclazz = addLocalReference(env, (Object*) clazz);
这行代码是增加类的引用计数。


bail:
    free(descriptor);


    JNI_EXIT();
    return jclazz;

}


//QQ:9073204 EMAIL:9073204@qq.com

//蔡军生  2011-10-6

引用自:http://blog.csdn.net/caimouse/article/details/6846547

-------------------------------------------------------

参考3:http://stackoverflow.com/questions/7248468/findclass-cannot-find-custom-java-class

参考4:http://stackoverflow.com/questions/7248468/findclass-cannot-find-custom-java-class

参考5:http://stackoverflow.com/questions/13263340/findclass-from-any-thread-in-android-jni

参考6:

2011.12.05(6)——— android JNI学习之五 JNI中常用的方法 

参考: http://blog.csdn.net/fontlose/article/details/6600122 
http://www.cnblogs.com/daniel-shen/archive/2006/10/16/530587.html 
http://blog.csdn.net/yang_hui1986527/article/details/6887052 
http://www.cnblogs.com/liangwind/archive/2009/08/26/1925511.html 
http://www.cnblogs.com/luxiaofeng54/category/315742.html 



Java代码   收藏代码
  1. ===============================================================================================================  
  2. //找到类文件  
  3. jclass  objectClass = (*env)->FindClass(env, "com.lp.MailInfo");  
  4. //或者 obj 参数表示要你想要得到类型的类对象。   
  5. jclass class_Field = (*env)->GetObjectClass(env,obj);     
  6.   
  7. //得到构造函数  
  8. jmethodID mid_date = (*env)->GetMethodID(env,objectClass, "<init>""()V");  
  9. //生成对象  
  10. jobject o = (*env)->NewObject(env,jclass,jmethod);  
  11. //jclass类 构造方法 后面的就是构造方法的参数  可以没有    
  12.   
  13. 例如   
  14. jobject     book;    
  15. jclass      class_book;     
  16. jmethodID  md_book;     
  17. class_book = (*env)->FindClass(env,"LBook1;");     
  18. md_book = (*env)->GetMethodID(env,class_book,"<init>","(IILjava/lang/String;)V");    
  19. book = (*env)->NewObject(env,class_book,md_book,100,1,"huanghe");    
  20.   
  21.   
  22. ===============================================================================================================  
  23. http://www.cnblogs.com/daniel-shen/archive/2006/10/16/530587.html  
  24.   
  25. 表 3-2 Java虚拟机类型签名   
  26. 类型签名 Java 类型  
  27. boolean   
  28. byte  
  29. char  
  30. short  
  31. int  
  32. long 注意 这个是J  
  33. float  
  34. double  
  35. L fully-qualified-class ; 全限定的类  
  36. [type type[]   
  37. (arg-types)ret-type 方法类型  
  38. 例如,Java 方法:  
  39.     long fun(int n, String s, int[] arr);  
  40. 具有以下类型签名:  
  41.     (ILjava/lang/String;[I)J  
  42.   
  43.       
  44. 属性  
  45. 访问对象实例域的相关函数如下:   
  46. jfieldID GetFieldID(JNIEnv *env, jclass cl, const char name[], const char sig[]) 获取实例对象的域ID  
  47. 该函数返回一个域的标识符。各参数含义如下:   
  48. env JNI 接口指针;cl 类对象 ; name 域名; sig 编码的域签名   
  49.   
  50. XXX GetXXXField(JNIEnv *env, jobject obj, jfieldID id)   
  51. 该函数返回域的值。域类型XXX是Object, Boolean, bytechar , shortint ,long ,float, double 中类型之一。   
  52. 参数 env JNI借口指针;obj为域所在对象;id为域的标识符。   
  53. void SetXXXField(JNIEnv *env,jobject obj, jfieldID id, XXX value)   
  54.   
  55. 例如:  
  56.   
  57. //招到类中名为"topic"的string类型的属性  
  58. jfieldID topicFieldId = (*env)->GetFieldID(env, objectClass,"topic""Ljava/lang/String;");  
  59. jstring str = (*env)->GetStringField(env,obj,topicFieldId);  
  60. jfieldID ival = (env)->GetFieldID(objectClass,"serial","I");  
  61. (env)->SetIntField(obj,ival,10);  
  62.   
  63.   
  64. 方法  
  65.   
  66.   
  67. 得到方法id  
  68. static jmethodID message;  
  69. static jmethodID audio;  
  70. static jmethodID video;  
  71.   
  72. video = (*env)->GetMethodID(env,ljls,"video_data_cb","([BI)I");   
  73. audio = (*env)->GetMethodID(env,ljls,"audio_data_cb","([BI)I");    
  74. message = (*env)->GetMethodID(env,ljls,"message_cb","(SI)I");  
  75.   
  76. 调用方法  
  77. int ret = (*env)->CallIntMethod(env,g_meida_obj,audio,jarray,size);   
  78. env obj对象 方法id 参数  
  79.   
  80. 静态方法 不推荐调用  
  81. GetStaticMethodID  
  82.   
  83.   
  84. ===============================================================================================================  
  85. jbyteArray 转换为unsigned char *  
  86. unsigned char *src;  
  87.   
  88. jbyte *arr = (*env)->GetByteArrayElements(env,yuvBuf,0);  
  89. src = (char *)arr;  
  90.   
  91. unsigned char *转换为jbyteArray  
  92. unsigned char *dst;  
  93. jbyte *byte = (jbyte*)dst;  
  94. jbyteArray rgbBuf = (*env)->NewByteArray(env,width*height*3);  
  95. (*env)->SetByteArrayRegion(env,rgbBuf, 0, width*height*3byte);  
  96. //释放  
  97. (*env)->ReleaseByteArrayElements(env,yuvBuf,arr,0);  
  98.   
  99. ===============================================================================================================  
  100. 让object成为全局变量  
  101.   
  102. 1、定义全局变量  
  103. static jobject g_meida_obj;  
  104. 2、声明全局变量  
  105. g_meida_obj = (*env)->NewGlobalRef(env,obj);  
  106.   
  107. 貌似用jmethodID不需要特殊设置 直接定义复制即可  
  108.   
  109. ===============================================================================================================  
  110.   
  111. 关于线程  
  112.   
  113. 1、定义全局变量  
  114.   
  115. static JavaVM   *g_jvm;  
  116.   
  117. 2、赋值  
  118.   
  119. (*env)->GetJavaVM(env,&g_jvm);  
  120.   
  121. 3、你在需要用的地方 特别是不是jni的方法 要想使用当前线程的JNIEnv   
  122.   
  123. JNIEnv *env;  
  124. (*g_jvm)->AttachCurrentThread(g_jvm,&env, NULL);   
  125. 。。。。。。  
  126. (*g_jvm)->DetachCurrentThread(g_jvm);  
  127.   
  128.   
  129. ===============================================================================================================  
  130. 回收内存  
  131.   
  132. jclass cls =  env->FindClass("sinashow1android/info/BlackIPUserInfo");  
  133. jobject black_ip_obj = env->NewObject(cls,construction_id);  
  134.   
  135. //回收局部对象的内存  
  136. env->DeleteLocalRef(black_ip_obj);  
  137. env->DeleteLocalRef(cls);  
  138.   
  139. 线程处理是 需要放开  
  140. (*g_jvm)->DetachCurrentThread(g_jvm);  
  141.   
  142.   
  143. ===============================================================================================================  
  144. char 转换为 string  
  145. jstring chartojstring(JNIEnv *env, const char *chars)  
  146.     {  
  147.         jstring result;  
  148.         jint len = strlen(chars);  
  149.         if(len <= 0)  
  150.         {  
  151.             result = env->NewStringUTF("");  
  152.             return result;  
  153.         }  
  154.   
  155.         jclass stringClass;  
  156.         jmethodID cid;  
  157.         jbyteArray elemArr;  
  158.         jstring jencoding;  
  159.   
  160.         stringClass = env->FindClass("java/lang/String");  
  161.         if (stringClass == NULL) {  
  162.             return NULL;   
  163.         }  
  164.   
  165.         cid = env->GetMethodID(stringClass,"<init>""([BLjava/lang/String;)V");  
  166.         if (cid == NULL) {  
  167.             return NULL;   
  168.         }  
  169.   
  170.         jencoding = env->NewStringUTF("GBK");  
  171.         elemArr = env->NewByteArray(len);  
  172.         if (elemArr == NULL) {  
  173.             return NULL;   
  174.         }  
  175.         env->SetByteArrayRegion(elemArr, 0, len, (jbyte*)chars);  
  176.         result = (jstring)(env->NewObject(stringClass, cid, elemArr, jencoding));  
  177.         env->DeleteLocalRef(elemArr);  
  178.         env->DeleteLocalRef(stringClass);  
  179.         env->DeleteLocalRef(jencoding);  
  180.         return result;  
  181.     }  



数组的常用操作: 

 

对象的常用操作: 


 



另外 

1、 
Java代码   收藏代码
  1. http://blog.csdn.net/yang_hui1986527/article/details/6887052  


这个里面 有JNI各个方法的说明  可以当做api 

2、
Java代码   收藏代码
  1. 推荐一个网址 里面关于jni的学习内容比较多  


http://blog.csdn.net/kangyaping/article/details/6584027 




引用自:http://lipeng88213.iteye.com/blog/1292570

-------------------------------------------------

参考7:http://blog.csdn.net/kangyaping/article/details/6584027

参考8:http://docs.oracle.com/javase/1.5.0/docs/guide/jni/spec/functions.html

参考9:http://doc-snapshot.qt-project.org/qt5-stable/qtbluetooth-index.html


  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值