Android开发——Cmakelist的使用

一、背景:Cmakelist的使用

项目创建好以后我们可以看到和普通Android项目有以下4个不同。

  1. main 下面增加了 cpp 目录,即放置 c/c++ 代码的地方
  2. module-level 的 build.gradle 有修改
  3. 增加了 CMakeLists.txt 文件
  4. 多了一个 .externalNativeBuild 目录

image.png

二、正文第一章:CMakeLists.txt 文件 讲解

2.1 一个基本的Cmakelist文件

 

cmake_minimum_required(VERSION 3.4.1)
# 编译出一个动态库 native-lib,源文件只有 src/main/cpp/native-lib.cpp
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).
             src/main/cpp/native-lib.cpp )
# 找到预编译库 log_lib 并link到我们的动态库 native-lib中
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
                       # Links the target library to the log library
                       # included in the NDK.
                       ${log-lib} )

2.1.1 add_library - 添加库(三方库.so 或者编译C/C++文件)

2.1.1 .1 添加库的第一种方法

 

add_library(
    avcodec-lib
    SHARED
    IMPORTED)
set_target_properties( avcodec-lib
                       PROPERTIES IMPORTED_LOCATION
                       ${distribution_DIR}/${ANDROID_ABI}/libavcodec-56.so)

添加库,也就是:add_library函数,里面要传入三个参数;第一个是要引入的库别名,第二个是库的类型,是静态库还是动态库。第三个是通过什么样方式引入进来,第三方的一般都是通过包含进来,所以第三个参数基本也是固定的都是写“IMPORTED”。
add_library后,就要设置.so的详细路径了,通过set_target_properties()函数来设置;该函数也是要传入三参数来指定.so库的路径。第一个参数和add_library的第一个参数一样,不过这里的库别名要和add_library的库别名要一致,要不然在编译时会报找不到库的错误。第二个参数是固定的,都是写“ PROPERTIES IMPORTED_LOCATION”主要用来指定库的引入方式。都是通过本地引入。第三个就是库的具体路径,这个不能写错,如果写错了,编译时也同样会找不到库的。只要是引入第三方的库使用add_library就要使用set_target_propeties这个组合,所以它们是成对出现的。add_library还有一种写法那就是CMakeList.txt中默认写法

2.1.1.2 添加库的第二种方法

 

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).
             src/main/cpp/native-lib.cpp )

这样是就是编译我们自己在项目中写的c/c++文件。这里就不用set_target_propeties()。add_library用来设置编译生成的本地库的名字为native-lib,SHARED表示编译生成的是动态链接库(这个概念前面已经提到过了),src/main/cpp/native-lib.cpp表示参与编译的文件的路径,这里面可以写多个文件的路径。

2.1.2 find_library - 查找库

 

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 )

find_library 看到这个名字相信都会知道它是干嘛用的,查找库用的,是用来添加一些我们在编译我们的本地库的时候需要依赖的一些库,这个主要是查找系统库用的,如果项目里面有用到系统的.so库就是要把库名写到这个函数里面去找到相对应的为。由于cmake已经知道系统库的路径,所以我们这里只是指定使用log库,然后给log库起别名为log-lib便于我们后面引用,此处的log库是我们后面调试时需要用来打log日志的库,是NDK为我们提供的。

2.1.3 target_link_libraries

 

target_link_libraries( # Specifies the target library.
                       native-lib
                       # Links the target library to the log library
                       # included in the NDK.
                       ${log-lib} )

函数target_link_libraries()这个是干嘛用的呢,讲到这个函数就要讲到c/c++的编译原理了,在linux中c/c++的编译一般都是用gcc来编译的,c/c++编译时会产生.o文件要通过make工具来把这些.o文件链接起来,这样才能得一个可执行程序。所以.so在编译时要把所有库链接起来才能编译。target_link_libraries()就是干这个事,target_link_libraries 是为了关联我们自己的库和一些第三方库或者系统库。把要链接的库别名都写到这里就可以了,如果是系统的库要用这个格式${库的名字}。

2.2 更多参数的Cmakelist文件

 

#指定需要CMAKE的最小版本
cmake_minimum_required(VERSION 3.4.1)

#C的编译选项是 CMAKE_C_FLAGS
#指定编译参数,可选
#SET(CMAKE_CXX_FLAGS "-Wno-error=format-security -Wno-error=pointer-sign")

#设置生成的so动态库最后输出的路径
#set(CMAKE_LIBRARY_OUTPUT_DIRECTORY ${PROJECT_SOURCE_DIR}/../jniLibs/${ANDROID_ABI})

#调用指定目录下的cmakelist
add_subdirectory(src/main/cpp/mbedtls)

#设置需要引用的库的类型(动/静态库),库的地址,名称
#add_library(lib SHARED/STATIC IMPORTED)
#set_target_properties(lib PROPERTIES IMPORTED_LOCATION ${LIB_DIR}/lib.a)

#设置头文件搜索路径(和此txt同个路径的头文件无需设置),可选
INCLUDE_DIRECTORIES(${CMAKE_CURRENT_SOURCE_DIR}/src/main/cpp/mbedtls/include)

