[cocos2dx]在cocos2dx中通过Jni实现Java与C++的互相调用(一)

[cocos2dx]在cocos2dx中通过Jni实现Java与C++的互相调用(一),第1张

概述JNI相关知识 JNI的意思是java本地接口,通过jni可以实现java层代码和其他语言写得代码进行互调。在cocos2d-x中,如果想要在c++层调用java层的代码,就是通过jni技术。通过调用java层的代码,我们就可以在Android平台下实现一些引擎没有提供给我们的功能,或者做一些其他的功能。比如加个广告,加个分享,调用Android原生的对话框等等吧。Cocos2d-x比较人性化的是 JNI相关知识 JNI的意思是java本地接口,通过jni可以实现java层代码和其他语言写得代码进行互调。在cocos2d-x中,如果想要在c++层调用java层的代码,就是通过jni技术。通过调用java层的代码,我们就可以在AndroID平台下实现一些引擎没有提供给我们的功能,或者做一些其他的功能。比如加个广告,加个分享,调用AndroID原生的对话框等等吧。Cocos2d-x比较人性化的是为我们封装了jni调用的一些接口,这个类就是JniHelper,我们只需要使用这个类提供给我们的接口就可以完成调用java层代码的功能,该文件位于cocos2d/cocos/platform/androID/jni目录下。 要想使用JNI,必须得包含头文件,androID是使用ndk编译c/c++的,这里jni.h文件位于:\androID-ndk-r8b\platforms\androID-14\arch-arm\usr\include\jni.h。 JniHelper类 通过JNI获取Java虚拟机,再获取当前程序的JNI环境,通过JNI环境获取需要调用的java类信息,再获取需要调用的java类中的函数信息。再通过JNI环境调用,使用类信息、函数信息,调用对应的java函数通过看JniHelper.cpp文件可以大致清楚这些。 头文件: #include "platform/androID/jni/JniHelper.h" 需要使用的接口如下:
static bool getStaticmethodInfo(JniMethodInfo &methodinfo,const char *classname,const char *methodname,const char *paramCode);static bool getmethodInfo(JniMethodInfo &methodinfo,const char *paramCode);
实现上我们只需要使用上面这两个接口,就可以获取java类的所有函数信息了。JNI环境的获取、各种错误处理都已经在这两个接口实现中封装好了。先上代码,再来依次讲解每个参数的意义和使用方法:
//函数信息结构体    JniMethodInfo minfo;    bool isHave = JniHelper::getStaticmethodInfo(minfo,/*JniMethodInfo的引用*/                                                 "com/omega/MyApp",/*类的路径*/                                                 "getJavaActivity",/*函数名*/                                                 "()Ljava/lang/Object;");/*函数类型简写*/    jobject activityObj;    if (isHave)    {        //CallStaticObjectMethod调用java函数,并把返回值赋值给activityObj        activityObj = minfo.env->CallStaticObjectMethod(minfo.classID,minfo.methodID);    }
OK,很简单。上面的代码,就是使用JNI在C++中调用java类静态函数的典型使用方法。只有两步: 获取java函数的信息,classID、methodID等等 选择jnienv中的接口,进行函数调用 getStaticmethodInfo参数详解 getmethodInfo和getStaticmethodInfo这两个接口的参数一样,意义也相同,详解如下: JniMethodInfo &methodinfo JniMethodInfo对象的引用,函数执行中会把jnIEvn、classID、methodID写入到引用中。 const char *classname 类的路径,把类的完整包名写全,用法如以上代码。 const char *methodname 函数名,函数名写上就行了。 const char *paramCode 函数类型简写,它的格式为:(参数)返回类型。如:无参数,voID返回类型函数,其简写为 ()V java中的类型对应的简写如下:
参数类型 参数简写
boolean Z
byte B
char C
short S
int I
long J
float F
double D
voID V
Object Ljava/lang/Object;
String Ljava/lang/String;
Array [

多参数的函数

如果函数有多个参数,直接把简写并列即可。注意Object与String型参数简写结尾的分号,示例: (IIII)V ==> voID fun(int a,int b,int c,int d) (ILjava/lang/String;I)Ljava/lang/String; ==> string fun(int x,String a,int y) (ILjava/lang/String;[I)J ==> long fun(int n,String s,int[] arr) 通过jnienv进行函数调用 JNIEvn有一系列的CallStatic[返回类型]Method、Call[返回类型]Method接口,需要针对不同的函数返回类型选择调用。[返回类型]以函数返回类型的不同,对应不同的函数名。例如: CallStaticVoIDMethod ——voID; CallVoIDMethod ——voID 其对应关系如下: 函数名 函数返回值类型 VoID voID Object jobject Boolean jboolean Byte jbyte Char jchar Short Jshort Int jint Long jlong float jfloat Double jdouble 参数传递 调用有参数的java函数时,需要把对应的参数传递进去。需要把参数按顺序加入到classID、methodID后面,并且需要做类型转换。例如:
jint jX = 10;jint jY = 10;minfo.env->CallStaticVoIDMethod(minfo.classID,minfo.methodID,jX,jY);
参数类型转换关系如下: C++类型 java类型 jboolean byte jbyte jchar short Jshort jint long jlong jfloat double jdouble Object jobject Class jclass Jstring Object[] jobjectArray boolean[] jbooleanArray byte[] jbyteArray char[] jchararray short[] JshortArray int[] jintArray long[] jlongArray float[] jfloatArray double[] jdoubleArray string类型的转换 实际上我们最常用的参数类型,主要是内建的数据类型、string字符串类型。数据类型可以直接转为j类型,但是string类型需要做如下处理:
Jstring jmsg = minfo.env->NewStringUTF("http://www.baIDu.com");minfo.env->CallStaticVoIDMethod(minfo.classID,jmsg);
非静态函数的调用 非静态函数的调用与静态函数的调用类似,但是需要通过一个静态函数获取java类对象。
JniMethodInfo info;    bool ret = JniHelper::getStaticmethodInfo(info,"org/cocos2dx/cpp/TestJniHelper","getobj","()Ljava/lang/Object;");    //先获得类的对象,然后用这个对象去调用它的非静态函数    jobject jobj;    if(ret)    {        log("call static method");        jobj = info.env->CallStaticObjectMethod(info.classID,info.methodID);    }    //getmethodInfo判断java定义的类非静态函数是否存在,返回bool    bool re = JniHelper::getmethodInfo(info,"func","()V");    if(re)    {        log("call no-static method");        //非静态函数调用的时候,需要的是对象,所以与静态函数调用的第一个参数不同        info.env->CallVoIDMethod(jobj,info.methodID);    }
总结

以上是内存溢出为你收集整理的[cocos2dx]在cocos2dx中通过Jni实现Java与C++的互相调用(一)全部内容,希望文章能够帮你解决[cocos2dx]在cocos2dx中通过Jni实现Java与C++的互相调用(一)所遇到的程序开发问题。

如果觉得内存溢出网站内容还不错,欢迎将内存溢出网站推荐给程序员好友。

欢迎分享,转载请注明来源:内存溢出

原文地址: http://outofmemory.cn/web/1083759.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2022-05-27
下一篇 2022-05-27

发表评论

登录后才能评论

评论列表(0条)

保存