最近在做jni,苦于沒(méi)高人指點(diǎn)啊乡小,全靠自己摸索阔加,掌握了一些之后,想著分享出來(lái)满钟,盡量寫的細(xì)一點(diǎn)胜榔,讓每個(gè)初學(xué)者都能看懂,如果能幫到一些人少走點(diǎn)彎路就最好了湃番。
要想成功加載第三方so夭织,其中心思想是:
你得有一個(gè).cpp的文件,通過(guò)這個(gè)文件在CMakeLists.txt的作用下就可以生成對(duì)應(yīng)的so吠撮,這個(gè)so在build目錄下的尊惰,可以理解為隱藏起來(lái)的,你不必管它但是你得有它泥兰,有了它以后你才能夠通過(guò).cpp去調(diào)用.h弄屡,再通過(guò).h文件去調(diào)用對(duì)應(yīng)的so文件,從而完成jni功能
主要分一下幾個(gè)步驟:
1:配置build.gradle(app)
2:新建cpp目錄鞋诗,cpp目錄下有include文件夾膀捷,include文件夾下放.h文件,cpp目錄下還需要.cpp文件
3:在app目錄下新建CMakeLists.txt削彬,這類似一個(gè)清單文件
4:在main目錄下新建jniLibs全庸,jniLibs目錄內(nèi)是arm64-v8a目錄,arm64-v8a目錄內(nèi)是so文件
5:用JniUtil去調(diào)用jni
那么下面就分別講一下它們?cè)趺磁渲玫?/p>
1:配置build.gradle(app)
我把不重要的省略了融痛,只留下關(guān)鍵的配置糕篇,這里配置比較多,別配錯(cuò)了
android {
defaultConfig {
//設(shè)置NDK版本
ndkVersion "25.1.8937393"
// gradle 執(zhí)行的任務(wù)名字
externalNativeBuild {
cmake {
cppFlags ""
abiFilters "arm64-v8a"
}
}
ndk {
// 設(shè)置支持的SO庫(kù)架構(gòu)酌心,第三方給的so庫(kù)哪幾種架構(gòu)拌消,就配置這幾種架構(gòu)
abiFilters 'arm64-v8a'
}
}
//配置CMakeLists.txt為jni編譯列表
externalNativeBuild {
cmake {
path 'CMakeLists.txt'
version "3.10.2"
}
}
sourceSets {
main {
jniLibs.srcDirs = ['libs']
}
}
2:新建cpp目錄
大家可以看到cpp目錄下有include文件夾,include文件夾下放.h文件,cpp目錄下還需要.cpp文件
2.1何為.h文件墩崩?
.h文件是與so文件的橋梁氓英,必不可少,c開發(fā)會(huì)一并給我們
2.2何為native-lib.cpp鹦筹?
當(dāng)然名稱可以隨便起铝阐,我這里叫native-lib.cpp,它是java跟.h溝通的橋梁铐拐,剛剛不是說(shuō).h文件是與so文件的橋梁嘛徘键,都是一步一步連接起來(lái)的,也沒(méi)那么神秘遍蟋,下面看它怎么寫:
#include <jni.h>
#include <string>
#include "include/WindowServiceSomeipServer.h"
extern "C"
jstring Java_com_example_aiosceneengine_util_JniUtil_test(JNIEnv* env, jclass thiz) {
return env->NewStringUTF("不支持(請(qǐng)關(guān)注官方群或本軟件中軟件更新鏈接)");
}
extern "C"
jint Java_com_example_aiosceneengine_util_JniUtil_windowStartSomeipServer(JNIEnv *env,jclass clazz) {
retval_en init = WindowService_StartSomeipServer();
return 1;
}
include <jni.h>:代表我要引用.h文件了
include <string>:代表我想引用string類型
include "include/WindowServiceSomeipServer.h":代表我要引用WindowServiceSomeipServer.h
Java_com_example_aiosceneengine_util_JniUtil_test:這個(gè)方法是不依賴任何so庫(kù)的吹害,因?yàn)槲乙矝](méi)調(diào)用任何的so庫(kù),你可以理解為引用我自己生成的so庫(kù)
Java_com_example_aiosceneengine:代表調(diào)用native-lib.cpp的類的包名
JniUtil:代表調(diào)用native-lib.cpp的類的類名
test:代表調(diào)用native-lib.cpp的類的方法名
JNIEnv* env, jclass thiz:這倆參數(shù)是默認(rèn)的虚青,其實(shí)在調(diào)用的時(shí)候無(wú)需參數(shù)
Java_com_example_aiosceneengine_util_JniUtil_windowStartSomeipServer:跟上面同理它呀,只是這里調(diào)用了.h的方法
WindowService_StartSomeipServer():調(diào)用了WindowServiceSomeipServer.h的方法,直接就可以調(diào)用棒厘,只需要在上面include 一下即可
3:在app目錄下新建CMakeLists.txt
CMakeLists.txt在app目錄下:
CMakeLists.txt代碼如下:
# For more information about using CMake with Android Studio, read the
# documentation: https://d.android.com/studio/projects/add-native-code.html
# Sets the minimum version of CMake required to build the native library.
cmake_minimum_required(VERSION 3.10.2)
#設(shè)置工程名字
project("valjni")
add_library( # Sets the name of the library.
#自己庫(kù)的名稱纵穿,可以隨便起
val
# Sets the library as a shared library.
SHARED
# Provides a relative path to your source file(s).
src/main/cpp/native-lib.cpp)
# 指向.H頭文件位置
target_include_directories(
val
PRIVATE
${CMAKE_SOURCE_DIR}/src/main/cpp/include
)
SET (LIBRARY_OUTPUT_PATH ${PROJECT_BINARY_DIR}/lib)
# 添加第三方庫(kù)
add_library(
#第三方庫(kù)的名稱
WindowServiceSomeipServer
# 設(shè)置引入的函數(shù)庫(kù)類型為靜態(tài)庫(kù)
SHARED
# 表示引入第三方靜態(tài)庫(kù)
IMPORTED)
# 配置第三方庫(kù)鏈接
set_target_properties(
WindowServiceSomeipServer
PROPERTIES
IMPORTED_LOCATION
${CMAKE_SOURCE_DIR}/src/main/jniLibs/${ANDROID_ABI}/libWindowServiceSomeipServer.so)
# 添加第三方庫(kù)
add_library(
#第三方庫(kù)的名稱
nsomeip-matrix
# 設(shè)置引入的函數(shù)庫(kù)類型為靜態(tài)庫(kù)
SHARED
# 表示引入第三方靜態(tài)庫(kù)
IMPORTED)
# 配置第三方庫(kù)鏈接
set_target_properties(
nsomeip-matrix
PROPERTIES
IMPORTED_LOCATION
${CMAKE_SOURCE_DIR}/src/main/jniLibs/${ANDROID_ABI}/libnsomeip-matrix.so)
# 添加第三方庫(kù)
add_library(
#第三方庫(kù)的名稱
nsomeip
# 設(shè)置引入的函數(shù)庫(kù)類型為靜態(tài)庫(kù)
SHARED
# 表示引入第三方靜態(tài)庫(kù)
IMPORTED)
# 配置第三方庫(kù)鏈接
set_target_properties(
nsomeip
PROPERTIES
IMPORTED_LOCATION
${CMAKE_SOURCE_DIR}/src/main/jniLibs/${ANDROID_ABI}/libnsomeip.so)
# 添加第三方庫(kù)
add_library(
#第三方庫(kù)的名稱
cjson
# 設(shè)置引入的函數(shù)庫(kù)類型為靜態(tài)庫(kù)
SHARED
# 表示引入第三方靜態(tài)庫(kù)
IMPORTED)
# 配置第三方庫(kù)鏈接
set_target_properties(
cjson
PROPERTIES
IMPORTED_LOCATION
${CMAKE_SOURCE_DIR}/src/main/jniLibs/${ANDROID_ABI}/libcjson.so)
# 添加第三方庫(kù)
add_library(
#第三方庫(kù)的名稱
c++
# 設(shè)置引入的函數(shù)庫(kù)類型為靜態(tài)庫(kù)
SHARED
# 表示引入第三方靜態(tài)庫(kù)
IMPORTED)
# 配置第三方庫(kù)鏈接
set_target_properties(
c++
PROPERTIES
IMPORTED_LOCATION
${CMAKE_SOURCE_DIR}/src/main/jniLibs/${ANDROID_ABI}/libc++.so)
## 添加第三方庫(kù)
#add_library(
# utils
# # 設(shè)置引入的函數(shù)庫(kù)類型為靜態(tài)庫(kù)
# SHARED
# # 表示引入第三方靜態(tài)庫(kù)
# IMPORTED)
#
## 配置第三方庫(kù)鏈接
#set_target_properties(
# utils
# PROPERTIES
# IMPORTED_LOCATION
# ${CMAKE_SOURCE_DIR}/src/main/jniLibs/${ANDROID_ABI}/libutils.so)
find_library( # Sets the name of the path variable.
log-lib
# Specifies the name of the NDK library that
# you want CMake to locate.
log)
target_link_libraries( # Specifies the target library.
val
c++
# utils
cjson
nsomeip
nsomeip-matrix
WindowServiceSomeipServer
# Links the target library to the log library
# included in the NDK.
${log-lib})
添加第三方so主要就是三步:
add_library
set_target_properties
set_target_properties
${CMAKE_SOURCE_DIR}:CMakeLists.txt所在的路徑,我是放在app下奢人,為什么要放在app目錄下呢谓媒?主要是這樣可以代表根目錄,方便指向?qū)?yīng)的第三方so庫(kù)
${ANDROID_ABI}:代表arm64-v8a何乎,當(dāng)然也可以代表x86之類的句惯,看你jniLibs里面放的啥了
4:在main目錄下新建jniLibs
把so放里面即可
5:用JniUtil去調(diào)用jni
package com.example.aiosceneengine.util;
public class JniUtil {
static {
//一定要記得加載so
System.loadLibrary("val");
// System.loadLibrary("utils");
// System.loadLibrary("c++");
System.loadLibrary("cjson");
System.loadLibrary("nsomeip");
System.loadLibrary("matrix");
System.loadLibrary("WindowServiceSomeipServer");
}
//native表示是加載的jni方法
public static native String test();
public static native int windowStartSomeipServer();
}
CMakeLists.txt里面我們不是已經(jīng)把每個(gè)第三方so起好名字了嘛,在這里就可以load了宪赶,load之后就可以通過(guò)方法去調(diào)用宗弯,方法前要加native哦。
好了以上就是本次內(nèi)容搂妻,散會(huì)蒙保!