使用CMake开发Jni相对传统方式要简单一些(传统方式参考:https://www.jianshu.com/p/7e0a7ede540b)
配置环境
使用CMake进行Jni开发需要使用CMake插件、LLDB插件、NDK插件,这些都可以通过Android Studio很快地安装。
打开SDK Manager,找到Android SDK->SDK Tool选项,安装CMake、LLDB、NDK插件。
创建支持c++代码的工程
勾选Include C++ support选项,勾选后Android Studio可以更好地支持及帮助我们检查jni代码,之后一路next即可。
image.png
如此构建就会自动生成一个demo示例,直接跑就可以正常调用演示了(具体自动生成的文件,同下方介绍)。
如果项目已经存在,想用cmake,那么可以对比上方demo,看看做出了什么变化,进行对应调整即可:
- 首先看主module的gradle文件,android节点下添加:
externalNativeBuild {
cmake {
path "CMakeLists.txt"
}
}
- 然后在主module的gradle文件,defaultConfig节点下添加:
externalNativeBuild {
cmake {
cppFlags "-std=c++14"
}
}
- 新增cpp文件夹 ,对应目录为src\main\cpp,与src\main\java同级,默认只有一个native-lib.cpp文件(本例子demo,具体的实现可自己写),没有.mk文件及其他,比较简单:
#include <jni.h>
#include <string>
extern "C"
JNIEXPORT jstring
JNICALL
Java_com_xxx_xxx_MainActivity_stringFromJNI(//注意这里的路径要和包名对应 Java_包名
JNIEnv *env,
jobject /* this */) {
std::string hello = "Hello from C++";
return env->NewStringUTF(hello.c_str());
}
- 新建 CMakeLists.txt,目录和主module gradle文件平级
# 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构造本地库所需要的最低版本
cmake_minimum_required(VERSION 3.4.1)
# Creates and names a library, sets it as either STATIC
# or SHARED, and provides the relative paths to its source code.
# You can define multiple libraries, and CMake builds them for you.
# Gradle automatically packages shared libraries with your APK.
# 创建并命名库,可以将库的类型设置为静态
# 或共享,并提供该库源代码的相对路径。
# 你可以定义多个library库(如果有多个so要生成,就可以在这里配置多个add_library),并使用CMake来构建。
# Gradle会自动打包库集成到apk中。
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 )
# Searches for a specified prebuilt library and stores the path as a
# variable. Because CMake includes system libraries in the search path by
# default, you only need to specify the name of the public NDK library
# you want to add. CMake verifies that the library exists before
# completing its build.
# NDK提供了一套实用的原生API和库,可以使用find_library搜索NDK中存在的库,只需要加入所需要使用库的名称即可,如下面的日志库log-lib。
# 更多的库可参考https://developer.android.com/ndk/guides/stable_apis
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 )
# Specifies libraries CMake should link to your target library. You
# can link multiple libraries, such as libraries you define in this
# build script, prebuilt third-party libraries, or system libraries.
# 指定CMake连接到目标库中,可以链接多个库,如上面的native-lib库以及NDK中自带的log-lib库(这里也可以同时添加多个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} )
整体目录结构:
看目录可以知道,相比普通工程,AndroidStudio主动为我们生成了一个cpp的目录,该目录主要存放调用的c++头文件,源代码及jni代码,还有一个CMakeLists.txt的文件,cpp目录下面包含一个native-lib.cpp文件(稍后会讲解),这里需要关注的主要文件有3个,native-llib.cpp、MainActivity和CMakeList.txt。下面重点讲解一下这几个文件的作用。
调用方式
public class MainActivity extends AppCompatActivity {
// Used to load the 'native-lib' library on application startup.
static {
System.loadLibrary("native-lib");
}
@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(stringFromJNI());
}
/**
* A native method that is implemented by the 'native-lib' native library,
* which is packaged with this application.
*/
public native String stringFromJNI();
}
这样项目直接运行,就会自动把生产的so打到apk里面了,就可以完成正常调用了。
都做了什么:
- 1、首先Gradle调用了工程中的CMakeLists.txt文件
- 2、CMake按照期里面的命令将C++源文件native-lib.cpp编译到共享对象库中,并命名为libnative-lib.so,之后由Gradle将其打包到APK中。**
说明:本文仅用做个人笔记,参考自:https://blog.csdn.net/u013564742/article/details/86512271
网友评论