【问题标题】:Generating .so files in Android Studio 1.0.2 with NDK使用 NDK 在 Android Studio 1.0.2 中生成 .so 文件
【发布时间】:2015-03-02 16:48:08
【问题描述】:

我一直致力于根据演练here 构建一个非常简单的 NDKSample 应用程序。 我的问题是,我无法让 Android Studio 生成 .so 文件,所以我没有库。

我知道现在已弃用 NDK 支持,今年年初将提供替代方案,但目前似乎没有任何东西阻止我使用此功能。当我构建我的项目时,我收到以下警告(不是错误):

警告 [Project: :app] 当前的 NDK 支持已弃用。 将来会提供替代方案。

我的项目已构建,但当我运行 .apk 时,它会崩溃(如预期的那样),因为它找不到库/.so 文件。根据示例,我们希望在构建项目时生成这些,这是正确的吗?这是错误:

java.lang.UnsatisfiedLinkError: dalvik.system.PathClassLoader[DexPathList[[zip 文件 "/data/app/com.example.ndksample-1/base.apk"],nativeLibraryDirectories=[/vendor/lib, /system/lib]]] 找不到“libMyLib.so”

关于我的环境

Windows 7、Android Studio 1.0.2、ADB 正在运行 Nexus 5 (emulator-5554)

我的代码

按照示例:

主 Activity.java

package com.example.ndksample;

//import android.support.v7.app.ActionBarActivity;
// This line is not needed as we are not targetting older devices
import android.app.Activity; //Import this app package to use onCreate etc.
import android.os.Bundle;
import android.view.Menu;
import android.view.MenuItem;
import android.widget.TextView;


public class MainActivity extends Activity {

    static{
        System.loadLibrary("MyLib");
    }

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);

        TextView tv = (TextView) findViewById(R.id.my_textview);
        tv.setText(getStringFromNative());
    }

    @Override
    public boolean onCreateOptionsMenu(Menu menu) {
        // Inflate the menu; this adds items to the action bar if it is present.
        getMenuInflater().inflate(R.menu.menu_main, menu);
        return true;
    }

    @Override
    public boolean onOptionsItemSelected(MenuItem item) {
        // Handle action bar item clicks here. The action bar will
        // automatically handle clicks on the Home/Up button, so long
        // as you specify a parent activity in AndroidManifest.xml.
        int id = item.getItemId();

        //noinspection SimplifiableIfStatement
        if (id == R.id.action_settings) {
            return true;
        }

        return super.onOptionsItemSelected(item);
    }

    public native String getStringFromNative();
}

activity_main.xml

<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:tools="http://schemas.android.com/tools" android:layout_width="match_parent"
    android:layout_height="match_parent" android:paddingLeft="@dimen/activity_horizontal_margin"
    android:paddingRight="@dimen/activity_horizontal_margin"
    android:paddingTop="@dimen/activity_vertical_margin"
    android:paddingBottom="@dimen/activity_vertical_margin" tools:context=".MainActivity">

    <TextView
        android:id="@+id/my_textview"
        android:text="@string/hello_world" android:layout_width="wrap_content"
        android:layout_height="wrap_content" />

</RelativeLayout>

ma​​in.c

#include "com_example_ndksample_MainActivity.h"
/* Header for class com_example_ndksample_MainActivity */

JNIEXPORT jstring JNICALL Java_com_example_ndksample_app_MainActivity_getStringFromNative
    (JNIEnv * env, jobject obj)
    {
        return (*env)->NewStringUTF(env, "Hello from Kyle");
    }

build.gradle 注意:应用程序

apply plugin: 'com.android.application'

android {
    compileSdkVersion 21
    buildToolsVersion "21.1.2"

    defaultConfig {
        applicationId "com.example.ndksample"
        minSdkVersion 15
        targetSdkVersion 21
        versionCode 1
        versionName "1.0"

        ndk {
            moduleName "MyLib"
        }
    }
    buildTypes {
        release {
            minifyEnabled false
            proguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-rules.pro'
        }
    }
}

dependencies {
    compile fileTree(dir: 'libs', include: ['*.jar'])
    compile 'com.android.support:appcompat-v7:21.0.3'
}

local.properties

## This file is automatically generated by Android Studio.
# Do not modify this file -- YOUR CHANGES WILL BE ERASED!
#
# This file should *NOT* be checked into Version Control Systems,
# as it contains information specific to your local configuration.
#
# Location of the SDK. This is only used by Gradle.
# For customization when using a Version Control System, please read the
# header note.
sdk.dir=D\:\\Programs\\Android\\Android SDK
ndk.dir=D\:\\Programs\\Android\\Android NDK

如果有人能提供帮助,我的问题是:

最终,我如何生成所需的 .so 文件!!??

