android 动态库 后缀,Android Robolectric加载运行本地So动态库

前言

Robolectric 是 Android 的单元测试框架,运行无需 Android 真机环境直接运行在 JVM 之上,所以在 test case 运行速度效率上有了很大提升,接近于 Java JUnit test(JUnit test > Robolectric ≫ androidTest)。不过框架本身并不支持 so 本地库的加载使用,加载时会直接报错,因为实际上运行环境是电脑机器,而我们打出的 so 文件是给手机上用的所以当然会报错。虽然在 GitHub 上很多人问过关于使用 so 的问题但基本都建议说不要在单元测试中去加载本地库,这在原则上是要这么做,但可能有些项目中做起来就有些困难了,比如在代码结构不够好、依赖耦合较大或者本身就对 so 库依赖很大的情况下。所以下面说说在项目中 Robolectric 要怎么解决需要加载运行本地 so 库这个问题。

动态库

动态库又称动态链接库(Dynamic-link library 缩写 DLL),是一个包含可由多个程序同时使用的代码和数据的库,DLL 不是可执行文件。动态链接提供了一种方法,使进程可以调用不属于其可执行代码的函数。函数的可执行代码位于一个 DLL 中,该 DLL 包含一个或多个已被编译、链接并与使用它们的进程分开存储的函数。DLL 还有助于共享数据和资源。多个应用程序可同时访问内存中单个DLL 副本的内容。DLL 是一个包含可由多个程序同时使用的代码和数据的库。Windows下动态库为 .dll 后缀(一般为 PE 格式),在 Linux 在为 .so 后缀(一般为 ELF 格式),macOS下为 .dylib 后缀(一般为 Mach-O 格式)。由于 CPU 架构和动态库文件格式的不同因而在不同平台下不能通用。其它细节的东西就不展开了因为也不会 :-)

而 Android 本身是 Linux 系统,所以用的动态库也是 .so 的文件,因而运行与 JVM 的 Robolectric 是不能直接加载使用的(Linux 某些情况下可用,下面提到)。

Robolectric 中使用动态库

我们知道动态库一般都是打给特定平台、特定 CPU 架构用的,所以要解决在 Robolectric 下加载运行 so 动态库的问题的思路就是在不同 Robolectric 运行平台下去处理加载不同的动态库,所以你要在 Ronbolectriv 中使用的 so 动态库***要有源码不然在 macOS 和 Windows 下就不就好处理了。

Note: 注意动态库名称已 lib 开头。

Linux 下 Robolectric 中使用动态库

Android 与 Linux 同气连枝,所以底层的东西很多是通用的,动态库也一样。我们 Android 使用 so 时一般也要对不同 CPU 架构的手机下使用不同的 so 文件,譬如:armeabi-v7a、mips、x86。而我们使用的 LInux 发行版一般都是 64 位的,所以原理上我们使用x86-64 的动态库是可以的,不过可能需要处理依赖库问题如果你的本地代码里有 include 其它依赖的话。如果没加进来 Robolectric 运行就会报如下的错误:

java.lang.UnsatisfiedLinkError: xxx/xxx.so xxx 动态库找不到。

xxx.so 就是你所使用 so 的依赖,比如把新浪微博 SDK 的 x86-64 的 libweibosdkcore.so 加载进来的话就会报 liblog.so 等找不到,因为 libweibosdkcore 中有对 Android liblog 等 so 库的依赖。那这个问题怎么解决呢。我们想想打包 so 库时用的是 ndk,需要使用 ndk-bundle 工具,我们想想,跟编译 apk 差不多,apk 打包需要 sdk 工具,compileSdk 里就是我们编译的依赖,里面有android.jar。所以我们可以到 ndk-bundle 里找找,***我们发现不同 CPU 架构下的 so 依赖库都是有的,像我们一般的电脑 64 位 CPU 即可使用 arch-x86_64 下的 so 动态库,所以我们只需要在加载我们程序的 so 库之前加载这些必须的依赖即可。处理代码后面贴出。

