Android NDK开发,使用ndk-build编译

/* DO NOT EDIT THIS FILE - it is machine generated /
#include <jni.h>
/
Header for class com_ang_ndkdemo_MainActivity */

#ifndef _Included_com_ang_ndkdemo_MainActivity
#define _Included_com_ang_ndkdemo_MainActivity
#ifdef __cplusplus
extern “C” {
#endif
/*

  • Class: com_ang_ndkdemo_MainActivity
  • Method: fromJNIString
  • Signature: ()Ljava/lang/String;
    */
    JNIEXPORT jstring JNICALL Java_com_ang_ndkdemo_MainActivity_fromJNIString
    (JNIEnv *, jobject);

/JNIEnv 是定义任意native函数的第一个参数(包括调用JNI的RegisterNatives函数注册的函数),指向JVM函数表的指针,函数表中的每一个入口指向一个JNI函数,每个函数用于访问JVM中特定的数据结构。*/

#ifdef __cplusplus
}
#endif
#endif

4,生成. h 头文件时候,如果出现 “找不到类文件” 的错误请参考  blog.csdn.net/ezconn/arti… 这篇文章

注意:

a. 包名或类名或方法名中含下划线 _ 要用 _1 连接;

b. 重载的本地方法命名要用双下划线 __ 连接;

c. 参数签名的斜杠 “/” 改为下划线 “_” 连接,分号 “;” 改为 “_2” 连接,左方括号 “[” 改为 “_3” 连接;

另外,对于 Java 的 native 方法,static 和非 static 方法的区别在于第二个参数,static 的为 jclass,非 static 的 为 jobject;JNI 函数中是没有修饰符的。

五、在 jni 目录下创建 c 或者 c++ 文件;

文件名可以随意写,但需要注意文件类型;Hello.c 文件(.c 后缀的文件为 C)代表内容是 C 代码;Hello.cpp(.cpp 后缀的文件为 C++)文件代表内容是 C++ 代码;

C++ 代码(注意 C 和 C++ 代码是有区别),以下分别给出 C 和 C++ 两种实现方式:

  • a,Hello.c 文件。在 C 中没有引用,传递的 env 是个两级指针,用(*env)-> 调用方法且方法中要传入 env.

#include <jni.h>
#include “com_ang_ndkdemo_MainActivity.h”
JNIEXPORT jstring JNICALL
Java_com_ang_ndkdemo_MainActivity_fromJNIString(JNIEnv* env, jobject obj) {
return (*env)->NewStringUTF(env,“I am From Native C”);
}

  • b, Hello.cpp 文件。C++ 中 env 为一级指针,用 env-> 调用方法,无需传入 env;C++ 语言在编译的时候为了解决函数的多态问题,会将函数名和参数联合起来生成一个中间的函数名称,而 C 语言则不会,因此会造成链接时找不到对应函数的情况,此时 C 函数就需要用 extern “C” 进行链接指定,这告诉编译器,请保持我的名称,不要给我生成用于链接的中间函数名;exter  “C”{jni 代码}。

#include <com_ang_ndkdemo_MainActivity.h>
#include <stdio.h>

JNIEXPORT jstring JNICALL
Java_com_ang_ndkdemo_MainActivity_fromJNIString(JNIEnv *env, jobject obj)
{
return env->NewStringUTF(“I am From Native C”);
}

Java 的 native 方法是如何链接 C/C++ 中的函数的呢?可以通过静态和动态的方式注册 JNI。 以上是通过静态注册的方式。

静态注册:根据函数名建立 Java 本地方法和 JNI 函数的一一对应关系。

动态注册:直接告诉 Java native 方法其在 JNI 中对应函数的指针。

六、配置 build.gradle(Model:App)

也可以不配置 ndk{}, 这里只是指定编译出哪几种对应的 abi 架构的. so 库,如果不配置,会根据 ndk-build 默认输出对应的 abi 架构的. so 库;最好配置,不然不能编译出自己想要的对应 ABI 架构的. so,如果自己的项目中已经引用其他的. so 库还要做适配;

defaultConfig {
applicationId “com.ang.demo”
minSdkVersion 19
targetSdkVersion 28
versionCode 1
versionName “1.0”
testInstrumentationRunner “android.support.test.runner.AndroidJUnitRunner”

//ndk编译生成.so文件
ndk{
moduleName “Java2c” //生成的so名字,Android.mk文件中已经指定了,这里可以不写
abiFilters “armeabi”, “armeabi-v7a”, “x86” //输出指定三种abi体系结构下的so库。
}
}

七、编写 Android.mk 文件

Android.mk 文件一般包含如下信息就够了,差不多可算得上一个模板;根据自己的. so 库名和 C 或者 C++ 文件名修改一下就可以用了;

LOCAL_PATH:= $(call my-dir)#不用修改

include $(CLEAR_VARS)#不用修改

LOCAL_MODULE:= hello #动态库名称
LOCAL_SRC_FILES:= hello.c #C文件,里面就是我们写的C代码

include $(BUILD_SHARED_LIBRARY)#生成.so动态库

#include $(BUILD_STATIC_LIBRARY) 编译出.a的静态库

还有一种方式,就是让 androidstudio 自动生成;如下是我获取自动生成的 Android.mk 文件的方式:

a, 紧接着步骤六之后,点击 Androidstudio 菜单栏 Build ->ReBuildProject

报错:

b, 在 app ——> build ——>intermediater——>ndk(自动创建) 目录下自动创建了一个 Android.mk 文件

Android.mk 文件如下:

LOCAL_PATH := $(call my-dir)
include $(CLEAR_VARS)

LOCAL_MODULE := Java2c
LOCAL_LDFLAGS := -Wl,–build-id
LOCAL_SRC_FILES :=
D:\Demo\NDKDemo\app\src\main\jni\Hello.cpp \

LOCAL_C_INCLUDES += D:\Demo\NDKDemo\app\src\debug\jni
LOCAL_C_INCLUDES += D:\Demo\NDKDemo\app\src\main\jni

include $(BUILD_SHARED_LIBRARY)

八、修改默认编译工具

鼠标右键项目,点击 Link C++ Project with Gradle 修改 Androidstudio 默认编译工具, 在 BuildSystem 栏选择 ndk—build, 在 ProjectPath 选项栏,找到刚才创建的 Android.mk 文件(其实就是 Android.mk 文件路径),点击 OK 之后就在 build.gradle(Model:App)的 android{} 中自动生成了 externalNativeBuild { ndkBuild { path ‘src/main/jni/Android.mk’} }

//增加之后如下信息之后,右键项目的时候Link C++ Project with Gradle选项不再显示;
externalNativeBuild {
ndkBuild {
path ‘src/main/jni/Android.mk’
}
}

注意:有的时候需要再次显示 Link C++ Project with Gradle 选项,删掉 externalNativeBuild {ndkBuild {   path ‘src/main/jni/Android.mk’} } 点击 sync now 同步一下;再次右键项目就可以出现了;

**相关知识:**要将 Gradle 关联到原生库,需要提供一个指向 CMake 或 ndk-build 脚本文件的路径。在构建应用时,Gradle 会以依赖项的形式运 CMake 或 ndk-build,并将共享的. so 库打包到 APK 中。externalNativeBuild 就是配置的脚本路径;

九、最后在 MainActivity 中加载我们生成的动态库:

注意:加载生成的动态库指定的文件名(System.loadLibrary(“Java2c”);)和生成. so 时指定的名字(buil.gradle 中的 ndk{moduleName “Java2c” }),还有 Android.mk 中 LOCAL_MODULE := Java2c 三者是否一致;

例如: 下图加载生成的动态库指定的文件名为:Java2JNI 和上面生成. so 时指定的名字为:Java2c 还有 Android.mk 中 LOCAL_MODULE := Java2c 三者不一致,就会出现 UnsatisfiedLinkError 异常

public class MainActivity extends AppCompatActivity {

@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
Toast.makeText(this,new Java2C().fromJNIString(),Toast.LENGTH_LONG).show();
}
//加载.so库 Java2c为库名
static {
System.loadLibrary(“Java2c”);
}

public native String fromJNIString();

}

