5

我开始使用 NDK 在 Android 上工作,我想检查设备在我的 c 代码上运行的 Android API 级别。我怎样才能做到这一点?

起初我以为我可以使用__ANDROID_API__/android/api-level.h 下的定义,但这是一个错误的假设。

**注意:我不是在问如何通过 java 检查 API 级别。

4

3 回答 3

9

我一直在处理一些 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 规范,但需要重申的要点!)。

于 2012-05-09T07:47:34.143 回答
4

存在完全原生的解决方案:

  1. 之前Android L你可以使用__system_property_get("ro.build.version.sdk")fromsys/system_properties.h
  2. 对于Android L和更新您可以使用popen("getprop ro.build.version.sdk")

使用编译时工具链常量检查#if __ANDROID_API__ < 21在这两个实现之间切换。

于 2015-10-21T20:46:30.917 回答
1

您可以一次性将您在 JAVA 中获得的 API 级别传递给 C 代码,并将其存储在全局变量中。对我来说,这是更简单的方法。

于 2012-04-18T16:41:04.900 回答