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不是正确的事情,但如果是这样,我不知道什么是正确的事情。obj传递给 JNI 函数的内容与thisJava 代码中的内容不一样吗?

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

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

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

4

3 回答 3

2

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

所以我们的解决方案是让我们的 Android 应用程序.getApplicationContext()实际上添加一个方法函数作为它自己的类的一部分。这反过来调用实际getSystemService()并返回结果。

代码没有改变:我们仍在调用

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

但是现在它可以工作了,当.getSystemService()类中有一个方法正在调用我们的 JNI 函数时。因此,envJNI 调用中的参数确实表示this(它不完全相同......我将值打印this为指针,然后打印env,它们不一样,但它们肯定是相关的)。

于 2012-09-17T20:53:31.387 回答
0

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

于 2015-02-03T09:58:43.067 回答
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);
于 2020-01-06T05:35:44.883 回答