关于 UnsatisfiedLinkError 异常的原因还有很多,这里针对 NDK 开发总结几种可能的原因:https://blog.csdn.net/ezconn/article/details/82531893             
自我介绍一下,小编13年上海交大毕业,曾经在小公司待过,也去过华为、OPPO等大厂,18年进入阿里一直到现在。

深知大多数初中级Android工程师,想要提升技能,往往是自己摸索成长或者是报班学习,但对于培训机构动则近万的学费,着实压力不小。自己不成体系的自学效果低效又漫长,而且极易碰到天花板技术停滞不前!

因此收集整理了一份《2024年Android移动开发全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。

img

img

img

img

既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上Android开发知识点,真正体系化!

由于文件比较大,这里只是将部分目录截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且会持续更新!

如果你觉得这些内容对你有帮助,可以扫码获取!!(资料价值较高,非无偿)

《Android学习笔记总结+移动架构视频+大厂面试真题+项目实战源码》,点击传送门即可获取!

2.imgtp.com/2024/03/13/H4lCoPEF.jpg" />

[外链图片转存中…(img-Ni2f8V4E-1711538215201)]

[外链图片转存中…(img-DhfI3CYb-1711538215201)]

《Android学习笔记总结+移动架构视频+大厂面试真题+项目实战源码》,点击传送门即可获取!
  • 9
    点赞
  • 25
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值