Android NDK开发入门2之适应idm环境

server/2025/1/12 23:35:45/

环境搭建

Android NDK开发实战之环境搭建篇(so库,Gemini ai)-CSDN博客

初始配置

前面已经运行了一个简单的初始程序,现在我们来往初始程序添加类和函数,并成功运行的实验。

一级配置

第一层配置主要是cmake文件环境和一些编译选项。

build配置 可参考: 

#build配置externalNativeBuild {cmake {path file('src/main/cpp/CMakeLists.txt')version '3.22.1'//cpp 编译选项cppFlags '-fexceptions -std=c++11'//设置c++stl为动态库 默认是静态库  arguments '-DANDROID_STL=c++_shared'}}

 

二级cmake配置

主要配置生成的动态库名称,具体看注释

为了测试 我们创建一个hello类


#include "iostream"
class hello {
public:std::string getVersion();
};

#include "hello.h"
std::string hello::getVersion(){return "v1.0 hello word";
}
#include <jni.h>
#include <string>#include "hello.h"
extern "C" JNIEXPORT jstring JNICALL
//stringFromJNI是c++function名称
Java_com_example_first_1ndk_1cpp_MainActivity_stringFromJNI(JNIEnv* env,jobject /* this */) {std::string helo = "Hello from  first native C++ ";hello h;helo += h.getVersion();return env->NewStringUTF(helo.data());
}

 更改cmake,把cpp文件加入

