这是我提出的另一个问题的后续行动: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 *env
和jobject obj
。第二个是什么?我该怎么办?有什么方法可以用它来调用getSystemService
,还是我必须通过额外的论证并传入this
?
答案 0 :(得分:1)
问题似乎与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函数时。因此,JNI调用中的env
参数确实代表this
(它不相同......我将this
的值打印为指针,并打印env
,并且他们不一样,但他们肯定是相关的。)
答案 1 :(得分:1)
如果要在对象上调用超类方法,则应使用CallNonvirtualObjectMethod()
How to call an overriden method in JNI
答案 2 :(得分: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);