当前位置: 首页 > news >正文

Android Jni(二)加载调用第三方 so 库

文章目录

      • Android Jni(二)加载调用第三方 so 库
      • 前置知识
        • CPU架构 ABI
      • 基本步骤
        • 1、将第三方 SO 库文件放入项目中的正确位置:
        • 2. 创建 JNI 接口
        • 3. 实现 JNI 层代码
        • 4、配置 CMake
      • 常见问题解决
        • 1、UnsatisfiedLinkError:
        • 2、函数找不到:
        • 3、ABI 不匹配:
      • 遇到的问题
        • 1、include 找不到头文件
        • 2、Jni 不同库的引入了重复 so,导致冲突
        • 3、多个本地库
        • 4、return NewStringUTF(char*) 报格式错误
        • 5、指针类型不一致报错
        • 6、Bitmap 转 yuv 反色
        • 7、接入的三方库头文件依赖了其他头文件找不到
      • 参考文章

Android Jni(二)加载调用第三方 so 库

前置知识

CPU架构 ABI

接入第三方 so 库时需要注意目标设备是否支持,不然会找不到 so 库

abiFilters是用于指定在构建Android应用程序时应包含哪些CPU架构ABI(Application Binary Interface)的一种配置参数。它的常见取值包括"armeabi-v7a"、“arm64-v8a”、“x86”、"x86_64"等,具体取决于应用程序要支持的目标设备的CPU架构。在构建Gradle/Android项目时,可以通过在build.gradle配置文件中设置abiFilters来指定所需的CPU架构ABI。

基本步骤

1、将第三方 SO 库文件放入项目中的正确位置:
app/src/main/jniLibs/armeabi-v7a/    // 32位 ARMlibthirdparty.soarm64-v8a/      // 64位 ARMlibthirdparty.sox86/            // x86libthirdparty.so

或者在 build.gradle 中指定库的位置:

