在android studio中包含共享库.so

时间:2016-04-12 08:00:43

标签: android android-studio ffmpeg android-ndk java-native-interface

我在ndk的源文件夹中提取了ffmpeg,然后在那里编译了它,我只跟着这个:http://www.roman10.net/2013/08/18/how-to-build-ffmpeg-with-ndk-r9/并成功生成 android 文件夹,带有 arm / lib < / strong>和 arm / include 文件。

之后我在$ NDK / sources / ffmpeg / android / arm中创建了一个 Android.mk 文件,在我的android项目中创建了一个 Android.mk (src / main / jni文件夹)。

我的 src / main / jni / Android.mk 是这样的:

LOCAL_PATH := $(call my-dir)

include $(CLEAR_VARS)

LOCAL_MODULE    := tutorial01
LOCAL_SRC_FILES := tutorial01.c
LOCAL_LDLIBS := -llog -ljnigraphics -lz
LOCAL_SHARED_LIBRARIES := libavcodec

include $(BUILD_SHARED_LIBRARY)
$(call import-module,ffmpeg-3.0.1/android/arm)

$ NDK / sources / ffmpeg / android / arm / Android.mk 是:

LOCAL_PATH:= $(call my-dir)

include $(CLEAR_VARS)
LOCAL_MODULE:= libavcodec
LOCAL_SRC_FILES:= lib/libavcodec-57.so
LOCAL_EXPORT_C_INCLUDES := $(LOCAL_PATH)/include
include $(PREBUILT_SHARED_LIBRARY)

我像这样更新了我的build.gradle并使用.so文件成功生成了jniLibs,但是我在构建项目时得到 libavcodec / avcodec.h:没有这样的文件或目录

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

apply plugin: 'com.android.model.application'

model {
    android {
        compileSdkVersion = 23
        buildToolsVersion = "23.0.2"

        defaultConfig.with {
            applicationId = "com.example.spartan.hello"
            minSdkVersion.apiLevel = 18
            targetSdkVersion.apiLevel = 23
            versionCode = 1
            versionName = "1.0"
        }
    }
    android.buildTypes {
        release {
            minifyEnabled = false
            proguardFiles.add(file('proguard-android.txt'))
        }
    }

    android.sources {
        main {
            jni {
                source {
                    srcDirs = []
                }

            }

        }

        main {
            jniLibs {
                source {
                    srcDirs = ['src/main/libs']
                }
            }
        }

    }

    android.ndk {
        moduleName = "tutorial01"
        stl = 'gnustl_shared'
    }
}


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
}

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

并将其加载到我的活动中:

static {
        System.loadLibrary("tutorial01");     
        System.loadLibrary("avcodec-57");   
    }

我是NDK新手所以,如果我们手动将头文件粘贴到jni文件夹中,可以吗?

2 个答案:

答案 0 :(得分:1)

我从第三方google示例https://github.com/googlesamples/android-ndk/tree/master/hello-thirdparty中获取了参考资料,其中基本上是针对静态lib(.a),这就是我使用gradle实验插件解决问题的方法:

我的build.gradle文件:

apply plugin: 'com.android.model.application'

model {
    android {
        compileSdkVersion = 23
        buildToolsVersion = "23.0.2"

        defaultConfig.with {
            applicationId = "com.example.spartan.hello"
            minSdkVersion.apiLevel = 18
            targetSdkVersion.apiLevel = 23
            versionCode = 1
            versionName = "1.0"
        }
    }
    android.buildTypes {
        release {
            minifyEnabled = false
            proguardFiles.add(file('proguard-android.txt'))
        }
    }

    repositories {
        libs(PrebuiltLibraries) {
            libavcodec {
                headers.srcDir "/home/spartan/AndroidStudioProjects/Hello/app/src/main/jni/include"
                binaries.withType(SharedLibraryBinary) {
                    sharedLibraryFile = file("/home/spartan/AndroidStudioProjects/Hello/app/src/main/jniLibs/${targetPlatform.getName()}/libavcodec-57.so")
                }
            }                

        }
    }

    android.ndk {
        moduleName = "tutorial01"

        def jniPath = "/home/spartan/AndroidStudioProjects/Hello/app/src/main/jniLibs"
        cppFlags.add("-I${file(jniPath)}".toString())
        file(jniPath).eachDirRecurse { dir ->
            cppFlags.add("-I${file(dir)}".toString())
        }
        ldLibs.addAll(["log", "android","jnigraphics"])
        stl = "gnustl_shared"
    }


    android.sources {
        main {
            jni {
                dependencies {
                    library "libavcodec" linkage "shared"
                }
            }
        }
    }

    android.productFlavors {
        create ("armv7") {
            ndk.abiFilters.add("armeabi-v7a")
        }
    }

    android.sources {
        main {
            jni {
                source {
                    srcDirs = ['src/main/none']
                }

            }

        }
    }

}

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

在我的情况下,我在 jniLibs 中复制 .so ,并在 jni 文件夹中包含 .h 文件夹而且没有必要在android.ndk中添加-l,因为我已经添加了 headers.srcDir (我刚才在这里添加)。

所以我在这里用存储库完全替换了 Android.mk ,而且非常简单。

答案 1 :(得分:0)

如果您自己使用ndk-build,则无需从gradle编译任何内容。

将您的jni源设置为不存在的文件夹或没有文件夹:

android.sources{
   main.jni {
    source {
        srcDirs = [] // or ['src/main/none']
    }
  }
}

并删除此块:

// New code
android.ndk {
    moduleName = "tutorial01"
}