CMake简介

CMake是跨平台的构建工具,可以用简单的语句来描述所有平台的安装(编译过程)。能够输出各种各样的makefile或project文件。
CMake不直接构建出最终的软件,而是产生其他工具的脚本(如makefile的),然后再依据这个工具的构建方式使用。
AndroidStudio利用CMake生成的是ninja,ninja是一个小型的关注速度的构建系统。
CMake是一个跨平台的支持产出各种不同的构建脚本的一个工具。

创建一个项目,默认生成的CMakeLists.txt内容

最低支持的版本 cmake_minimum_required

1
2
# 最低支持的版本,并不是最终的版本,最终版本在 app/build.gradle 中设置的
cmake_minimum_required(VERSION 3.4.1)

添加库 add_library

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
# 默认生成的,导入native-lib源文件
add_library(
native-lib
SHARED
native-lib.cpp)

# 可以写成下方这样
# GLOB 收集源文件列表; 定义一个变量SOURCE
# 添加库 native-lib为库的名字,最终为libnative-lib.so;
# SHARED为动态库xxx.so,STATIC为静态库xxx.a;
# ${SOURCE} 为把对应的源文件变异成libnative-lib.so库

file(GLOB SOURCE *.cpp *.c)
add_library(
native-lib
SHARED
${SOURCE})
1
2
3
4
5
6
7
8
9
10
# 查找动态库log,并把它命名为log-lib别名
find_library(
log-lib
log)

# 设置链接, native-lib为总库,位置在 apk/lib/arm64-v8a/libnative-lib.so,如下方截图
# 通过${log-lib}把log库链接到总库中,总库的cpp代码,就可以使用android/log.h的库实现代码了
target_link_libraries(
native-lib
${log-lib})

1
2
3
4
# 也可以写成这样
target_link_libraries(
native-lib
log)

这里是log库,是否还有其它库能写,保证写正确呢?

  1. 先看下ndk版本,在local.properties文件中
    ndk.dir=C:\Users\xxxxxx\Android\Sdk\ndk\20.0.5594570

  2. 看 app/build.gradle 中的 minSdkVersion设置的多少
    minSdkVersion 21

  3. 根据上方的内容去查找这文件即可
    C:\Users\xxxxxx\Android\Sdk\ndk\20.0.5594570\build\cmake\system_libs.cmake
    文件内容为:
    set(NDK_SYSTEM_LIBS “libEGL.so;libGLESv1_CM.so;libGLESv2.so;libGLESv3.so;libOpenMAXAL.so;libOpenSLES.so;libaaudio.so;libamidi.so;libandroid.so;libbinder_ndk.so;libc.so;libcamera2ndk.so;libdl.so;libjnigraphics.so;liblog.so;libm.so;libmediandk.so;libnativewindow.so;libneuralnetworks.so;libstdc++.so;libsync.so;libvulkan.so;libz.so”)

信息输出 - message

  • (无) = 重要消息;
  • STATUS = 非重要消息;
  • WARNING = CMake 警告, 会继续执行;
  • AUTHOR_WARNING = CMake 警告 (dev), 会继续执行;
  • SEND_ERROR = CMake 错误, 继续执行,但是会跳过生成的步骤;
  • FATAL_ERROR = CMake 错误, 终止所有处理过程;
1
2
3
message("hello1 ================================")
message(STATUS "hello2 =========================")
message(WARNING "hello3 =========================")

输出的信息在这里查看:(记得安装到手机上先)
地址:\app.cxx\cmake\debug\arm64-v8a\cmake_server_log.txt

先refresh,再make一下,也能在最下方的build中看:

变量 set

1
2
3
4
5
6
7
8
9
10
# 声明变量
# 只有字符串,没有int
set(mNumber 666)
message(STATUS "mNumber = ${mNumber}") # 打印结果: mNumber = 666

# 列表 都是string类型
set(mList1 1 2 3 66 100) # 这两种方式等价的
set(mList2 "100;200;500") # 这两种方式等价的
message(STATUS "mList1 = ${mList1}") # 打印结果: mList1 = 1;2;3;66;100
message(STATUS "mList2 = ${mList2}") # 打印结果: mList2 = 100;200;500

条件命令 if

CMake关于 if 的介绍:https://cmake.org/cmake/help/v3.10/command/if.html

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
# true(1,ON,YES,TRUE,Y,非0的值)
# false(0,OFF,NO,FALSE,N,IGNORE,NOTFOUND)
set(mFlagIsON ON)
set(mFlagIsOFF OFF)

if(${mFlagIsON})
message("mFlagIsON为 1,ON,YES,TRUE,Y,非0的值!")
endif()

if(NOT ${mFlagIsOFF})
message("mFlagIsOFF为 0,OFF,NO,FALSE,N,IGNORE,NOTFOUND!")
endif()

set(mNum 101)
if(${mNum} STREQUAL 100)
message("mNum 为 100!")
elseif(${mNum} STREQUAL 101)
message("mNum 为 101!")
else()
message("mNum 不为 100、101!")
endif()

循环 while

1
2
3
4
5
6
7
8
9
10
11
12
13
set(mStr "hello")
while(mStr STREQUAL "hello")
message("mStr 为 hello!")
set(mStr "${mStr}world")
message("mStr 被修改为 ${mStr}")
endwhile()

set(mStr2 "helloworld")
while(NOT mStr2 STREQUAL "hello")
message("mStr2 不为 hello!")
break()
# continue()
endwhile()

循环 foreach

CMake关于 foreach 的介绍:https://cmake.org/cmake/help/v3.10/command/foreach.html?highlight=foreach

