Warning: file_get_contents(/data/phpspider/zhask/data//catemap/4/c/59.json): failed to open stream: No such file or directory in /data/phpspider/zhask/libs/function.php on line 167

Warning: Invalid argument supplied for foreach() in /data/phpspider/zhask/libs/tag.function.php on line 1116

Notice: Undefined index: in /data/phpspider/zhask/libs/function.php on line 180

Warning: array_chunk() expects parameter 1 to be array, null given in /data/phpspider/zhask/libs/function.php on line 181
调用getSystemService的CallObjectMethod中的Android NDK崩溃_Android_C_Android Ndk_Java Native Interface - Fatal编程技术网

调用getSystemService的CallObjectMethod中的Android NDK崩溃

调用getSystemService的CallObjectMethod中的Android NDK崩溃,android,c,android-ndk,java-native-interface,Android,C,Android Ndk,Java Native Interface,这是我问的另一个问题的后续: 我正在尝试用安卓系统获取手机的ID。我有一些JNI代码和一个简单的测试应用程序来调用JNI代码。下面是我的简单测试应用程序中的Java代码: TelephonyManager tm = (TelephonyManager)this.getSystemService(Context.TELEPHONY_SERVICE); String id = tm.getDeviceId(); 字符串id设置为我想要的电话id值。但是我需要从JNI获得它,仅仅使用上面的代码并传入

这是我问的另一个问题的后续:

我正在尝试用安卓系统获取手机的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虚拟机内部的某种对象引用。。。它实际上与
这个
不同


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

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

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

代码没有改变:我们还在打电话

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

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

如果要在对象上调用超类方法,您应该使用
CallNonvirtualObjectMethod()
试试这个。。。它是有效的。。。“上下文”来自java:)


你调试了吗?首先将所有变量值打印到LogCat(使用
\u android\u log\u print()
),查看所有类和方法ID是否都有效。很可能错误在实际方法调用之前。据我所知,如果返回值不是
NULL
,则返回值有效;并且没有一个返回值是
NULL
。我的代码在每个步骤后都会进行日志记录,并以
%p
格式打印每个指针;这些数字似乎可信。(当我删除了在这里发布的错误检查代码时,我也删除了日志代码。)问题是您应该使用
obj
类来查找方法,因为这是您将在
CallObjectMethod()
中使用的,而不是您正在使用的硬连接类,我有同样的问题,我在应用程序及其工作中使用此代码,但当我在库模块中使用它时,我得到一个异常。请给我一个简单的android中的for.getApplicationContext()来传递此异常。只有在子类中有重写时才需要。
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);