所以我一直試圖從C++調用java方法,但沒有任何運氣。這是錯誤,我得到:JNI - 從C++調用Java方法
JNI ERROR(應用錯誤):訪問過時的本地參考0x5cb00019在0xdeadd00d(索引 6大小爲2的表) VM中止 致命信號11(SIGSEGV)(代碼= 1)
以下是我在代碼執行(Java端):
public class Wrapper extends Activity{
private native void initJNIBridge();
static final String TAG = "Wrapper";
protected void onCreate(Bundle savedInstanceState){
super.onCreate(savedInstanceState);
initJNIBridge(); // Calls C++ function.
}
public void upgradeAdFree() {
Log.d(TAG, "Wrapper::upgradeAdFree()");
}
這裏是C++方面:
typedef struct JniMethodInfo_
{
JNIEnv * env;
jclass classID;
jmethodID methodID;
} JniMethodInfo;
extern "C"
{
static jobject javaObj;
// get env and cache it
static JNIEnv* getJNIEnv(void)
{
JavaVM* jvm = cocos2d::JniHelper::getJavaVM();
if (NULL == jvm) {
LOGD("Failed to get JNIEnv. JniHelper::getJavaVM() is NULL");
return NULL;
}
JNIEnv *env = NULL;
// get jni environment
jint ret = jvm->GetEnv((void**)&env, JNI_VERSION_1_4);
switch (ret) {
case JNI_OK :
// Success!
return env;
case JNI_EDETACHED :
// Thread not attached
// TODO : If calling AttachCurrentThread() on a native thread
// must call DetachCurrentThread() in future.
// see: http://developer.android.com/guide/practices/design/jni.html
if (jvm->AttachCurrentThread(&env, NULL) < 0)
{
LOGD("Failed to get the environment using AttachCurrentThread()");
return NULL;
} else {
// Success : Attached and obtained JNIEnv!
return env;
}
case JNI_EVERSION :
// Cannot recover from this error
LOGD("JNI interface version 1.4 not supported");
default :
LOGD("Failed to get the environment using GetEnv()");
return NULL;
}
}
// get class and make it a global reference, release it at endJni().
static jclass getClassID(JNIEnv *pEnv)
{
jclass ret = pEnv->FindClass(CLASS_NAME);
if (! ret)
{
LOGD("Failed to find class of %s", CLASS_NAME);
}
return ret;
}
static bool getMethodInfo(JniMethodInfo &methodinfo, const char *methodName, const char *paramCode)
{
jmethodID methodID = 0;
JNIEnv *pEnv = 0;
bool bRet = false;
do
{
pEnv = getJNIEnv();
if (! pEnv)
{
break;
}
jclass classID = getClassID(pEnv);
methodID = pEnv->GetMethodID(classID, methodName, paramCode);
if (! methodID)
{
LOGD("Failed to find method id of %s", methodName);
break;
}
methodinfo.classID = classID;
methodinfo.env = pEnv;
methodinfo.methodID = methodID;
bRet = true;
} while (0);
return bRet;
}
JNIEXPORT void JNICALL Java_org_test_games_Wrapper_initJNIBridge(JNIEnv *, jobject jobj){
LOGD("Java_org_test_games_Wrapper_initJNIBridge()");
javaObj = jobj;
return;
}
void upgradeAdFreeJNI()
{
LOGD("upgradeAdFreeJNI");
JniMethodInfo methodInfo;
if (! getMethodInfo(methodInfo, "upgradeAdFree", "()V"))
{
LOGD("Cannot find method!");
return;
}
methodInfo.env->CallVoidMethod(javaObj, methodInfo.methodID);
}
}
在這裏的任何建議將不勝感激,因爲我已經被困在這幾天,它不是世界上最簡單的任務來調試。
感謝
感謝mkaes您的回覆!那就是訣竅。 – kub 2013-02-12 16:52:52