hello 大家好贮聂,上一篇文章中提到了如何在mac下編譯適配于Android的ffmpeg so文件庫,這篇文章給大家介紹下如何運用編譯好的類庫進行jni調(diào)用歼冰。
在上一篇文章中我們在文件夾中得到一個include文件夾和一個包含了so文件的lib文件夾耻警,在include文件夾內(nèi)分別有l(wèi)ibavcodec,libavfilter,libavformat,libavutil,libswresample,libswscale文件夾,下面是具有全功能的ffmpeg類庫各個包的功能(對號入座):
libavcodec編碼/解碼庫
libavfilter基于幀的編輯庫圖(加特效)
libavformatI/O多路復用/解復用庫
libavdevice復用/解復用庫專用設備(讀設備)
libavutil常見的實用工具庫
libswresample音頻重采樣腮恩、格式轉換和混合
libpostproc后處理庫
libswscale顏色轉換和縮放的圖書館(圖像拉伸温兼,像素格式轉換)
基于以上的介紹募判,我們現(xiàn)在進行jni調(diào)用ffmpeg庫
1:在Android新建一個工程咒唆,在src/main下新建一個文件夾释液,作者這里命名為jni:
2:將我們編譯好的lib下的so文件復制粘貼到jni文件夾下同時再把編譯好的include文件夾直接拷貝到jni文件夾误债,在jni下新建Android.mk以及Application.mk(Makefile格式的文件)以及隨便起個名字的c文件(這里用的是ffmpeg.c)此時,我們看到的目錄是這樣的
3:我們來到MainActivity在Activityxml文件寫一個按鈕和TextView控件以便于測試李命, 在Activity中加載類庫 類庫的加載方法為System.loadLibrary(...Java代碼中不包含前面的“l(fā)ib”和后面的“.so”) 并聲明一個stringFromJNI()方法躺盛。(聲明c方法前面需要加上native)
此時的MainActivity的代碼是這樣的
public class MainActivity extends AppCompatActivity {
private TextView tv;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
tv= (Button) findViewById(R.id.tv);
}
public native String stringFromJNI();
static {
System.loadLibrary("avcodec-57");
System.loadLibrary("avfilter-6");
System.loadLibrary("avformat-57");
System.loadLibrary("avutil-55");
System.loadLibrary("swresample-2");
System.loadLibrary("swscale-4");
System.loadLibrary("ffmpeg");
}
public void showFormat(View view){
tv.setText(stringFromJNI());
}
}
現(xiàn)在我們看到stringFromJNI()是紅色的槽惫,我們需要將MainActivity運用javah生成一個C\C++頭文件
此時運用Androidstudio下的terminal命令行,運用
javah MainActivity的全路徑
得到一個MainActivity的頭文件
如下圖所示:
將生成的MainActivity的C\C++頭文件移動到jni目錄下
4:打開Android.mk添加以下代碼
LOCAL_PATH := $(call my-dir)
# FFmpeg library
include $(CLEAR_VARS)
LOCAL_MODULE := avcodec
LOCAL_SRC_FILES := libavcodec-57.so
include $(PREBUILT_SHARED_LIBRARY)
include $(CLEAR_VARS)
LOCAL_MODULE := avfilter
LOCAL_SRC_FILES := libavfilter-6.so
include $(PREBUILT_SHARED_LIBRARY)
include $(CLEAR_VARS)
LOCAL_MODULE := avformat
LOCAL_SRC_FILES := libavformat-57.so
include $(PREBUILT_SHARED_LIBRARY)
include $(CLEAR_VARS)
LOCAL_MODULE := avutil
LOCAL_SRC_FILES := libavutil-55.so
include $(PREBUILT_SHARED_LIBRARY)
include $(CLEAR_VARS)
LOCAL_MODULE := swresample
LOCAL_SRC_FILES := libswresample-2.so
include $(PREBUILT_SHARED_LIBRARY)
include $(CLEAR_VARS)
LOCAL_MODULE := swscale
LOCAL_SRC_FILES := libswscale-4.so
include $(PREBUILT_SHARED_LIBRARY)
# Program
include $(CLEAR_VARS)
LOCAL_MODULE := ffmpeg
LOCAL_SRC_FILES := ffmpeg.c
LOCAL_C_INCLUDES += $(LOCAL_PATH)/include
LOCAL_LDLIBS := -llog -lz
LOCAL_SHARED_LIBRARIES := avcodec avfilter avformat avutil swresample swscale
include $(BUILD_SHARED_LIBRARY)
其中
LOCAL_SRC_FILES := libavcodec-57.so//自己打出來的so包的名字
LOCAL_MODULE := ffmpeg
LOCAL_SRC_FILES := ffmpeg.c//改成你創(chuàng)建的c文件的名字
5:Application.mk添加以下代碼:(可以不添加 在build.gradle添加了過濾)
APP_ABI := armeabi-v7a
APP_MODULES := libffmpeg
6:利用Terminal進入到jni命令 進行build 輸入如下命令
ndk-build
7:編寫ffmpeg.cpp代碼
此時ffmpeg需要繼承重寫javah生成的頭文件的方法:
筆者生成的頭文件的方法為
JNIEXPORT jstring JNICALL Java_hi_testffmpeg_MainActivity_stringFromJNI
所以ffmpeg.cpp中代碼如下
JNIEXPORT jstring JNICALL Java_hi_testffmpeg_MainActivity_stringFromJNI(JNIEnv *env, jobject obj)
{
char info[10000] = {0};
av_register_all();
sprintf(info, "%s\n", avcodec_configuration());
//LOGE("%s", info);
return (*env)->NewStringUTF(env, info);
}
8:在App下的buidl.gradle下添加abi過濾 選擇輸出支持手機的abi的cpu架構 (一般的手機都是Arm架構)
菜雞可以參考這篇文章(http://www.cnblogs.com/Bugtags2015/p/5578541.html)
代碼如下:
defaultConfig {
applicationId "hi.test"
minSdkVersion 15
targetSdkVersion 26
versionCode 1
versionName "1.0"
testInstrumentationRunner "android.support.test.runner.AndroidJUnitRunner"
// abi過濾
ndk{
abiFilters "armeabi","armeabi-v7a"
}
}
externalNativeBuild{
ndkBuild{
path "src/main/jni/Android.mk"
}
}
同步代碼项贺,
這是可以看到文件的結構是這樣的:
這時我們就可以完美的運行啦