當(dāng)我們創(chuàng)建一個(gè)NDK工程時(shí),會(huì)自動(dòng)創(chuàng)建一個(gè)CMakeLists.txt的文件,在AS中c++的編譯器是使用LLVM,規(guī)則為cmake,今天來學(xué)習(xí)下cmake的基本套路
首先,我創(chuàng)建了兩個(gè)NDK工程,第一個(gè)工程為lib,為第二個(gè)工程提供so庫

我們修改cpp文件,新增一個(gè)求和方法
#include <jni.h>
#include <string>
int sum(int a, int b) {
return a + b;
}
由于我們這個(gè)工程要提供so庫,所以在CMakeLists中改下編譯出來的so庫名稱:test-lib

編譯下,找到生成的so庫文件夾,復(fù)制到第二個(gè)工程

如果想要指定平臺(tái)可以在gradle中配置:
在defaultConfig目錄里面
ndk {
abiFilters "armeabi","x86"
}
復(fù)制到libs下

AS中默認(rèn)存放so庫的目錄需要在src/main中創(chuàng)建一個(gè)jniLibs的文件夾,也可以通過gradle配置,指定目錄
在app.gralde中的android目錄下
sourceSets.main {
jniLibs.srcDirs = ['libs']
jni.srcDirs = []
}
這邊使用的是libs目錄作為so庫的存放目錄,接下來我們來配置第二個(gè)工程的CMakeLists
1.首先,為了以后方便使用,我們?yōu)閟o庫的路徑設(shè)置一個(gè)別名
#設(shè)置so庫路徑
set(my_lib_path ${CMAKE_SOURCE_DIR}/../../../libs)
${CMAKE_SOURCE_DIR}為CMakeLists文件的當(dāng)前路徑,以后我們就可以直接使用my_lib_path了
2.第二步,我們配置導(dǎo)入的so庫
#將第三方庫作為動(dòng)態(tài)庫引用
add_library(test-lib
SHARED
IMPORTED)
這邊我們只需要修改庫的名稱(test-lib)就可以了,其他的復(fù)制粘貼
3.第三步,配置第三方庫的路徑,這邊就要用到我們開始定義的my_lib_path別名了
#指定第三方庫的絕對(duì)路徑
set_target_properties(test-lib
PROPERTIES IMPORTED_LOCATION
${my_lib_path}/${ANDROID_ABI}/libtest-lib.so)
同樣的,我們只需要關(guān)注上一步定義好的名稱和連接的so庫的絕對(duì)路徑,另外,關(guān)于${ANDROID_ABI}的說明:系統(tǒng)會(huì)自動(dòng)根據(jù)apk安裝時(shí)的平臺(tái),自動(dòng)將相應(yīng)平臺(tái)下的so庫導(dǎo)入
4.最后在target_link_libraries中添加第三方庫名稱
target_link_libraries( # Specifies the target library.
native-lib
test-lib
# Links the target library to the log library
# included in the NDK.
${log-lib})
完整CMakeLists:
cmake_minimum_required(VERSION 3.4.1)
#設(shè)置so庫路徑
set(my_lib_path ${CMAKE_SOURCE_DIR}/libs)
#將第三方庫作為動(dòng)態(tài)庫引用
add_library(test-lib
SHARED
IMPORTED)
#指定第三方庫的絕對(duì)路徑
set_target_properties(test-lib
PROPERTIES IMPORTED_LOCATION
${my_lib_path}/${ANDROID_ABI}/libtest-lib.so)
add_library( # Sets the name of the library.
native-lib
# Sets the library as a shared library.
SHARED
# Provides a relative path to your source file(s).
native-lib.cpp)
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.
native-lib
test-lib
# Links the target library to the log library
# included in the NDK.
${log-lib})
接下來,我們開始使用導(dǎo)入的庫中的方法:求和方法
1.首先,創(chuàng)建一個(gè).h文件,聲明導(dǎo)入方庫的方法:

//
// Created by aruba on 2020/4/13.
//
#ifndef NDKAPPLICATION_TEST_LIB_H
#define NDKAPPLICATION_TEST_LIB_H
//申明外部 函數(shù) 外部屬性
extern int sum(int a, int b);
#endif //NDKAPPLICATION_TEST_LIB_H
2.在需要使用的cpp中引入頭文件,并調(diào)用
#include <jni.h>
#include <string>
#include <android/log.h>
#include <assert.h>
#include "test-lib.h"
#define TAG "C++"
#define LOGI(...) __android_log_print(ANDROID_LOG_INFO, TAG, __VA_ARGS__)
#define NELEM(x) ((int) (sizeof(x) / sizeof((x)[0])))
extern "C"
JNIEXPORT jstring JNICALL native_stringFromJNI(JNIEnv *env, jclass type) {
return env->NewStringUTF("C++");
}
JNIEXPORT jint JNICALL native_sum(JNIEnv *env, jclass type, jint a, jint b) {
return sum(a, b);
}
static const JNINativeMethod gMethods[] = {
{
"stringFromJNI", "()Ljava/lang/String;", (void *) native_stringFromJNI
},
{
"sum", "(II)I", (void *) native_sum
}
};
static int registerNatives(JNIEnv *env) {
LOGI("registerNatives begin");
jclass clazz;
//找到j(luò)ava的類
clazz = env->FindClass("com/aruba/ndkapplication/JniUtils");
if (clazz == NULL) {
LOGI("clazz is null");
return JNI_FALSE;
}
if (env->RegisterNatives(clazz, gMethods, NELEM(gMethods)) < 0) {
LOGI("RegisterNatives error");
return JNI_FALSE;
}
return JNI_TRUE;
}
JNIEXPORT jint JNI_OnLoad(JavaVM *vm, void *reserved) {
LOGI("jni_OnLoad begin");
JNIEnv *env = NULL;
if (vm->GetEnv((void **) &env, JNI_VERSION_1_4) != JNI_OK) {
LOGI("ERROR: GetEnv failed\n");
return -1;
}
assert(env != NULL);
registerNatives(env);
return JNI_VERSION_1_4;
}
在java中調(diào)用
package com.aruba.ndkapplication;
public class JniUtils {
static {
System.loadLibrary("native-lib");
}
public static native String stringFromJNI();
public static native int sum(int a, int b);
}
public class MainActivity extends AppCompatActivity {
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
// Example of a call to a native method
TextView tv = findViewById(R.id.sample_text);
tv.setText(JniUtils.sum(10, 20) + "");
}
}
