cocos C++调用java

参考:

C++调用Java_u010912122的专栏-CSDN博客_hiworld

原理:

通过JNI获取java虚拟机,再获取当前程序的JNI环境,通过JNI环境获取需要调用的java类信息,再获取需要调用的java类中的函数信息。再通过JNI环境调用,使用类信息、函数信息,调用对应的java函数。

实现:

cocos2dx 封装了一个JniHelper的类,主要通过如下接口实现C++调用java

typedef struct JniMethodInfo_
{
    JNIEnv *    env;           //
    jclass      classID;       //
    jmethodID   methodID;      //
} JniMethodInfo;    

/*
methodinfo:JniMethodInfo的引用,将引用中的env,classID,methodID
className: java类的完全路径
methodName: java类方法
paramCode: 函数类型简写

注意:两个方法的唯一不同处,是前者调用java中的static方法,后者调用普通的方法
*/
static bool getStaticMethodInfo(JniMethodInfo &methodinfo,
                                const char *className,
                                const char *methodName,
                                const char *paramCode);
static bool getMethodInfo(JniMethodInfo &methodinfo,
                            const char *className,
                            const char *methodName,
                            const char *paramCode);

示例:

// 摘录于: cddandroidAndroidJavaEngine.cpp
void AndroidJavaEngine::playBackgroundMusic(const char* filePath, bool loop) {
    std::string fullPath = CocosDenshion::android::getFullPathWithoutAssetsPrefix(filePath);

    cocos2d::JniMethodInfo methodInfo;

    // 判定能否获取java中的playBackgroundMusic方法
    if (!getStaticMethodInfo(methodInfo, "playBackgroundMusic", "(Ljava/lang/String;Z)V")) {
        return;
    }
    // 将C++中的char*类型通过NewStringUTF转换为Java中的jstring类型
    jstring stringArg = methodInfo.env->NewStringUTF(fullPath.c_str());
    // 通过JniEnv调用java中的指定方法,并将参数传进去 
    methodInfo.env->CallStaticVoidMethod(methodInfo.classID, methodInfo.methodID, stringArg, loop);
    // 
    methodInfo.env->DeleteLocalRef(stringArg);
    methodInfo.env->DeleteLocalRef(methodInfo.classID);
}
// 摘录:cocos2dxHelper.java
public static void playBackgroundMusic(final String pPath, final boolean isLoop) {
    Cocos2dxHelper.sCocos2dMusic.playBackgroundMusic(pPath, isLoop);
}

解析

1. getStaticMethodInfo

第1个参数methodInfo 将jniEnv, classId, methodId 写入到引用中

第2,3个参数将className, methodName写入

第4个参数paramCode,其格式为: (参数简写)返回类型简写

参数类型简写参数类型简写
booleanZbyteB
char  CshortS
intIlongJ
floatFdoubleD
voidVobjectLjava/lang/String;
Array[Ljava/lang/String; 

比如:

 2. CallStaticIntMethod

JniEnv在调用指定的java函数时,会根据java的函数返回类型选择不同的方法们。它们都有着类似的特点,比如:
-- CallStaticVoidMethod  调用的是 static void 类型方法
-- CallStaticStringMethod 调用的是 static String类型方法
-- CallIntMethod 表示调用的是 int类型的方法
因此我们可以理解方法有着这样的格式: 
Call[是否静态][返回类型]Method

而且针对于[返回类型]所对应的关系如下:

C++返回类型编写对应java返回类型C++返回类型编写java方法返回类型
VoidvoidShortjshort
ObjectjobjectIntjint
BooleanjbooleanLongjlong
BytejbyteFloatjfloat
CharjcharDoublejdouble
在调用该方法时,我们传入的参数主要有三组:
1. classID
2. methodID
3. C++调用java方法所需要的自定义参数
其中针对于第3组,我们需要将C++转换为java的指定格式,类似于如上字符串的转换:
// 将C++中的char*类型通过NewStringUTF转换为Java中的jstring类型
jstring stringArg = methodInfo.env->NewStringUTF(fullPath.c_str());

而针对于整型类型的转化,我们可以直接强制转换,比如:

// C++ int 转换为java jint
jint jx = (int)x;
jint jy = (int)y;

他们参数类型对应的转换关系如下:

C++类型Java类型C++类型   Java类型     

boolean

jboolean

boolean[]

jbooleanArray 

byte

jbyte

byte[]

jbyteArray

char

jchar

char[]

jcharArray

short

jshort

short[]

jshortArray

int

jint

int[]

jintArray
longjlonglong[]jlongArray

float

jfloat

float[]

jfloatArray

doubule

jdouble

double[]

jdoubleArray

Object

jobject

Oject[]

jobjectArray
ClassjclassStringjstring
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

鹤九日

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值