android {sourceSets {main {jniLibs.srcDirs = ['libs']}}
}
2. 创建 JNI 接口
public class NativeWrapper {static {// 先加载依赖库(如果有)System.loadLibrary("dependency");// 然后加载目标库System.loadLibrary("thirdparty");// 最后加载你自己的JNI库System.loadLibrary("mylibrary");}// 声明native方法public native int callThirdPartyFunction(int param);
}
3. 实现 JNI 层代码

创建 jni/mylibrary.c 文件:

#include <jni.h>
#include <android/log.h>// 声明第三方库的函数
extern int third_party_function(int param);JNIEXPORT jint JNICALL
Java_com_example_NativeWrapper_callThirdPartyFunction(JNIEnv *env, jobject instance, jint param) {// 调用第三方库函数int result = third_party_function(param);return (jint)result;
}
4、配置 CMake
add_library(thirdparty SHARED IMPORTED)
set_target_properties(thirdparty PROPERTIES IMPORTED_LOCATION${CMAKE_SOURCE_DIR}/../jniLibs/${ANDROID_ABI}/libthirdparty.so)add_library(mylibrary SHAREDmylibrary.c)target_link_libraries(mylibrary thirdparty)

常见问题解决

1、UnsatisfiedLinkError:

检查库文件名是否正确(前缀 lib,后缀 .so)
检查库是否放在正确的 ABI 目录下
检查是否有依赖库未加载

2、函数找不到:

使用 nm -D libthirdparty.so 检查导出的函数名

可能需要 extern “C” 包装

3、ABI 不匹配:

确保应用和所有库使用相同的 ABI

64位设备可以运行32位库,但反过来不行

遇到的问题

1、include 找不到头文件

原因: include_directories 设置的路径不对
解决方法:根据自己项目实际情况设置路径
在这里插入图片描述

2、Jni 不同库的引入了重复 so,导致冲突

确保项目中只包含一个libc++_shared.so版本。可以通过在项目的build.gradle文件中配置packagingOptions来选择第一个找到的libc++_shared.so文件,使用pickFirst策略5。

3、多个本地库

CMakeLists.txt 额外配置 target_link_libraries

target_link_libraries(local_lib1# List libraries link to the target libraryandroidlog)target_link_libraries(local_lib2# List libraries link to the target libraryandroidlog)
4、return NewStringUTF(char*) 报格式错误
对数组进行初始化赋值
char a[10] = {""};
5、指针类型不一致报错

严格按照 api 文档传入指针

6、Bitmap 转 yuv 反色

问题原因是如下代码中,g b 数据通道,赋值反了,如何保存的 ARGB_8888 注释中有写。
在这里插入图片描述

#include <android/bitmap.h>
#include <jni.h>
#include <cstdint>
#include <cstring>// ARGB 转 NV12 的函数
void ARGB_to_NV12(jint *argb, jbyte *nv12, jint width, jint height) {int frameSize = width * height;int yIndex = 0;int uvIndex = frameSize;for (int j = 0; j < height; ++j) {for (int i = 0; i < width; ++i) {int R = (argb[(j * width) + i] >> 16) & 0xFF;int G = (argb[(j * width) + i] >> 8) & 0xFF;int B = argb[(j * width) + i] & 0xFF;// 计算 YUV 分量int Y = ((66 * R + 129 * G + 25 * B + 128) >> 8) + 16;int U = ((-38 * R - 74 * G + 112 * B + 128) >> 8) + 128;int V = ((112 * R - 94 * G - 18 * B + 128) >> 8) + 128;// 存储 Y 分量nv12[yIndex++] = static_cast<jbyte>((Y < 0) ? 0 : ((Y > 255) ? 255 : Y));// 存储 UV 分量(交错存储)if (j % 2 == 0 && i % 2 == 0) {nv12[uvIndex++] = static_cast<jbyte>((U < 0) ? 0 : ((U > 255) ? 255 : U));nv12[uvIndex++] = static_cast<jbyte>((V < 0) ? 0 : ((V > 255) ? 255 : V));}}}
}// JNI 函数
extern "C"
JNIEXPORT jbyteArray JNICALL
Java_com_example_YourClass_convertBitmapToNV12(JNIEnv *env, jobject thiz, jobject bitmap) {AndroidBitmapInfo info;void *pixels;jbyteArray nv12Array = nullptr;// 获取 Bitmap 信息if (AndroidBitmap_getInfo(env, bitmap, &info) != ANDROID_BITMAP_RESULT_SUCCESS) {return nullptr;}// 检查 Bitmap 格式是否为 RGBA_8888if (info.format != ANDROID_BITMAP_FORMAT_RGBA_8888) {return nullptr;}// 锁定 Bitmap 像素数据if (AndroidBitmap_lockPixels(env, bitmap, &pixels) != ANDROID_BITMAP_RESULT_SUCCESS) {return nullptr;}int width = info.width;int height = info.height;int frameSize = width * height;// 创建 NV12 数组(Y 分量占 width * height,UV 分量占 width * height / 2)nv12Array = env->NewByteArray(frameSize * 3 / 2);jbyte *nv12 = env->GetByteArrayElements(nv12Array, nullptr);// 将 ARGB 转换为 NV12ARGB_to_NV12(static_cast<jint *>(pixels), nv12, width, height);// 释放资源env->ReleaseByteArrayElements(nv12Array, nv12, 0);AndroidBitmap_unlockPixels(env, bitmap);return nv12Array;
}
7、接入的三方库头文件依赖了其他头文件找不到

算法给的头文件,可能会包含一些你不需要的代码,之间删除即可,不影响调用。

参考文章

Android jni引用第三方so动态库和.a静态库并且调用©方法

Android JNI学习-调用第三方SO库

Android 通过JNI调用三方so 高效教程

cmake使用详细教程(日常使用这一篇就足够了)

相关文章:

  • portainer.io篇
  • 【Qt】信号和槽
  • Zookeeper三台服务器三节点集群部署(docker-compose方式)
  • Zookeeper单机三节点集群部署(docker-compose方式)
  • Java EE(8)——线程安全总结(翻新版)——定时器(Timer)线程池(ThreadPoolExecutor)
  • LeetCode算法题(Go语言实现)_48
  • 智能做题:拍照解题API使用指南
  • AI对百度搜索与抖音社区的影响差异?
  • 【Python语言基础】20、模块与包
  • 用 Vue 3 + OpenAI API 实现一个智能对话助手(支持上下文、多角色)
  • fastjson2 使用bug
  • 每日算法-链表(2.两数相加、24.两两交换链表中的节点、143.重排链表)
  • 怎么通过OPPO手机进行图片编辑?图片编辑攻略,打造专业级照片
  • vscode格式化为什么失效?自动保存和格式化(Prettier - Code formatter,vue-format)
  • MySQL入门:数据表的创建
  • HTTP HTTPS RSA
  • AI核心概念之“结构化输出(Structured Output)” - 来自DeepSeek
  • Terraform - 理解 Count 和 For_Each 循环
  • 力扣 283 移动零的两种高效解法详解
  • 深度学习--神经网络的构造
  • 这场迪图瓦纪念拉威尔的音乐会,必将成为乐迷反复品味的回忆
  • 乌方称泽连斯基与特朗普进行简短会谈
  • 光线传媒:正与部分重点地区洽谈主题乐园合作,首款3A游戏预计三年左右推出
  • 著名茶叶专家谢丰镐逝世,享年95岁
  • 东部战区新闻发言人就美“劳伦斯”号导弹驱逐舰过航台湾海峡发表谈话
  • 湖南永州公安全面推行“项目警官制”,为重点项目建设护航