Android:清晰讲解JNI 与 NDK(含实例教学)
ztj100 2024-12-06 20:56 16 浏览 0 评论
作者:Carson_Ho 来源:掘金
前言
- 在Android开发中,使用 NDK开发的需求正逐渐增大
- 但很多人却搞不懂 JNI 与 NDK 到底是怎么回事
- 今天,我将先介绍JNI 与 NDK & 之间的区别,手把手进行 NDK的使用教学,希望你们会喜欢
目录
1. JNI介绍
1.1 简介
- 定义:Java Native Interface,即 Java本地接口
- 作用: 使得Java 与 本地其他类型语言(如C、C++)交互
即在 Java代码 里调用 C、C++等语言的代码 或 C、C++代码调用 Java 代码
- 特别注意: JNI是 Java 调用 Native 语言的一种特性 JNI 是属于 Java 的,与 Android 无直接关系
1.2 为什么要有 JNI
- 背景:实际使用中,Java 需要与 本地代码 进行交互
- 问题:因为 Java 具备跨平台的特点,所以Java 与 本地代码交互的能力非常弱
- 解决方案: 采用 JNI特性 增强 Java 与 本地代码交互的能力
1.3 实现步骤
- 在Java中声明Native方法(即需要调用的本地方法)
- 编译上述 Java源文件javac(得到 .class文件)
- 通过 javah 命令导出JNI的头文件(.h文件)
- 使用 Java需要交互的本地代码 实现在 Java中声明的Native方法
如 Java 需要与 C++ 交互,那么就用C++实现 Java的Native方法
- 编译.so库文件
- 通过Java命令执行 Java程序,最终实现Java调用本地代码
2. NDK介绍
2.1 简介
- 定义:Native Development Kit,是 Android的一个工具开发包
NDK是属于 Android 的,与Java并无直接关系
- 作用:快速开发C、 C++的动态库,并自动将so和应用一起打包成 APK
即可通过 NDK在 Android中 使用 JNI与本地代码(如C、C++)交互
- 应用场景:在Android的场景下 使用JNI
即 Android开发的功能需要本地代码(C/C++)实现
- 特点
- 额外注意
2.2 使用步骤
- 配置 Android NDK环境
- 创建 Android 项目,并与 NDK进行关联
- 在 Android 项目中声明所需要调用的 Native方法
- 使用 Android需要交互的本地代码 实现在Android中声明的Native方法
比如 Android 需要与 C++ 交互,那么就用C++ 实现 Java的Native方法
- 通过 ndk - bulid 命令编译产生.so库文件
- 编译 Android Studio 工程,从而实现 Android 调用本地代码
3. NDK与JNI关系
4. 具体使用
本文根据版本的不同介绍了两种在Android Studio中实现 NDK的方法:Android Studio2.2 以下 & 2.2以上
4.1 Android Studio2.2 以下实现NDK
- 步骤如下 配置 Android NDK环境 关联 Andorid Studio项目 与 NDK 创建本地代码文件(即需要在 Android项目中调用的本地代码文件) 创建 Android.mk文件 & Application.mk文件 编译上述文件,生成.so库文件,并放入到工程文件中 在 Andoird Studio项目中使用 NDK实现 JNI 功能
- 步骤详解
步骤1:配置 Android NDK环境
具体请看文章一定能成功的Android NDK环境配置教程
步骤2: 关联Andorid Studio项目 与 NDK
- 当你的项目每次需要使用 NDK 时,都需要将该项目关联到 NDK
此处使用的是Andorid Studio,与Eclipse不同 还在使用Eclipse的同学请自行查找资料配置
- 具体配置如下
a. 在Gradle的 local.properties中添加配置
ndk.dir=/Users/Carson_Ho/Library/Android/sdk/ndk-bundle
若ndk目录存放在SDK的目录中,并命名为ndk-bundle,则该配置自动添加
b. 在Gradle的 gradle.properties中添加配置
android.useDeprecatedNdk=true
// 对旧版本的NDK支持
c. 在Gradle的build.gradle添加ndk节点
- 至此,将Andorid Studio的项目 与 NDK 关联完毕
- 下面,将真正开始讲解如何在项目中使用NDK
步骤3:创建本地代码文件
- 即需要在Android项目中调用的本地代码文件
此处采用 C++作为展示
test.cpp
# include <jni.h>
# include <stdio.h>
extern "C"
{
JNIEXPORT jstring JNICALL Java_scut_carson_1ho_ndk_1demo_MainActivity_getFromJNI(JNIEnv *env, jobject obj ){
// 参数说明
// 1. JNIEnv:代表了VM里面的环境,本地的代码可以通过该参数与Java代码进行操作
// 2. obj:定义JNI方法的类的一个本地引用(this)
return env -> NewStringUTF("Hello i am from JNI!");
// 上述代码是返回一个String类型的"Hello i am from JNI!"字符串
}
}
此处需要注意:
- 如果本地代码是C++(.cpp或者.cc),要使用extern "C" { }把本地方法括进去
- JNIEXPORT jstring JNICALL中的JNIEXPORT 和 JNICALL不能省
- 关于方法名Java_scut_carson_1ho_ndk_1demo_MainActivity_getFromJNI 格式 = Java _包名 _ 类名_Java需要调用的方法名 Java必须大写 对于包名,包名里的.要改成_,_要改成_1 如我的包名是:scut.carson_ho.ndk_demo,则需要改成scut_carson_1ho_ndk_1demo
最后,将创建好的test.cpp文件放入到工程文件目录中的src/main/jni文件夹
若无jni文件夹,则手动创建。
下面我讲解一下JNI类型与Java类型对应的关系介绍
步骤4:创建Android.mk文件
- 作用:指定源码编译的配置信息
如工作目录,编译模块的名称,参与编译的文件等
- 具体使用
Android.mk
LOCAL_PATH := $(call my-dir)
// 设置工作目录,而my-dir则会返回Android.mk文件所在的目录
include $(CLEAR_VARS)
// 清除几乎所有以LOCAL——PATH开头的变量(不包括LOCAL_PATH)
LOCAL_MODULE := hello_jni
// 设置模块的名称,即编译出来.so文件名
// 注,要和上述步骤中build.gradle中NDK节点设置的名字相同
LOCAL_SRC_FILES := test.cpp
// 指定参与模块编译的C/C++源文件名
include $(BUILD_SHARED_LIBRARY)
// 指定生成的静态库或者共享库在运行时依赖的共享库模块列表。
最后,将上述文件同样放在src/main/jni文件夹中。
步骤5:创建Application.mk文件
- 作用:配置编译平台相关内容
- 具体使用
Application.mk
APP_ABI := armeabi
// 最常用的APP_ABI字段:指定需要基于哪些CPU平台的.so文件
// 常见的平台有armeabi x86 mips,其中移动设备主要是armeabi平台
// 默认情况下,Android平台会生成所有平台的.so文件,即同APP_ABI := armeabi x86 mips
// 指定CPU平台类型后,就只会生成该平台的.so文件,即上述语句只会生成armeabi平台的.so文件
最后,将上述文件同样放在src/main/jni文件夹中
步骤6:编译上述文件,生成.so库文件
- 经过上述步骤,在src/main/jni文件夹中已经有3个文件
- 打开终端,输入以下命令
// 步骤1:进入该文件夹
cd /Users/Carson_Ho/AndroidStudioProjects/NDK_Demo/app/src/main/jni
// 步骤2:运行NDK编译命令
ndk-build
- 编译成功后,在src/main/会多了两个文件夹libs & obj,其中libs下存放的是.so库文件
步骤7:在src/main/中创建一个名为jniLibs的文件夹,并将上述生成的so文件夹放到该目录下
- 要把名为 CPU平台的文件夹放进去,而不是把.so文件放进去
- 如果本来就有.so文件,那么就直接创建名为jniLibs的文件夹并放进去就可以
步骤8:在Andoird Studio项目中使用NDK实现JNI功能
- 此时,我们已经将本地代码文件编译成.so库文件并放入到工程文件中
- 在Java代码中调用本地代码中的方法,具体代码如下:
MainActivity.java
public class MainActivity extends AppCompatActivity {
// 步骤1:加载生成的so库文件
// 注意要跟.so库文件名相同
static {
System.loadLibrary("hello_jni");
}
// 步骤2:定义在JNI中实现的方法
public native String getFromJNI();
// 此处设置了一个按钮用于触发JNI方法
private Button Button;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
// 通过Button调用JNI中的方法
Button = (Button) findViewById(R.id.button);
Button.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
Button.setText(getFromJNI());
}
});
主布局文件:activity_main.xml
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:tools="http://schemas.android.com/tools"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:paddingBottom="@dimen/activity_vertical_margin"
android:paddingLeft="@dimen/activity_horizontal_margin"
android:paddingRight="@dimen/activity_horizontal_margin"
android:paddingTop="@dimen/activity_vertical_margin"
tools:context="scut.carson_ho.ndk_demo.MainActivity">
// 此处设置了一个按钮用于触发JNI方法
<Button
android:id="@+id/button"
android:layout_centerInParent="true"
android:layout_width="300dp"
android:layout_height="50dp"
android:text="调用JNI代码" />
</RelativeLayout>
结果展示
源码地址
Carson-Ho的Github地址:NDK_Demo
4.2 Android Studio2.2 以上实现NDK
- 如果你的Android Studio是2.2以上的,那么请采用下述方法
因为Android Studio2.2以上已经内部集成 NDK,所以只需要在Android Studio内部进行配置就可以
- 步骤讲解
步骤1:按提示创建工程
在创建工程时,需要配置 NDK,根据提示一步步安装即可。
步骤2:根据需求使用NDK
- 配置好NDK后,Android Studio会自动生成C++文件并设置好调用的代码
- 你只需要根据需求修改C++文件 & Android就可以使用了。
相关推荐
- 电脑装系统用GHOST好,还是原装版本好?老司机都是这么装的
-
Hello大家好,我是兼容机之家的咖啡。安装Windows系统是原版ISO好还是ghost好呢?针对这个的问题,我们先来科普一下什么是ghost系统,和原版ISO镜像两者之间有哪些优缺点。如果是很了解...
- 苹果 iOS 14.5.1/iPadOS 14.5.1 正式版发布
-
IT之家5月4日消息今日凌晨,苹果发布了iOS14.5.1与iPadOS14.5.1正式版更新。这一更新距iOS14.5正式版发布过去了一周时间。IT之家了解到,苹果表示,...
- iOS 13.1.3 正式版发布 包含错误修复和改进
-
苹果今天发布了iOS13.1.3和iPadOS13.1.3,这是iOS13发布之后第四个升级补丁。iOS13.1.2两周前发布。iOS13.1.3主要包括针对iPad和...
- 还不理解 Error 和 Exception 吗,看这篇就够了
-
在Java中的基本理念是结构不佳的代码不能运行,发现错误的理想时期是在编译期间,因为你不用运行程序,只是凭借着对Java基本理念的理解就能发现问题。但是编译期并不能找出所有的问题,有一些N...
- Linux 开发人员发现了导致 MacBook“无法启动”的 macOS 错误
-
“多个严重”错误影响配备ProMotion显示屏的MacBookPro。...
- 启动系统时无法正常启动提示\windows\system32\winload.efi
-
启动系统时无法正常启动提示\windows\system32\winload.efi。该怎么解决? 最近有用户遇到了开机遇到的问题,是Windows未能启动。原因可能是最近更改了硬件或软件。虽然提...
- 离线部署之两种构建Ragflow镜像的方式,dify同理
-
在实际项目交付过程中,经常遇到要离线部署的问题,生产服务器无法连接外网,这时就需要先构建好ragflow镜像,然后再拷到U盘或刻盘,下面介绍两种构建ragflow镜像的方式。性能测试(网络情况好的情况...
- Go语言 error 类型详解(go语言 异常)
-
Go语言的error类型是用于处理程序运行中错误情况的核心机制。它通过显式的返回值(而非异常抛出)来管理错误,强调代码的可控性和清晰性。以下是详细说明及示例:一、error类型的基本概念内置接口...
- Mac上“闪烁的问号”错误提示如何修复?
-
现在Mac电脑的用户越来越多,Mac电脑在使用过程中也会出现系统故障。当苹果电脑无法找到系统软件时,Mac会给出一个“闪烁的问号”的标志。很多用户受到过闪烁问号这一常见的错误提示的影响,如何解决这个问...
- python散装笔记——177 sys 模块(python sys模块详解)
-
sys模块提供了访问程序运行时环境的函数和值,例如命令行参数...
- 30天自制操作系统:第一天(30天自制操作系统电子书)
-
因为咱们的目的是为了研究操作系统的组成,所以直接从系统启动的第二阶段的主引导记录开始。前提是将编译工具放在该文件目录的同级目录下,该工具为日本人川合秀实自制的编译程序,优化过的nasm编译工具。...
- 五大原因建议您现在不要升级iOS 13或iPadOS
-
今天苹果放出了iPadOS和iOS13的公测版本,任何对新版功能感兴趣的用户都可以下载安装参与测试。除非你想要率先体验Dark模式,以及使用AppleID来登陆Facebook等服务,那么外媒CN...
- Python安装包总报错?这篇解决指南让你告别pip烦恼!
-
在Python开发中,...
- 苹果提供了在M1 Mac上修复macOS重装错误的方案
-
#AppleM1芯片#在苹果新的M1Mac推出后不久,我们看到有报道称,在这些机器上恢复和重新安装macOS,可能会导致安装错误,使你的Mac无法使用。具体来说,错误信息如下:"An...
- 黑苹果卡代码篇三:常见卡代码问题,满满的干货
-
前言...
你 发表评论:
欢迎- 一周热门
- 最近发表
-
- 电脑装系统用GHOST好,还是原装版本好?老司机都是这么装的
- 苹果 iOS 14.5.1/iPadOS 14.5.1 正式版发布
- iOS 13.1.3 正式版发布 包含错误修复和改进
- 还不理解 Error 和 Exception 吗,看这篇就够了
- Linux 开发人员发现了导致 MacBook“无法启动”的 macOS 错误
- 启动系统时无法正常启动提示\windows\system32\winload.efi
- 离线部署之两种构建Ragflow镜像的方式,dify同理
- Go语言 error 类型详解(go语言 异常)
- Mac上“闪烁的问号”错误提示如何修复?
- python散装笔记——177 sys 模块(python sys模块详解)
- 标签列表
-
- idea eval reset (50)
- vue dispatch (70)
- update canceled (42)
- order by asc (53)
- spring gateway (67)
- 简单代码编程 贪吃蛇 (40)
- transforms.resize (33)
- redisson trylock (35)
- 卸载node (35)
- np.reshape (33)
- torch.arange (34)
- node卸载 (33)
- npm 源 (35)
- vue3 deep (35)
- win10 ssh (35)
- exceptionininitializererror (33)
- vue foreach (34)
- idea设置编码为utf8 (35)
- vue 数组添加元素 (34)
- std find (34)
- tablefield注解用途 (35)
- python str转json (34)
- java websocket客户端 (34)
- tensor.view (34)
- java jackson (34)