我开始使用 NDK 在 Android 上工作,我想检查设备在我的 c 代码上运行的 Android API 级别。我怎样才能做到这一点?
起初我以为我可以使用__ANDROID_API__
/android/api-level.h 下的定义,但这是一个错误的假设。
**注意:我不是在问如何通过 java 检查 API 级别。
我开始使用 NDK 在 Android 上工作,我想检查设备在我的 c 代码上运行的 Android API 级别。我怎样才能做到这一点?
起初我以为我可以使用__ANDROID_API__
/android/api-level.h 下的定义,但这是一个错误的假设。
**注意:我不是在问如何通过 java 检查 API 级别。
我一直在处理一些 JNI 代码,并希望按照Jona的描述查询正在运行的 OS 构建版本。我想尽早做到这一点(即在JNI_OnLoad中),所以宁愿不按照FUBUs的描述从 Java 提交它。自 API 级别 4 以来,此信息已作为android.os.Build.VERSION中的 int 字段SDK_INT提供,这是我在此代码段中查找的内容:
static const char* TAG = "testjnjni";
static bool _disableHttpKeepAliveOnBuggyPlatforms(JNIEnv *env)
{
// Based on article here:
// http://android-developers.blogspot.co.uk/2011/09/androids-http-clients.html
// Which references the issue documented here:
// http://code.google.com/p/android/issues/detail?id=2939
// We need to set "http.keepAlive" to "false" if running an OS version earlier than Froyo (API Level 8)
if ((*env)->ExceptionCheck(env))
return false; // already got an exception pending
bool success = true;
// VERSION is a nested class within android.os.Build (hence "$" rather than "/")
jclass versionClass = (*env)->FindClass(env, "android/os/Build$VERSION");
if (NULL == versionClass)
success = false;
jfieldID sdkIntFieldID = NULL;
if (success)
success = (NULL != (sdkIntFieldID = (*env)->GetStaticFieldID(env, versionClass, "SDK_INT", "I")));
jint sdkInt = 0;
if (success)
{
sdkInt = (*env)->GetStaticIntField(env, versionClass, sdkIntFieldID);
__android_log_print(ANDROID_LOG_VERBOSE, TAG, "sdkInt = %d", sdkInt);
}
if (success && sdkInt < 8)
{
jclass systemClass = (*env)->FindClass(env, "java/lang/System");
if (NULL == systemClass)
success = false;
jmethodID setPropertyMethodID = NULL;
if (success)
success = (NULL != (setPropertyMethodID = (*env)->GetStaticMethodID(env, systemClass, "setProperty", "(Ljava/lang/String;Ljava/lang/String;)Ljava/lang/String;")));
jstring propString = NULL;
if (success)
success = (NULL != (propString = (*env)->NewStringUTF(env, "http.keepAlive")));
jstring valueString = NULL;
if (success)
success = (NULL != (valueString = (*env)->NewStringUTF(env, "false")));
jobject oldValueString = NULL;
if (success)
{
__android_log_print(ANDROID_LOG_VERBOSE, TAG, "Disabling http.keepAlive");
oldValueString = (*env)->CallStaticObjectMethod(env, systemClass, setPropertyMethodID, propString, valueString);
}
// cleanup
(*env)->DeleteLocalRef(env, propString);
(*env)->DeleteLocalRef(env, valueString);
(*env)->DeleteLocalRef(env, oldValueString);
(*env)->DeleteLocalRef(env, systemClass);
}
// cleanup
(*env)->DeleteLocalRef(env, versionClass);
return success;
}
我编写此代码所需的所有信息都清楚地记录在由 Sheng Liang 撰写的题为“Java Native Interface: Programmer's Guide and Specification”的 PDF 中,以前可以从 Oracle 的站点这里获得,但也可以作为一本书购买(例如这里)。JNI 是一项非常强大的技术,我强烈建议任何想要掌握它的开发人员阅读 PDF 以及 Android 开发人员的JNI 提示。
哦,最后,不能强调理解本地和全球参考的重要性。Android 的开发者博客在这里有一篇很好的文章,涵盖了 ICS 中的变化(没有什么偏离 JNI 规范,但需要重申的要点!)。
存在完全原生的解决方案:
Android L
你可以使用__system_property_get("ro.build.version.sdk")
fromsys/system_properties.h
Android L
和更新您可以使用popen("getprop ro.build.version.sdk")
使用编译时工具链常量检查#if __ANDROID_API__ < 21
在这两个实现之间切换。
您可以一次性将您在 JAVA 中获得的 API 级别传递给 C 代码,并将其存储在全局变量中。对我来说,这是更简单的方法。