82c1a93ad163b3b29f25663851c2962f.png

注意 ndk-bundle 里的 so 也是只能在 Linux 下用的,如果用于其它平台会报错,原因前面已说明。

java.lang.UnsatisfiedLinkError: xxx.so: unknown file type,firsteight bytes: 0x7F 0x45 0x4C 0x46 0x02 0x01 0x01 0x00

macOS 下 Robolectric 中使用动态库

前面已提到,不同平台下动态链接库是不通用的,所以必须对源码重新编译打包以移植到不同平台下,如果你的 so 没有源码的话那在 macOS 和 Windows 下就行不通了。重新打包我们可以按如下两步进行:

# 先生成 .o ,-I 后加进 Java jni 的编译依赖

cc -c -I/System/Library/Frameworks/JavaVM.framework/Headers *.cpp

# 打包成 .dylib

g++ -dynamiclib -undefined suppress -flat_namespace *.o -o something.dylib

某些依赖库可以到 /usr/lib 下找找,比如 libc 和 libstdc++ 。

Windows 下 Robolectric 中使用动态库

本人没有在 Windows 下开发所以这部分就略过了,思路是一样的。

Sample

下面是简单的处理代码示例。首先新建一个包含 jni 的工程,里面写个基本的本地库,如下:

正常流程

// native-lib.cpp#include 

#include 

extern "C"

jstring

Java_xyz_rocko_rsnl_nativeinterface_NativeSample_stringFromJNI(

JNIEnv *env,

jobject /* this */) {

// 简单返回个字符串

std::string hello = "Hello from Native.";

returnenv->NewStringUTF(hello.c_str());

}

然后在 Application 启动时会加载这个本地库:

// NativeLibsApplication.java

publicclass NativeLibsApplication extends Application {

// Used toloadthe'native-lib'libraryonapplication startup.

static{

System.loadLibrary("native-lib");

}

}

此时运行 Robolectric 的 test case 就发生如下报错:

java.lang.UnsatisfiedLinkError:nonative-libinjava.library.path

859ed679c388a67b8cd9394f540ab6b9.png

处理后的流程

首先流程应该在我们的代码里避免可以直接加载 so 动态库,然后 Robolectric 在启动时自己去加载需要的动态库。

// NativeLibsApplication.java

publicclass NativeLibsApplication extends Application {

@Override publicvoid onCreate() {

super.onCreate();

loadNativeLibraries();

}

/**

* 简单让子类可自己实现

*/

protected void loadNativeLibraries() {

// 代码里真正加载本地库的地方,当然你自己的可以处理地更解耦一点。

NativeLibrariesManager.loadNativeLibraries();

}

}

然后我们的 Robolectric 里自定义自己的 Application,里面根据需要在不同运行平台下自己加载需要的本地动态库,首先复制我们给 Robolectric 用的本地库到 test 的 libs 文件夹里,按不同平台分类,如下图:

9d6ce50117ad5662066f4a443c4ca76e.png

Linux 下的我们从 ndk-bundle 里复制我们需要的 .so,然后我们自己的本地库打一个 x86-64 的即可,注意 compileSdkVersion 选上高一点支持 x86-64 的版本。

然后重新移植打出 macOS 下的动态库,简单写个打包脚本如下:

// make_macOS_dylib.sh

#!/usr/bin/env bash

OUTPUT=../../../build/intermediates/dylibs

mkdir -p ${OUTPUT}

# .o file

cc -c -I/System/Library/Frameworks/JavaVM.framework/Headers *.cpp -o ${OUTPUT}/libnative-lib.o

# .dylib file