#cmake最低版本
cmake_minimum_required(VERSION 3.22.1)#最终项目名称 通过${CMAKE_PROJECT_NAME}获得
project("first_ndk_cpp")add_library(#生成的动态库名称${CMAKE_PROJECT_NAME}#生成库类型:SHARED动态 默认静态SHARED#相关cpp文件 假设有全局变量,交叉使用需要注意编译顺序。native-lib.cpphello.cpp)target_link_libraries(#最终生成目标库名称${CMAKE_PROJECT_NAME}# 链接其他目标库androidlog)

MainActivity

MainActivity 通常被认为是 Android 应用程序的主要入口点。

public class MainActivity extends AppCompatActivity {// 静态初始化块, 程序启动时调用static {//加载名为 first_ndk_cpp 的动态库System.loadLibrary("first_ndk_cpp");}private ActivityMainBinding binding;@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);//视图绑定binding = ActivityMainBinding.inflate(getLayoutInflater());setContentView(binding.getRoot());// Example of a call to a native methodTextView tv = binding.sampleText;tv.setText(stringFromJNI()); //java调用}/**这声明了一个名为 stringFromJNI 的本地方法。该实现位于 first_ndk_cpp 库中。*/public native String stringFromJNI();
}

运行

最后运行输出

 小结

熟悉基本配置,知道增加类代码如何重新编译成动态库使用。

动态库实战

场景:编译两份版本动态库 分别输出v1和v2,然后切换运行。

设置动态库导出路劲

前面操作,我们成功编译出动态库。但是路劲很隐蔽,其次规范处理。

实现方法

CMakeLists.txt 设置导出so路劲:

CMAKE_ARCHIVE_OUTPUT_DIRECTORY :默认存放静态库的⽂件夹位置; CMAKE_LIBRARY_OUTPUT_DIRECTORY :默认存放动态库的⽂件夹位置;

# 第一种做法:单独设置动态库的默认输出路径
# 将动态库输出到 `main/jniLibs/${ANDROID_ABI}` 目录
# ${ANDROID_ABI} 是 Gradle 配置的 ABI 策略(如 armeabi-v7a, arm64-v8a 等)
set(CMAKE_LIBRARY_OUTPUT_DIRECTORY${CMAKE_CURRENT_SOURCE_DIR}/../libs/jniLibs/${ANDROID_ABI})# 第二种做法:设置动态库和静态库的默认输出路径
# 将动态库和静态库输出到 `main/jniLibs/${ANDROID_ABI}` 目录
# 注意:LIBRARY_OUTPUT_PATH 是旧版 CMake 的变量,
#推荐使用 CMAKE_LIBRARY_OUTPUT_DIRECTORY 和 CMAKE_ARCHIVE_OUTPUT_DIRECTORY
set(LIBRARY_OUTPUT_PATH${CMAKE_CURRENT_SOURCE_DIR}/../libs/jniLibs/${ANDROID_ABI})

 

 根据这个方法步骤编译两个版本

主要区别:

std::string hello::getVersion(){return "v2.0 hello word";
}

 

尝试版本切换

//定义项目的源代码和资源文件的目录结构sourceSets {main {//指定 JNI 库文件(.so 文件)的存放目录jniLibs.srcDirs = ['src/main/libs/jniLibs']}}

 小结

学习如何指定动态库导出路劲,并根据动态库切换,实现多种版本调用。

特定cpu架构实战

上述讨论了如何导出不同的动态库,现在如果只想根据x86架构导出单个so库如何实现?

先看看为什么需要指定cpu

⽣成指定cpu平台对应的so库⽂件

应⽤⼆进制接⼝(ABI),包含的内容:

1、可执⾏⼆进制⽂件的格式以及⽀持的内容类型,⼆进制⽂件如:程序、共享库...

2、可使⽤的CPU指令集

3、运⾏时内存存储和加载的字节顺序

4、应⽤和系统之间传递数据的规范,以及系统调⽤函数时,如何使⽤堆栈、寄存器

5、如何重整C++名称

abi与 指令集

Android 支持多种 CPU 架构,每种架构对应不同的 ABI。以下是常见的 CPU 架构及其对应的 ABI:

CPU 架构ABI说明
ARMv5armeabi32 位 ARM 架构,已过时,Android NDK r17 已不再支持。
ARMv7armeabi-v7a32 位 ARM 架构,支持硬件浮点运算和 NEON 指令集。
ARMv8arm64-v8a64 位 ARM 架构,支持 AArch64 指令集。
MIPSmips32 位 MIPS 架构,已过时,Android NDK r17 已不再支持。
MIPS64mips6464 位 MIPS 架构,已过时,Android NDK r17 已不再支持。
x86x8632 位 Intel x86 架构,主要用于模拟器和部分低端设备。
x86_64x86_6464 位 Intel x86 架构,主要用于模拟器和高性能设备。

ABI 兼容性是指应用程序在不同 CPU 架构上的运行能力。以下是各 ABI 的兼容性说明:

ABI兼容性说明
armeabi兼容 ARMv5 和 ARMv7,但不兼容 ARMv8(64 位)。
armeabi-v7a兼容 ARMv7,但不兼容 ARMv5 和 ARMv8(64 位)。
arm64-v8a兼容 ARMv8(64 位),但不兼容 ARMv5 和 ARMv7(32 位)。
mips仅兼容 MIPS 架构,已过时。
mips64仅兼容 MIPS64 架构,已过时。
x86兼容 x86 架构,同时兼容 armeabi 和 armeabi-v7a(通过二进制翻译运行)。
x86_64兼容 x86_64 架构,同时兼容 arm64-v8a(通过二进制翻译运行)。

x86 和 x86_64 的特殊性

x86:主要用于模拟器和部分低端设备。通过二进制翻译,可以运行 armeabi 和 armeabi-v7a 的代码,但性能较低。

x86_64:主要用于模拟器和高性能设备。通过二进制翻译,可以运行 arm64-v8a 的代码。

ABI 配置

在 Android 项目中,可以通过 Gradle 配置支持的 ABI。

android {defaultConfig {ndk {abiFilters 'armeabi-v7a', 'arm64-v8a', 'x86', 'x86_64'}}
}

小结

ARM 架构armeabi-v7a 和 arm64-v8a 是当前主流的架构。

x86 架构:主要用于模拟器,通过二进制翻译可以运行 ARM 架构的代码。

ABI 配置:通过 Gradle 的 abiFilters 指定支持的 ABI。

问题

当遇到不能run,编译没错误信息,则进行sync now即可解决

  

学习资料分享

0voice · GitHub


http://www.ppmy.cn/server/157230.html

相关文章

预测市场平台 Aegis:Al Agent + Web3,为预测市场提供新的动力

2024 年是美国的大选年&#xff0c;并且在此期间也发生了许多值得关注的事件。我们看到&#xff0c;围绕特朗普、哈里斯的一场大选好戏&#xff0c;持续为加密市场提供了话题热度&#xff0c;同样大选也让预测市场的影响力日益凸显。事实上&#xff0c;预测市场的数据正在被采用…

Redis 管道技术(Pipeline)

1、基本概念 Redis 管道技术是一种批量处理命令的方式。在没有使用管道时&#xff0c;客户端发送一个命令&#xff0c;等待 Redis 服务器响应&#xff0c;然后再发送下一个命令。而管道允许客户端一次性发送多个命令到 Redis 服务器&#xff0c;不需要等待每个命令的响应&…

Kubernetes Gateway API-4-TCPRoute和GRPCRoute

1 TCPRoute 目前 TCP routing 还处于实验阶段。 Gateway API 被设计为与多个协议一起工作&#xff0c;TCPRoute 就是这样一个允许管理TCP流量的路由。 在这个例子中&#xff0c;我们有一个 Gateway 资源和两个 TCPRoute 资源&#xff0c;它们按照以下规则分配流量&#xff1…

怎么用java创建一个简单的项目

一个使用 Java 创建简单项目的步骤&#xff1a; 一、创建项目结构 首先&#xff0c;创建一个基本的项目结构。可以使用 IDE&#xff08;如 IntelliJ IDEA 或 Eclipse&#xff09;&#xff0c;也可以手动创建。 使用 IDE 创建项目&#xff1a; 打开 IntelliJ IDEA。选择 "C…

AWS re:Invent 2024 现场实录 - It‘s all about Scale

时隔五年&#xff0c;再度造访美国&#xff0c;也是同样的主题&#xff0c;参加在拉斯维加斯举行的 AWS re:Invent 大会。 会场 从 2012 起第一届开始&#xff0c;每年的 re:Invent 大会都放在拉斯维加斯&#xff0c;主会场也都放在威尼斯人酒店 (Venetian)。有小伙伴好奇这背…

SwiftUI 是如何改变 iOS 开发游戏规则的?

SwiftUI 是 Apple 推出的现代化声明式 UI 框架&#xff0c;适用于 iOS、macOS、watchOS 和 tvOS 开发。 SwiftUI 与传统 UIKit&#xff08;Swift 和 Objective-C&#xff09; 的优劣势对比&#xff1a; SwiftUI 的优势 一. 声明式编程 优势: SwiftUI 使用声明式语法&#xff…

“AI智能实训系统:让学习更高效、更轻松!

大家好&#xff0c;作为一名资深产品经理&#xff0c;今天我来跟大家聊聊一款备受瞩目的产品——AI智能实训系统。在这个人工智能技术飞速发展的时代&#xff0c;AI智能实训系统应运而生&#xff0c;为广大学习者提供了全新的学习体验。那么&#xff0c;这款产品究竟有哪些亮点…

网络安全 | WAF防护开通流程与技术原理详解

关注&#xff1a;CodingTechWork 引言 随着互联网安全形势的日益严峻&#xff0c;Web应用防火墙&#xff08;WAF, Web Application Firewall&#xff09;逐渐成为网站和应用的标准防护措施。WAF能够有效识别和防止如SQL注入、跨站脚本攻击&#xff08;XSS&#xff09;、恶意流…