1
2
3
4
5
foreach(item 1 2 3 66 100)
message("foreach1 item=${item}")
endforeach()

打印结果:遍历每一个
1
2
3
4
5
foreach(item RANGE 10)
message("foreach2 item=${item}")
endforeach()

打印结果: 从 1 到 10
1
2
3
4
5
6
7
# foreach(loop_var RANGE start stop [step])

foreach(item RANGE 1 20 2)
message("foreach3 item=${item}")
endforeach()

打印结果: 从 1 开始,每次下标加2
1
2
3
4
5
6
7
set(mList1 1 2 3 66 100)

foreach(item IN LISTS mList1)
message("foreach4 item=${item}")
endforeach()

打印结果:遍历每一个

函数 function

CMake关于 function 的介绍:https://cmake.org/cmake/help/v3.10/command/function.html?highlight=function
ARGC:传入的参数个数
ARGV:所有的参数
ARGV0:第一个参数

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
function(myFunction n1 n2 n3)
message("n1= ${n1}")
message("n2= ${n2}")
message("n3= ${n3}")

message("argc= ${ARGC}")

message("argv= ${ARGV}")
message("argv0= ${ARGV0}, argv1= ${ARGV1}, argv2= ${ARGV2}")
endfunction()

myFunction(100 299 99999)

打印结果:
n1= 100
n2= 299
n3= 99999
argc= 3
argv= 100;299;99999
argv0= 100, argv1= 299, argv2= 99999

静态库和动态库的区别

静态库

在程序编译时会被链接到目标代码中,相当于静态库中的代码被拷贝到总库中;
程序运行期将不再需要该静态库。

动态库

在程序编译时并不会被链接到目标代码中,只是做地址记录;
在程序运行期,通过地址记录,做地址会填;
因此程序运行期还需要动态库存在。

CMakeLists.txt 路径配置

默认是在:\app\src\main\cpp\CMakeLists.txt

也可以设置在其他位置,但名称必须是CMakeLists.txt

创建一个新的CMakeLists.txt

拷贝一份cpp下的CMakeLists.txt,到main目录下

修改app/build.gradle文件指定新的CMakeLists.txt

导入头文件 及 动态库

方式1

CMAKE_SOURCE_DIR 等于 CMakeLists.txt所在的地址目录
CMAKE_ANDROID_ARCH_ABI 等于 当前手机的CPU架构

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
## 导入fmod头文件
include_directories("fmodinc")

## 导入库文件
# CMAKE_SOURCE_DIR 等于 CMakeLists.txt所在的地址目录
# CMAKE_ANDROID_ARCH_ABI 等于 当前手机的CPU架构
set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -L${CMAKE_SOURCE_DIR}/../jniLibs/${CMAKE_ANDROID_ARCH_ABI}")

# 链接到总库中
# native-lib 总库
target_link_libraries(
native-lib
log
fmod
fmodL
)

方式2

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
## 导入fmod头文件
include_directories("fmodinc")

## 导入库文件
add_library(fmod SHARED IMMPORTED)
set_target_properties(fmod PROPERTIES IMPORTED_LOCATION ${CMAKE_SOURCE_DIR}/../jniLibs/${CMAKE_ANDROID_ARCH_ABI}/libfmod.so)

add_library(fmodL SHARED IMPORTED)
set_target_properties(fmodL PROPERTIES IMPORTED_LOCATION ${CMAKE_SOURCE_DIR}/../jniLibs/${CMAKE_ANDROID_ARCH_ABI}/libfmodL.so)

# 链接到总库中
target_link_libraries(
native-lib
log
fmod
fmodL
)

导入静态库

1
2
3
4
5
6
7
8
add_library(getndk STATIC IMPORTED)
set_target_properties(getndk PROPERTIES IMPORTED_LOCATION ${CMAKE_SOURCE_DIR}/libgetndk.a)

target_link_libraries(
native-lib
log
getndk
)

源码构建方式

libcount:

libget:

app\src\main\cpp\CMakeLists.txt

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
cmake_minimum_required(VERSION 3.4.1)

file(GLOB SOURCE *.cpp *.c)
add_library(
native-lib
SHARED
${SOURCE})

add_subdirectory(${CMAKE_SOURCE_DIR}/libget)
add_subdirectory(${CMAKE_SOURCE_DIR}/libcount)

target_link_libraries(
native-lib
log
get
count
)

app\src\main\cpp\native-lib.cpp

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
#include <jni.h>
#include <string>
#include <android/log.h>

#define TAG "AAAAAAAAAAAAAAAA"
#define LOGD(...) __android_log_print(ANDROID_LOG_DEBUG, TAG, __VA_ARGS__);

extern "C" {
#include "libget/getutil.h"
}

#include "libcount/countutil.h"

extern "C" JNIEXPORT jstring JNICALL Java_com_example_mycmaketest_MainActivity_stringFromJNI(
JNIEnv *env,
jobject /* this */) {

// get信息的输出显示
LOGD("get1_action: %s", get1_action());
LOGD("get2_action: %s", get2_action());

// count信息的输出显示
LOGD("add_action: %d", add_action(10, 20));
LOGD("sub_action: %d", sub_action(100, 10));

std::string hello = "Hello from C++";
return env->NewStringUTF(hello.c_str());
}

参考文档

developers 配置 CMake:https://developer.android.com/studio/projects/configure-cmake
Android NDK导入C库示例(fmod): https://blog.csdn.net/yan13507001470/article/details/120559455
参考链接:https://blog.csdn.net/yan13507001470/article/details/120807408