參考:java
https://blog.csdn.net/u010912122/article/details/19341565 android
原理:函數
經過JNI獲取java虛擬機,再獲取當前程序的JNI環境,經過JNI環境獲取須要調用的java類信息,再獲取須要調用的java類中的函數信息。再經過JNI環境調用,使用類信息、函數信息,調用對應的java函數。oop
實現:url
cocos2dx 封裝了一個JniHelper的類,主要經過以下接口實現C++調用javaspa
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);
示例:.net
// 摘錄於: 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); }
解析3d
1. getStaticMethodInfocode
第1個參數methodInfo 將jniEnv, classId, methodId 寫入到引用中blog
第2,3個參數將className, methodName寫入
第4個參數paramCode,其格式爲: (參數簡寫)返回類型簡寫
參數類型 | 簡寫 | 參數類型 | 簡寫 |
boolean | Z | byte | B |
char | C | short | S |
int | I | long | J |
float | F | double | D |
void | V | object | Ljava/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方法返回類型 |
Void | void | Short | jshort |
Object | jobject | Int | jint |
Boolean | jboolean | Long | jlong |
Byte | jbyte | Float | jfloat |
Char | jchar | Double | jdouble |
在調用該方法時,咱們傳入的參數主要有三組:
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 |
long | jlong | long[] | jlongArray |
float |
jfloat |
float[] |
jfloatArray |
doubule |
jdouble |
double[] |
jdoubleArray |
Object |
jobject |
Oject[] |
jobjectArray |
Class | jclass | String | jstring |