#指定用到的系统库或者NDK库或者第三方库的搜索路径,可选。
#LINK_DIRECTORIES(/usr/local/lib)
add_library( nodepp
             SHARED
             src/main/cpp/utils.c
             src/main/cpp/xxtea.c
             src/main/cpp/mbed_client.c)
target_link_libraries( nodepp
             mbedcrypto
             mbedx509
             mbedtls
             log )


三、正文第二章:NDK自定义配置

3.1 添加多个参与编译的C/C++文件

首先,我们发现我们上面的例子都是涉及到一个C++文件,那么我们实际的项目不可能只有一个C++文件,所以我们首先要改变CMakeLists.txt文件,如下 :

 

add_library( HelloNDK
             SHARED
             src/main/cpp/HelloNDK.c
             src/main/cpp/HelloJNI.c)

简单吧,简单明了,但是这里要注意的是,你在写路径的时候一定要注意当前的CMakeLists.txt在项目中的位置,上面的路径是相对于CMakeLists.txt 写的。

3.2 我们想编译出多个so库

大家会发现,我们上面这样写,由于只有一个CMakeLists.txt文件,所以我们会把所有的C/C++文件编译成一个so库,这是很不合适的,这里我们就试着学学怎么编译出多个so库。

先放上我的项目文件夹结构图:

image.png


然后看看我们每个CMakeLists.txt文件是怎么写的:
one文件夹内的CMakeLists.txt文件的内容:

 

 

ADD_LIBRARY(one-lib SHARED one-lib.c)
target_link_libraries(one-lib log)

two文件夹内的CMakeLists.txt文件的内容:

 

ADD_LIBRARY(two-lib SHARED two-lib.c)
target_link_libraries(two-lib log)

app目录下的CMakeLists.txt文件的内容

 

# Sets the minimum version of CMake required to build the native library.
cmake_minimum_required(VERSION 3.4.1)
add_library( HelloNDK
             SHARED
             src/main/cpp/HelloNDK.c
             src/main/cpp/HelloJNI.c)
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(HelloNDK log)
ADD_SUBDIRECTORY(src/main/cpp/one)
ADD_SUBDIRECTORY(src/main/cpp/two)

通过以上的配置我们可以看出CMakeLists.txt 文件的配置是支持继承的,所以我们在子配置文件中只是写了不同的特殊配置项的配置,最后在最上层的文件中配置子配置文件的路径即可,现在编译项目,我们会在 <项目目录>\app\build\intermediates\cmake\debug\obj\armeabi 下面就可以看到生成的动态链接库。而且是三个动态链接库.

3.3 更改动态链接库生成的目录

我们是不是发现上面的so库的路径太深了,不好找,没事,可以配置,我们只需要在顶层的CMakeLists.txt文件中加入下面这句就可以了

设置生成的so动态库最后输出的路径

 

set(CMAKE_LIBRARY_OUTPUT_DIRECTORY ${PROJECT_SOURCE_DIR}/src/main/jniLibs/${ANDROID_ABI})

然后我们就可以在app/src/main下看到jniLibs目录,在其中看到我们的动态链接库的文件夹和文件(这里直接配置到了系统默认的路径,如果配置到其他路径需要在gradle文件中使用jinLibs.srcDirs = ['newDir']进行指定)。

四、正文第三章:Cmakelist综合运用

4.1 native-lib文件调用一个第三方库

第三方库是被编译成so库直接调用的,配置so库的时候,对应的头文件也要添加上

(1)指定头文件目录:

image.png

 

(2)添加库:

image.png

 

(3)链接到本地库:

image.png

 

譬如加载一个FFMPEG模块

 

include_directories(${pathToFFMPEG}/include)
 #添加ffmpeg对应的头文件目录,${pathToFFMPEG}为前面配置过的路径,可以替换为include_directories(E:/ffmpeg/include)这种路径格式
add_library( ffmpeg 
             SHARED 
             IMPORTED)
 #添加库文件,实际上就是引入so文件,IMPORT代表从第三方引入的意思
set_target_properties( ffmpeg 
                       PROPERTIES IMPORTED_LOCATION 
                       ${pathToProject}/app/src/main/jniLibs/
                       ${ANDROID_ABI}/libffmpeg.so
#这句话是ffmpeg对应的so文件,so文件是放到JNILibs这个文件夹中 
target_link_libraries( $\{log-lib} 
                       native-lib 
                       ffmpeg) 
#为native-lib加载ffmpeg库.

4.2使用两个依赖库,然后生成两个工具库

4. 主要有: 添加库的数目,C/C++文件数目,生成so库的数目

五、正文第四章:Cmakelist使用常见问题

5.1 missing and no known rule to make it 问题

很蛋疼,就是一个很简单的.so路径不对。但是路径放哪儿对呢,要看项目的build.gradle中设置的lib文件夹在哪儿。当你没有设置时,默认是jniLibs文件夹,所以要把so放在这里面,但是如果你设置了,那就要把文件夹放在那个指定的目录下面,就是下面这个设置。

 

 sourceSets {
        main {
            jniLibs.srcDirs = ["libs"]
        }
    }

 

  • 7
    点赞
  • 14
    收藏
    觉得还不错? 一键收藏
  • 1
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值