g++ -dynamiclib -undefined suppress -flat_namespace ${OUTPUT}/*.o -o ${OUTPUT}/libnative-lib.dylib

libnative-lib.dylib 就是我们要的。

然后我们自定义 Application 处理加载这些动态库:

// RobolectricApplication.java

publicclass RobolectricApplication extends NativeLibsApplication {

static{

ShadowLog.stream = System.out; //Android logcatoutput.

}

@Override protected void loadNativeLibraries() {

//Disable super class loadso file.

//super.loadNativeLibraries();

Log.d(TAG, "=====>> Robolectric start native libraries.");

String libsBasePath =

new File(new File("").getAbsolutePath() +"/src/test/libs").getAbsolutePath();

String os = System.getProperty("os.name");

os = !TextUtils.isEmpty(os) ? os : "";

List soFileList = new ArrayList<>();

String systemArchPath = libsBasePath + "/framework/";

//!!! 64 位机器下处理

if (os.contains("Mac")) {

//loadsystem library if need

String macSysSoBasePath = systemArchPath + "macOS/";

soFileList.addAll(addLibs(macSysSoBasePath));

// App so...

String macAppSoPath = libsBasePath + "/macOS_x86-64/";

// mac下so要使用macOS专用库

soFileList.addAll(addLibs(macAppSoPath));

} elseif (os.contains("Linux")) {

//loadsystem library if need

String linuxSysSoBasePath = systemArchPath + "arch_x86-64/";

soFileList.addAll(addLibs(linuxSysSoBasePath));

// App so...

String linuxAppSoPath = libsBasePath + "/linux_x86-64/";

soFileList.addAll(addLibs(linuxAppSoPath));

} elseif (os.contains("Windows")) {

// ignore

}

for(File soFie : soFileList) {

System.load(soFie.getAbsolutePath());

}

}

private List addLibs(@NonNull String path) {

File[] basePathFiles = new File(path).listFiles();

List pathFilesList = new ArrayList<>();

if (basePathFiles != null&& basePathFiles.length > 0) {

pathFilesList.addAll(Arrays.asList(basePathFiles));

}

returnpathFilesList;

}

}

现在就可以加载了,运行如下 test case,结果如下图,成功了。

@Testpublicvoid testLoadNativeLibrariesSuccess() throws Exception {

String nativeExcepted = "Hello from Native.";

String result = NativeSample.stringFromJNI();

Log.d(TAG, "result: "+ result);

assertEquals(nativeExcepted, result);

}

2ebd1fdbc84eaebc502ba6b18faf7891.png

End

Linux 下使用最快速方便,只需要打包程序的 so 时顺便打包出 x86-64 的 so ,然后复制 ndk-bundle 的 so 加上需要的依赖即可。macOS 和 Windows 下就需要自己打包出各自平台下的动态库才可使用,如果代码里有 Android 自带 so 依赖的话那就需要自己去重新移植编译打包 ndk-bundle 里的动态库了。

参考

【编辑推荐】

【责任编辑:枯木 TEL:(010)68476606】

点赞 0

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/551560.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

java web输出语句到控制台_Java工程师(6).循环结构

搞清楚循环结构语句的执行效果与使用场景。程序中的循环流程程序中的循环流程试着编写程序实现下列功能&#xff1a;在控制台中输出100个Hello字符串。在控制台中输出从 1 到 100 的数字。… …循环结构的三要素循环结构必然包含如下三要素&#xff1a;循环变量 &#xff08;循…

sap生产工单报工_SAP系统PP (ProductionPlanning) 模块知识培训

SAP系统PP(ProductionPlanning)模块知识培训主要内容为支持生产系统规范使用SAP系统开展日常业务&#xff0c;掌握SAP系统常用操作和异常处理的方法&#xff0c;人力资源部在6月10日下午组织实施了一期《SAP系统PP模块知识》的培训。本次培训由信息技术部主管工程师解虎授课&am…

64位程序怎么判断指针是否有效_AArch64应用程序级编程模型

根据实现选择&#xff0c;体系结构支持多级执行特权&#xff0c;由从EL0到EL3的不同异常级别表示。EL0对应于最低的特权级别&#xff0c;通常被描述为无特权。应用层程序员模型是在EL0上执行软件的程序员模型。系统软件决定异常级别&#xff0c;因此决定软件运行的特权级别。当…

n个小球放入m个盒子中_飞么盒子卫生巾自助售卖机前景

嘿&#xff0c;你有没有经历过&#xff0c;出门在外来大姨妈却没带M巾呢&#xff1f;尴尬丛生的你选择怎么做&#xff1f;现在&#xff0c;有了新选择&#xff0c;这也是今天我们迫不及待想要和你们介绍的新朋友—飞么盒子&#xff0c;由Faimes飞么品牌自主研发出品的卫生巾售卖…

Android代码设置角标,Android上的Badge,快速实现给应用添加角标

应用角标是iOS的一个特点&#xff0c;原生Android并不支持。或许是由于当时iOS的通知栏比较鸡肋(固然如今已经改进了不少)&#xff0c;而Android的通知栏功能强大&#xff1f;因此才出现了一方依赖于数字角标&#xff0c;一方坚持强大的通知栏&#xff0c;在平常使用中这两种交…

opencv模糊图像变清晰_opencv-python 4.2图像模糊

图像模糊可以去除噪音。import cv2 import numpy as npimage cv2.imread("dogcat1.jpg") cv2.namedWindow("input image", cv2.WINDOW_AUTOSIZE) cv2.imshow("input image", image)#均值模糊 dst cv2.blur(image, (5, 5)) cv2.imshow("…

datatable怎么根据两列分组_公司要IT转型,我该怎么办?用Python进行数据处理

“ 以后数据处理&#xff0c;都用pandas”01 面临问题作为运维人员&#xff0c;每周要统计很多数据&#xff0c;特别是周四写周报的时候&#xff0c;基本要花半天时间。既然已经学了Python&#xff0c;那就试试优化它。以统计不同厂家、不同设备类型告警量为例。需要将多份类似…

pandas尾部添加一条_Numpy与Pandas

Numpy#导入numpy包 import numpy as np ##from numpy import * #定义数组 #一维数组 anp.array([1,2,3,4]) #二维数组 bnp.array([(1,2,3),(4,5,6)]) #定义数组类型 cnp.array([(1,2),(3,4)],dtypecomplex) #array([[ 1.0.j, 2.0.j],[ 3.0.j, 4.0.j]])#数组的维数 a.shape #(…

python爬虫获取url_Python爬虫如何获取页面内所有URL链接?本文详解

如何获取一个页面内所有URL链接&#xff1f;在Python中可以使用urllib对网页进行爬取&#xff0c;然后利用Beautiful Soup对爬取的页面进行解析&#xff0c;提取出所有的URL。什么是Beautiful Soup&#xff1f; Beautiful Soup提供一些简单的、python式的函数用来处理导航、搜索…

iview table增加一行减少一行_PQ入门函数:Table.ReplaceValue

一、基础介绍Table.ReplaceValue函数的官方说明如下&#xff1a;Table.ReplaceValue(table as table, oldValue as any, newValue as any, replacer as function, columnsToSearch as list)该函数实现的功能是将Table中的值替换成新值&#xff0c;语法翻译成汉语大致如下&#…

gprs模块ftp 远程升级_基于GPRS无线通信技术的冷链监测系统

GPRS&#xff08;general packet radio service&#xff09;是通用分组无线业务的简称&#xff0c;该技术建立在GSM网络的基础上&#xff0c;被称为2.5 代移动通信技术&#xff0c;它将无线通信与Internet 紧密结合。基于GPRS的远程数据采集系统是通过中国移动的GPRS无线通信网…

小米登录协议分析_联想前副总裁常程跳槽小米数月后,波澜再起

► 文 观察者网 吕栋今年初&#xff0c;联想集团前副总裁常程离职2天即加盟小米&#xff0c;这一举动是否违反竞业协议&#xff0c;双方当时曾各执一词。最近&#xff0c;由于联想方面在北京提起劳动仲裁&#xff0c;此事也再度引发舆论关注。9月21日&#xff0c;针对“联想与…

值从哪里来_Linux used内存到底去哪里了呢?

Linux used内存到底去哪里了呢&#xff1f;阅读文章之前请先思考这么个问题我ps aux看到的RSS内存只有不到30M&#xff0c;但是free看到内存却已经使用了7,8G了&#xff0c;已经开始swap了&#xff0c;请问ps aux的实际物理内存统计是不是漏了哪些内存没算&#xff1f;我有什么…

html文本最小长度,CSS中处理不同长度文本的几种小技巧

CSS中处理不同长度文本的几种小技巧【推荐教程&#xff1a;CSS视频教程 】当我们使用 CSS 构建布局时&#xff0c;考虑长短文本内容很重要&#xff0c;如果能清楚地知道当文本长度变化时需要怎么处理&#xff0c;可以避免很多不必要的问题。在许多情况下&#xff0c;添加或删除…

matlab 判断鼠标按下_Simulink(其他校验模块)+Matlabgui(鼠标响应事件)+Stateflow汽车运动逻辑状态(二)...

1 SimulinkSimulink-其他校验模块 如下图所示为一些其他的校验模块&#xff0c;分别为声明模块&#xff0c;离散梯度模块&#xff0c;输入分辨率检测模块&#xff1b;声明模块&#xff1a;当输入值非零时检测通过&#xff0c;当输入值中包含有0时&#xff0c;检测模块报错。…

鸿蒙电脑操作系统最新消息,5G专家预测:7年后鸿蒙将成全球第一大操作系统

在公布两年之后&#xff0c;华为的鸿蒙系统历尽千难万险&#xff0c;蓄势待发&#xff0c;即将在6月2日的线上发布会上正式发布。对于鸿蒙的前景&#xff0c;通信行业的 5G 专家项立刚在接受采访中&#xff0c;在谈到对鸿蒙的看法时&#xff0c;他表示他相信 7 年后鸿蒙会成为全…

vivado顶层模块怎么建_【第2040期】Node 模块化之争:为什么 CommonJS 和 ES Modules 无法相互协调...

前言又到周五了。今日早读文章由Shopee周雨楠翻译授权分享。周雨楠&#xff0c;Shopee金融事业群前端研发&#xff0c;自主学习前端技术3年&#xff0c;喜爱各类数字媒体技术、创意设计&#xff0c;多次参与翻译工作。福利&#xff1a;有两张门票&#xff0c;有需要的跟情封联系…

centos7 转换为lvm_(建议收藏)CentOS7挂载未分配的磁盘空间以及LVM详细介绍

简述本文主要介绍CentOS7下如何挂载未分配磁盘空间的详细操作步骤。LVMLVM&#xff0c;逻辑卷管理&#xff0c;英文全称Logical Volume Manager&#xff0c;是Linux环境下对磁盘分区进行管理的一种机制。是在硬盘分区和文件系统之间添加的一个逻辑层&#xff0c;为文件系统屏蔽…

基于python的图书管理系统测试步骤_Django admin实现图书管理系统菜鸟级教程完整实例...

Django 有着强大而又及其易用的admin后台,在这里,你可以轻松实现复杂代码实现的功能,如搜索,筛选,分页,题目可编辑,多选框. 简单到,一行代码就可以实现一个功能,而且模块之间耦合得相当完美. 不信,一起来看看吧!?用Django实现管理书籍的系统,并能在前台界面对书籍进行增删查改…

c# image转换为bitmap_Python PIL.Image与numpy.array之间的相互转换

前言有时我们使用PIL库读入图像数据后需要查看图像数据的维度&#xff0c;比如shape&#xff0c;或者有时我们需要对图像数据进行numpy类型的处理&#xff0c;所以涉及到相互转化&#xff0c;这里简单记录一下。方法当使用PIL.Image.open()打开图片后&#xff0c;如果要使用img…