背景:
最近遇到了一個問題芬失,需要調(diào)用第三方的so方法故河,但這個so內(nèi)的方法不是標(biāo)準(zhǔn)的jni方法巍虫。這就需要我們自己寫jni然后彭则,鏈接到已經(jīng)存在的so庫,通過jni調(diào)用so庫中的方法占遥,就可以實現(xiàn)我們的需求俯抖。
一、準(zhǔn)備工作
1.工作環(huán)境瓦胎,當(dāng)前是mac10.13+AndroidStudio3.1(開始使用的3.4版本芬萍,但遇到了些問題)
一路next之后,完成項目的創(chuàng)建搔啊。
二柬祠、編寫依賴的c++文件(非標(biāo)準(zhǔn)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)
#設(shè)置生成的so動態(tài)庫最后輸出的路徑
set(CMAKE_LIBRARY_OUTPUT_DIRECTORY ${CMAKE_SOURCE_DIR}/src/main/jniLibs/${ANDROID_ABI})
#生成so
add_library( # 設(shè)置生成庫的名字
add
# 生成動態(tài)庫
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文件信息
接下來我們的程序想要調(diào)用so里面的方法示罗。
有時候我們并不知道第三方so中的方法名是什么惩猫,有什么參數(shù),可以在命令行中使用strings或者nm指令查看蚜点。我們進到so的目錄下執(zhí)行strings libadd.so轧房。
可以看到,
so中的類庫依賴關(guān)系和類名方法名都可以看到绍绘,這個就是我們接下來要依賴的送文件奶镶。
五、編寫jni的文件
我們想要訪問so的方法陪拘,就需要通過android studio厂镇,生成native方法:
下面編寫jni,這里直接調(diào)用了so中的Add類的add方法
#include <jni.h>
#include <string>
#include <test.h>//導(dǎo)入需要的.h文件,這個是必須的左刽,如果依賴的第三方庫沒有.h捺信,需要自己編寫
extern "C"
JNIEXPORT jint JNICALL
Java_com_example_jnitest_jnidemo_MainActivity_nativeAdd(JNIEnv *env, jobject instance, jint a,
jint b) {
//生成add對象并調(diào)用方法
Add addObj;
int result = addObj.add(a,b);
return result;
}
五、編寫jni的cmake文件
#指定cmake最小版本
cmake_minimum_required(VERSION 3.4.1)
#設(shè)置生成的so動態(tài)庫最后輸出的路徑
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( # 設(shè)置生成庫的名字
native-lib
# 生成動態(tài)庫
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文件掌挚。
五雨席、調(diào)用生成的so文件
在第三部生成jni方法的時候已經(jīng)通過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);
}
然后運行:
可以看到運行的結(jié)果為10吠式,結(jié)果正確陡厘。
(運行的時候,如果提示
就去build.gradle中設(shè)置
android {
...
packagingOptions {
pickFirst 'lib/xxx/libnative-lib.so'
}
}
即可奇徒。
)
通過以上的步驟雏亚,我們實現(xiàn)了簡單的調(diào)用第三方so庫和其中的方法缨硝,復(fù)雜的原理基本一致摩钙。