正在準(zhǔn)備中的項(xiàng)目里,有一部分打算直接移殖Linux開發(fā)組在之前就完成的功能,他們是使用C語言開發(fā)。考慮到維護(hù)的問題,準(zhǔn)備讓他們將代碼打包成so文件,再引用到我的項(xiàng)目中。這樣也就相當(dāng)于我去引用一個(gè)第三方庫,并且這個(gè)庫中的代碼格式也不一定是我們JNI開發(fā)時(shí)規(guī)定的命名,因此,需要通過我自己的C文件再去調(diào)用so庫中的方法。
1 生成SO庫
1.1 Native方法
新建項(xiàng)目JNISODemo,在MainActivity中定義Native方法:
public native String getString();
1.2 頭文件生成
.h文件的生成是在命令行cd到main目錄下,再使用javah生成。
這次是想介紹下快捷方式。
File -> Settings -> Tools -> External tools -> +

Program: $JDKPath$\bin\javah.exe
Arguments: -classpath . -jni -o $ModuleFileDir$\src\main\jni\$Prompt$ $FileClass$
Working directory: $ModuleFileDir$\src\main\java
在我們聲明native方法的類上點(diǎn)擊右鍵,javah,輸入命名(我命名為Test.h),之后就會(huì)先自動(dòng)創(chuàng)建一個(gè)jni文件夾,然后生成一個(gè)Test.h文件,copy Test.h,并將命名改為Test.c。

1.3 完成C代碼
在Test.c中簡單完成下我們定義的方法,返回一個(gè)字符串:
#include <jni.h>
/* Header for class com_david_jnisodemo_MainActivity */
#ifndef _Included_com_david_jnisodemo_MainActivity
#define _Included_com_david_jnisodemo_MainActivity
#ifdef __cplusplus
extern "C" {
#endif
/*
* Class: com_david_jnisodemo_MainActivity
* Method: getString
* Signature: ()Ljava/lang/String;
*/
JNIEXPORT jstring JNICALL Java_com_david_jnisodemo_MainActivity_getString
(JNIEnv *env, jobject instance) {
return (*env)->NewStringUTF(env, "This is a test!");
}
#ifdef __cplusplus
}
#endif
#endif
1.4 CMakeList.txt
cmake_minimum_required(VERSION 3.4.1)
add_library( # Sets the name of the library.
Test
# Sets the library as a shared library.
SHARED
# Provides a relative path to your source file(s).
src/main/jni/Test.c )
find_library(log-lib log )
target_link_libraries( # Specifies the target library.
Test
${log-lib} )
1.5 build.gradle
ndk {
ldLibs "log"http://實(shí)現(xiàn)__android_log_print
abiFilters "armeabi-v7a" //平臺(tái)配置,因?yàn)樵贏ndroid上,就只寫了一個(gè)
}
1.6 編譯生成so
編譯完成后就如下圖,產(chǎn)生一個(gè)libTest.so的文件,這就是我們要的。把它當(dāng)做Linux最后打包成的so文件。

2 導(dǎo)入第三方so
新建一個(gè)項(xiàng)目JNIUseSoDemo,項(xiàng)目結(jié)構(gòu)如下。同樣是在MainActivity中定義Native方法,生成UseSo.c。將我們上一步生成的so文件拷貝到j(luò)niLibs下(armeabi-v7a是平臺(tái))。以及上一步中的頭文件也拷貝到j(luò)ni下。

2.1 完成C代碼
我在UseSo中g(shù)etString方法去調(diào)用了so庫中的getString方法。
#include <jni.h>
#include "Test.h" //so庫的頭文件,必須要引用!
/* Header for class com_david_jniusesodemo_MainActivity */
#ifndef _Included_com_david_jniusesodemo_MainActivity
#define _Included_com_david_jniusesodemo_MainActivity
#ifdef __cplusplus
extern "C" {
#endif
/*
* Class: com_david_jniusesodemo_MainActivity
* Method: getString
* Signature: ()Ljava/lang/String;
*/
JNIEXPORT jstring JNICALL Java_com_david_jniusesodemo_MainActivity_getString
(JNIEnv *env, jobject instance) {
return Java_com_david_jnisodemo_MainActivity_getString(env, instance);
}
#ifdef __cplusplus
}
#endif
#endif
當(dāng)然還沒有完成。
2.2 CMakeList.txt
在CMake中將LibTest.so導(dǎo)入工程
cmake_minimum_required(VERSION 3.4.1)
add_library( # Sets the name of the library.
UseSo
# Sets the library as a shared library.
SHARED
# Provides a relative path to your source file(s).
src/main/jni/UseSo.c )
#導(dǎo)入第三方so包,并聲明為 IMPORTED 屬性,指明只是想把 so 導(dǎo)入到項(xiàng)目中
add_library( Test
SHARED
IMPORTED )
#指明 so 庫的路徑,CMAKE_SOURCE_DIR 表示 CMakeLists.txt 的路徑
set_target_properties(
Test
PROPERTIES IMPORTED_LOCATION
${CMAKE_SOURCE_DIR}/src/main/jniLibs/armeabi-v7a/libTest.so )
#指明頭文件路徑,不然會(huì)提示找不到 so 的方法
include_directories(src/main/jni/)
find_library(log-lib
log )
target_link_libraries( # Specifies the target library.
UseSo
Test
${log-lib} )
2.3 build.gradle
ndk {
abiFilters 'armeabi-v7a'
}
2.4 最終調(diào)用
public class MainActivity extends Activity {
static {
System.loadLibrary("UseSo"); //加載SO庫
}
@Override
protected void onCreate(@Nullable Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
Log.e("TEST",getString());
}
public native String getString(); //它會(huì)調(diào)用Java_com_david_jniusesodemo_MainActivity_getString方法,然后該方法又回去調(diào)用so庫中的Java_com_david_jnisodemo_MainActivity_getString方法,得到返回字符串。
}

驗(yàn)證沒有問題,導(dǎo)入第三方so庫完成。