[CallObjectMethod调用getSystemService的Android NDK崩溃

问题描述 投票:3回答:3

这是我问的另一个问题的后续:Android -- get MEID from JNI

我正在尝试获取Android手机的ID。我有一些JNI代码和一个简单的测试应用程序来调用JNI代码。这是我的简单测试应用程序中的有效Java代码:

TelephonyManager tm = (TelephonyManager)this.getSystemService(Context.TELEPHONY_SERVICE);
String id = tm.getDeviceId();

字符串id设置为我想要的电话ID值。但是我需要从JNI获得它,仅使用上面的代码并传递ID值是不可接受的解决方案。 (这是为了使某些JNI代码有些防篡改,我不应该相信Java层发送正确的ID值。)

这是我编写的JNI代码,其中删除了错误处理,因此更容易理解。它一直工作到指示的行,然后整个应用程序崩溃。

// "env" and "obj" are passed to a JNI function and are used unmodified in this code
// JNIEnv *env, jobject obj

jclass cls_context = NULL;
jclass cls_tm = NULL;
jobject tm = NULL;
jmethodID mid;
jfieldID fid;
jstring jstr;
jsize len_jstr;


cls_context = (*env)->FindClass(env, "android/content/Context");
fid = (*env)->GetStaticFieldID(env, cls_context, "TELEPHONY_SERVICE",
        "Ljava/lang/String;");
jstr = (*env)->GetStaticObjectField(env, cls_context, fid);

mid = (*env)->GetMethodID(env, cls_context, "getSystemService",
        "(Ljava/lang/String;)Ljava/lang/Object;");

tm = (*env)->CallObjectMethod(env, obj, mid, jstr);  // THIS LINE CRASHES

cls_tm = (*env)->FindClass(env, "android/telephony/TelephonyManager");

mid = (*env)->GetMethodID(env, cls_tm, "getDeviceId",
        "()Ljava/lang/String;");

jstr = (*env)->CallObjectMethod(env, tm, mid);

len_jstr = (*env)->GetStringUTFLength(env, jstr);

(*env)->GetStringUTFRegion(env, jstr, 0, len_jstr, buf_devid);

我认为问题是obj不是通过的正确方法,但是如果是这样,我不知道什么是正确的方法。传递给JNI函数的obj与Java代码中的this是不是一回事?

编辑:好的,我们发现,如果向JNI函数添加类型为jobject的额外参数,并在该参数中显式传递this的副本,然后将其传递给CallObjectMethod()(导致上述代码崩溃的代码),一切正常。我们获得了TelephonyManager实例,我们可以查询电话ID值。

[使用记录宏,我记录了obj指针和传入的this指针。它们是相似的数字(地址彼此靠近),但不相同。因此,我认为obj是Java VM内部的某种对象引用……实际上与this不同。

在JNI函数中,前两个参数是JNIEnv *envjobject obj。第二个是做什么用的?我该怎么办?有什么方法可以使用它来调用getSystemService,还是我必须传递一个额外的参数并传递this

android c android-ndk java-native-interface
3个回答
1
投票

该问题似乎与Java继承有关:在Java中,即使this.getSystemService()实际上不是this的实例,您也可以调用Context并起作用。当您进行JNI调用时,调用只会失败。

所以我们的解决方案是让我们的Android应用程序实际上将.getApplicationContext()方法功能作为其自身类的一部分来添加。依次调用实际的getSystemService()并返回结果。

代码未更改:我们仍在呼叫

mid = (*env)->GetMethodID(env, cls_context, "getSystemService", "(Ljava/lang/String;)Ljava/lang/Object;");

但是现在,当类中有一个调用我们的JNI函数的.getSystemService()方法时,它可以工作。因此,JNI调用中的env参数确实代表this(这是不相同的...我将this的值打印为指针,并打印了env,它们并不相同,但是它们是绝对相关)。


1
投票

如果要在对象上调用超类方法,则应使用CallNonvirtualObjectMethod()How to call an overriden method in JNI


0
投票

尝试一下...。。。“上下文”来自java:)

jclass ctx = env->FindClass("android/content/Context");
jfieldID fid = env->GetStaticFieldID(ctx,"TELEPHONY_SERVICE","Ljava/lang/String;");
jstring str = (jstring) env->GetStaticObjectField(ctx, fid);
jmethodID mid = env->GetMethodID(ctx, "getSystemService","(Ljava/lang/String;)Ljava/lang/Object;");
jobject tm = env->CallObjectMethod(context, mid, str);

jclass ctx_tm = env->FindClass("android/telephony/TelephonyManager");
jmethodID mid_tm = env->GetMethodID(ctx_tm,"getDeviceId","()Ljava/lang/String;");
jstring str_tm = (jstring) env->CallObjectMethod(tm, mid_tm);

strReturn = env->GetStringUTFChars(str_tm, 0);
© www.soinside.com 2019 - 2024. All rights reserved.