可能有助于回答主要问题的子问题:

我的目录布局在 app 下有我的 jni 目录(所以 NDKSample/app/jni),这是正确的吗?我被告知 here 不要将 c 文件放在标准 jni 目录中。我对此进行了尝试,在构建项目时,它崩溃了。错误:

*FAILURE: 构建失败并出现异常。

  • 出了什么问题: 任务 ':app:compileDebugNdk' 执行失败。

    com.android.ide.common.internal.LoggedErrorException:无法运行命令: D:\Programs\Android\Android NDK\ndk-build.cmd NDK_PROJECT_PATH=null APP_BUILD_SCRIPT=C:\Users\Kyle\AndroidStudioProjects\NDKSample\app\build\intermediates\ndk\debug\Android.mk APP_PLATFORM=android-21 NDK_OUT =C:\Users\Kyle\AndroidStudioProjects\NDKSample\app\build\intermediates\ndk\debug\obj NDK_LIBS_OUT=C:\Users\Kyle\AndroidStudioProjects\NDKSample\app\build\intermediates\ndk\debug\lib APP_ABI=all 错误代码: 1*

上面来自英特尔的示例没有指示我构建 Android.mk 文件,示例没有,他生成了一个工作应用程序。我已经尝试将一个放在 jni 目录中,但它没有帮助。我应该创建一个,如果是,我应该把它放在哪里

我的目录下面的图片是否正确?

任何帮助将不胜感激。

凯尔

【问题讨论】:

标签: android android-ndk android-studio java-native-interface android-gradle-plugin


【解决方案1】:

Android Studio 中当前对 NDK 的支持很少,因此他们在 1.0 中弃用了它。

使用它时,会自动生成一个 Android.mk 文件,并在您构建项目时编译您的源代码。现在看来编译失败了,但您应该能够从 Gradle 控制台获取初始错误消息。

如果您没有收到任何其他错误消息,我怀疑对 ndk-build 的调用失败,因为您的 NDK 的安装路径中有空格 (Android NDK)

您可以修复当前错误并继续当前设置,或者禁用默认 NDK 集成,创建 Android.mk/Application.mk 文件并调用 ndk-build(.cmd) 自己或来自任务:

import org.apache.tools.ant.taskdefs.condition.Os

android {
    sourceSets.main {
        jniLibs.srcDir 'src/main/libs' //set .so files location to libs
        jni.srcDirs = [] //disable automatic ndk-build call
    }

    // call regular ndk-build(.cmd) script from app directory
    task ndkBuild(type: Exec) {
        if (Os.isFamily(Os.FAMILY_WINDOWS)) {
            commandLine 'ndk-build.cmd', '-C', file('src/main').absolutePath
        } else {
            commandLine 'ndk-build', '-C', file('src/main').absolutePath
        }
    }
    tasks.withType(JavaCompile) {
        compileTask -> compileTask.dependsOn ndkBuild
    }
}

【讨论】:

  • 嗨@phOb,感谢您的提醒。我从路径中删除了空格,然后将 c 文件移动到 src/main/jni 下的默认 jni 文件夹中。这会生成 .so 文件,但是 Gradle 现在没有将它们打包到 .apk 中。我曾尝试编写脚本 gradle 自己来完成,但它会在实际生成它们之前尝试复制 .so 文件并压缩到 .jar 。我认为这应该是一个非常自动化的系统,而编写这么深的脚本表明存在更深层次的问题?
  • com.md.NDKSample.MainActivity 的 .class 文件位于目录 app/build/intermediates/classes/debug 中,因此生成的 .so 文件将转到目录 app/build/ intermediates/ndk/debug/lib - 但是,当我在资源管理器中导航时,我只能看到这些文件; AndroidStudio 仅显示 app/build/intermediates 中的爆炸 aar 目录 - 没有其他子目录,例如资产、类、ndk 等。这是一个问题吗?
  • ph0b,谢谢。我现在已经解决了这个问题。它是从错误的目录生成 java 头文件的组合(应该根据您的示例从 app/src/main 调用它们,而不管 1.0 中的新目录结构如何)并且在我的 ndk 路径中有空格。谢谢。
  • 如果您在 Windows 上并且 Android Studio 安装在您自己的用户目录(默认)中,您的名称中有该空间,您可以在用户目录中创建一个链接,您可以在其中 %20 空间,例如为“First Last”创建一个链接“First%20Last”,然后将该链接添加到 local.properties。 [不小心加错帖了,抱歉]
  • 我想我在这里太快了。看起来它正在工作,但是编译失败,因为 cmd 无法识别链接并且 ndk 使用命令 shell 执行命令:( 所以我想我们需要实际移动 ndk。
猜你喜欢
  • 2015-12-03
  • 2015-04-04
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
相关资源
最近更新 更多