2013-08-20 53 views
3

我有以下JNI方法目前调用静态Java方法:JNI/Android:从C++调用Java中的非静态方法?

void bindToMarketBillingServiceJNI(const char * inappID) 
    { 
     JniMethodInfo t; 

     if (JniHelper::getStaticMethodInfo(t 
      , "com/mycompany/games/js/TestsDemo" 
      , "bindToMarketBillingService" 
      , "(Ljava/lang/String;)V")) 
     { 
      jstring stringArg1; 

      if (! inappID) 
      { 
       stringArg1 = t.env->NewStringUTF("1"); 
      } 
      else 
      { 
       stringArg1 = t.env->NewStringUTF(inappID); 
      } 

      t.env->CallStaticVoidMethod(t.classID, t.methodID, stringArg1); 

      t.env->DeleteLocalRef(stringArg1); 
      t.env->DeleteLocalRef(t.classID); 

     } 
    } 

这里是我的Java方法:

public static void bindToMarketBillingService(final String mSku) 
    { 

// Some code.... 

} 

,一切工作正常。

但现在,我想让我的Java方法非静态。我想,我只是不得不改变:

  • JniHelper :: getStaticMethodInfo到JniHelper :: getMethodInfo
  • t.env-> CallStaticVoidMethod到t.env-> CallVoidMethod
  • “公共静态无效bindToMarketBillingService(终字符串MSKU)”到‘公共无效bindToMarketBillingService(最后弦乐MSKU)’

但它崩溃了...以下(崩溃日志)

任何人知道如何解决,这是起诉?

谢谢!

08-20 11:54:11.009: A/libc(9379): Fatal signal 11 (SIGSEGV) at 0x0c1f4072 (code=1) 
08-20 11:54:11.109: D/dalvikvm(31592): GC_CONCURRENT freed 735K, 12% free 9743K/10951K, paused 4ms+7ms 
08-20 11:54:11.184: D/dalvikvm(1976): GC_CONCURRENT freed 1720K, 35% free 28243K/43399K, paused 5ms+13ms 
08-20 11:54:11.194: I/ApplicationPolicy(1976): getActualApplicationStateEnabled() : true 
08-20 11:54:11.219: D/PJ_MainApplication(9698): onApplicationCreate -> Now 
08-20 11:54:11.334: D/dalvikvm(31592): GC_CONCURRENT freed 273K, 9% free 9981K/10951K, paused 3ms+3ms 
08-20 11:54:11.409: D/PJ_CrashManager(9698): registerHandler -> Current handler class = com.android.internal.os.RuntimeInit$UncaughtHandler 
08-20 11:54:11.469: I/DEBUG(9331): *** *** *** *** *** *** *** *** *** *** *** *** *** *** *** *** 
08-20 11:54:11.469: I/DEBUG(9331): Build fingerprint: 'samsung/GT-I9100/GT-I9100:4.0.3/IML74K/XWLPG:user/release-keys' 
08-20 11:54:11.469: I/DEBUG(9331): pid: 9379, tid: 9403 >>> com.audioguidia.games.js <<< 
08-20 11:54:11.469: I/DEBUG(9331): signal 11 (SIGSEGV), code 1 (SEGV_MAPERR), fault addr 0c1f4072 
08-20 11:54:11.474: I/DEBUG(9331): r0 00000000 r1 10000000 r2 fffffffc r3 4cbb9ef0 
08-20 11:54:11.474: I/DEBUG(9331): r4 00273f98 r5 4c7f7fc4 r6 51134ad4 r7 415077f8 
08-20 11:54:11.474: I/DEBUG(9331): r8 00000001 r9 0c1f4071 10 4cbb9ee4 fp 51134a58 
08-20 11:54:11.474: I/DEBUG(9331): ip ffffffec sp 51134a58 lr 4096864d pc 40968aac cpsr 00000030 
08-20 11:54:11.474: I/DEBUG(9331): d0 0000000000000000 d1 3f80000000000000 
08-20 11:54:11.474: I/DEBUG(9331): d2 0000000000000000 d3 0000000000000000 
08-20 11:54:11.474: I/DEBUG(9331): d4 0000000000000000 d5 3f80000000000000 
etc.... 
+0

你声明Java类中的新的非静态方法?你需要有一个对象引用来调用非静态方法 –

+0

我简单地将'public static void bindToMarketBillingService(final String mSku)'改为'public void bindToMarketBillingService(final String mSku)'。我需要做其他事吗?谢谢。 –

+0

是作为c.s.说,你需要将'jobject'实例作为第一个参数传递给'CallVoidMethod(** here **,jmethodId ....)' – Tom

回答

6

改变只是env对象上调用是不够的。

首先您的新实例方法应该在Java类中声明,如:public void myMethod(...)(即没有0​​关键字在这里)。

其次,您需要一个java对象引用来调用实例方法,而不仅仅是类。例如,你会做这样的事情(请注意我用的C语法):

jmethodID midCallBack = (*env)->GetMethodID(env, thisClass, "myMethod", "()V"); 
if (NULL == midCallBack) return; 

// Call back the method (which returns void), based on the Method ID 
(*env)->CallVoidMethod(env, thisObj, midCallBack); 

见部分“5.3回调实例方法和静态方法”this link了解更多信息

+0

谢谢。对不起,但我真的是JNI的新手!在我最初的问题的具体代码中,我明白我必须用't.env'替换'env'变量。正确吗?我如何创建'thisClass'和'thisObj'变量?什么是“回拨”? (我不需要回调,我只需要通过JNI从C++中激发我的Java方法)。如果你能告诉我在我最初的问题中如何修改代码,我将非常感激。感谢您的支持。 –

+1

Regis_AG,我认为这个问题需要更多的上下文!你可以做'jclass jThatObjClass = jEnv-> FindClass(“com/package/thatobject”);','const char * ctorSignature =/*你的对象签名* /; jmethodID ctor = jEnv-> GetMethodID(jThatObjClass,“”,ctorSignature);','jobject thisObj = jEnv-> NewObject(jThatObjClass,ctor,jYourObjectCtorArgs ...)',是的,但是这样做然后只是扔掉thisOBj将表明一些相当奇怪的设计。 – Tom

2

如果你想打电话非静态方法,您必须创建该类的对象,然后才调用成员方法。代码如下所示

jmethodID constructor = env->GetMethodID(cls, "<init>", "()V"); //this is for defaul c-tor 
jobject object = env->NewObject(cls, constructor); 
//getting your method id 
env->CallVoidMethod(object, methodId/*,your parameters*/); 
+0

这为我节省了很多挫折!谢谢 :) –

0

我只是做了Java中的静态方法调用不是静态的方法:

static public void startActivity() { 
    try { 
     Activity mother = QtNative.activity(); 
     Intent intent = new Intent(mother, MainActivity.class); 
     mother.startActivity(intent); 
    } catch (Exception e) { 
     Log.e(msgTag, e.toString()); 
     e.printStackTrace(); 
    } 
} 

我从C++这样的呼吁:

void FacebookAndroid::startAndroidFacebook() { 
QAndroidJniObject::callStaticMethod<void>("org.qtproject.example.MainActivity", 
              "startFacebookActivity", 
              "()V"); 
} 
相关问题