背景:
最近遇到了一個問題,需要調用第三方的so方法,但這個so內的方法不是標準的jni方法。這就需要我們自己寫jni然后,鏈接到已經存在的so庫,通過jni調用so庫中的方法,就可以實現我們的需求。
一、準備工作
1.工作環境,當前是mac10.13+AndroidStudio3.1(開始使用的3.4版本,但遇到了些問題)
image
一路next之后,完成項目的創建。
image
二、編寫依賴的c++文件(非標準jni方式)
test.h文件
#ifndef _TEST_JNI_ADD_H_
#define _TEST_JNI_ADD_H_
class Add {
public:
Add();
~Add();
int add(int x, int y);
};
#endif
test.cpp文件
#include "test.h"
Add::Add() {
}
Add::~Add() {
}
int Add::add(int x, int y) {
return x + y;
}
三、編寫依賴的cmake文件
#指定cmake最小版本
cmake_minimum_required(VERSION 3.4.1)
#設置生成的so動態庫最后輸出的路徑
set(CMAKE_LIBRARY_OUTPUT_DIRECTORY ${CMAKE_SOURCE_DIR}/src/main/jniLibs/${ANDROID_ABI})
#生成so
add_library( # 設置生成庫的名字
add
# 生成動態庫
SHARED
# 指定源碼文件,這里指定test.cpp文件
src/main/cpp/test.cpp )
#依賴的頭文件
INCLUDE_DIRECTORIES(${CMAKE_CURRENT_SOURCE_DIR}/src/main/cpp)
find_library( # log庫的別名
log-lib
#log庫
log )
#鏈接代碼到指定的庫
target_link_libraries( # Specifies the target library.
add
# Links the target library to the log library
# included in the NDK.
${log-lib} )
點擊build->Make Project,可以看到在jniLibs文件夾下生成了so文件
生成so文件
四、查看依賴so文件信息
接下來我們的程序想要調用so里面的方法。
有時候我們并不知道第三方so中的方法名是什么,有什么參數,可以在命令行中使用strings或者nm指令查看。我們進到so的目錄下執行strings libadd.so。
執行strings指令
可以看到,
執行結果
so中的類庫依賴關系和類名方法名都可以看到,這個就是我們接下來要依賴的送文件。
五、編寫jni的文件
我們想要訪問so的方法,就需要通過android studio,生成native方法:
生成native方法
下面編寫jni,這里直接調用了so中的Add類的add方法
#include <jni.h>
#include <string>
#include <test.h>//導入需要的.h文件,這個是必須的,如果依賴的第三方庫沒有.h,需要自己編寫
extern "C"
JNIEXPORT jint JNICALL
Java_com_example_jnitest_jnidemo_MainActivity_nativeAdd(JNIEnv *env, jobject instance, jint a,
jint b) {
//生成add對象并調用方法
Add addObj;
int result = addObj.add(a,b);
return result;
}
五、編寫jni的cmake文件
#指定cmake最小版本
cmake_minimum_required(VERSION 3.4.1)
#設置生成的so動態庫最后輸出的路徑
set(CMAKE_LIBRARY_OUTPUT_DIRECTORY ${CMAKE_SOURCE_DIR}/src/main/jniLibs/${ANDROID_ABI})
#指定文件夾下所有的c和c++文件
#file(GLOB_RECURSE cpp_srcs "src/main/cpp/*.c" "src/main/cpp/*.cpp")
#message會生成日志,放在.extrrnalNativeBuild/cmake/debug/xxxx/cmake_build_output.txt文件中
#message(STATUS "src_files ${cpp_srcs}")
#生成so
add_library( # 設置生成庫的名字
native-lib
# 生成動態庫
SHARED
# 指定源碼文件,這里指定test.cpp文件
src/main/cpp/native-lib.cpp )
#依賴的頭文件
INCLUDE_DIRECTORIES(${CMAKE_CURRENT_SOURCE_DIR}/src/main/cpp)
#依賴的add庫
add_library(add SHARED IMPORTED)
set_target_properties(add
PROPERTIES IMPORTED_LOCATION
${CMAKE_CURRENT_SOURCE_DIR}/src/main/jniLibs/${ANDROID_ABI}/libadd.so)
find_library( # log庫的別名
log-lib
#log庫
log )
#鏈接代碼到指定的庫
target_link_libraries( # Specifies the target library.
native-lib
#add庫需要鏈接
add
# Links the target library to the log library
# included in the NDK.
${log-lib} )
完成之后,Build->Make Project,可以看到在jniLibs目錄下生成了新的libnative-lib.so文件。
五、調用生成的so文件
在第三部生成jni方法的時候已經通過Sydstem.loadLibrary加載so文件,
package com.example.jnitest.jnidemo;
import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;
import android.widget.TextView;
public class MainActivity extends AppCompatActivity {
// Used to load the 'native-lib' library on application startup.
static {
System.loadLibrary("native-lib");//加載so
}
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
// Example of a call to a native method
TextView tv = (TextView) findViewById(R.id.sample_text);
tv.setText(String.valueOf(nativeAdd(4,6)));
}
/**
* A native method that is implemented by the 'native-lib' native library,
* which is packaged with this application.
*/
public native int nativeAdd(int a, int b);
}
然后運行:
運行結果
可以看到運行的結果為10,結果正確。
(運行的時候,如果提示
運行提示
就去build.gradle中設置
android {
...
packagingOptions {
pickFirst 'lib/xxx/libnative-lib.so'
}
}
即可。
)
通過以上的步驟,我們實現了簡單的調用第三方so庫和其中的方法,復雜的原理基本一致。