Android Robolectric 加载运行本地 So 动态库

原文发表于:http://rocko.xyz/2016/11/27/Android-Robolectric-加载运行本地-so-动态库/

前言

Robolectric](http://robolectric.org/)) 是 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-v7amipsx86。而我们使用的 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 库之前加载这些必须的依赖即可。处理代码后面贴出。

ndk-bundle依赖库
ndk-bundle依赖库
](http://rocko-blog.qiniudn.com/2016-11-27_01-03-43_liblog.so.png?imageView2/2/w/900/h/400/q/100))

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

java.lang.UnsatisfiedLinkError: xxx.so: unknown file type, first eight 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 下找找,比如 libclibstdc++

Windows 下 Robolectric 中使用动态库

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

Sample

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

正常流程

// native-lib.cpp

#include <jni.h>#include <string>extern "C"jstringJava_xyz_rocko_rsnl_nativeinterface_NativeSample_stringFromJNI(       JNIEnv *env,       jobject /* this */) {
   // 简单返回个字符串   std::string hello = "Hello from Native.";   return env->NewStringUTF(hello.c_str());}

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

public class NativeLibsApplication extends Application { // Used to load the 'native-lib' library on application startup. static {   System.loadLibrary("native-lib"); }}

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

java.lang.UnsatisfiedLinkError: no native-lib in java.library.path

正常运行 Test case
正常运行 Test case
](http://rocko-blog.qiniudn.com/2016-11-27_09-11-08_no_deal_with.png?imageView2/2/w/900/h/400/q/100))

处理后的流程

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

public class NativeLibsApplication extends Application { @Override public void onCreate() {   super.onCreate();   loadNativeLibraries(); } /**  * 简单让子类可自己实现  */ protected void loadNativeLibraries() {
    // 代码里真正加载本地库的地方,当然你自己的可以处理地更解耦一点。   NativeLibrariesManager.loadNativeLibraries(); }}

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

test 内 libs 文件
test 内 libs 文件
](http://rocko-blog.qiniudn.com/2016-11-27_09-24-25_test_libs_folder.png))
Linux 下的我们从 ndk-bundle 里复制我们需要的 .so,然后我们自己的本地库打一个 x86-64 的即可,注意 compileSdkVersion 选上高一点支持 x86-64 的版本。

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

#!/usr/bin/env bashOUTPUT=../../../build/intermediates/dylibsmkdir -p ${OUTPUT}# .o filecc -c -I/System/Library/Frameworks/JavaVM.framework/Headers *.cpp -o ${OUTPUT}/libnative-lib.o# .dylib fileg++ -dynamiclib -undefined suppress -flat_namespace ${OUTPUT}/*.o -o ${OUTPUT}/libnative-lib.dylib

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

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

public class RobolectricApplication extends NativeLibsApplication { static {   ShadowLog.stream = System.out; //Android logcat output. } @Override protected void loadNativeLibraries() {   //Disable super class load so 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<File> soFileList = new ArrayList<>();   String systemArchPath = libsBasePath + "/framework/";   //!!! 64 位机器下处理   if (os.contains("Mac")) {     //load system 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));   } else if (os.contains("Linux")) {     //load system 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));   } else if (os.contains("Windows")) {     // ignore   }   for (File soFie : soFileList) {     System.load(soFie.getAbsolutePath());   } } private List<File> addLibs(@NonNull String path) {   File[] basePathFiles = new File(path).listFiles();   List<File> pathFilesList = new ArrayList<>();   if (basePathFiles != null && basePathFiles.length > 0) {     pathFilesList.addAll(Arrays.asList(basePathFiles));   }   return pathFilesList; }}

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

@Test public void testLoadNativeLibrariesSuccess() throws Exception {      String nativeExcepted = "Hello from Native.";      String result = NativeSample.stringFromJNI();      Log.d(TAG, "result: " + result);      assertEquals(nativeExcepted, result);}
运行加载本地库 Test case 成功
运行加载本地库 Test case 成功

End

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

项目实例源码:RobolectricSupportNativeLibs

参考

Core Java APIs and the Java Runtime on OS X

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,539评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,911评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,337评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,723评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,795评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,762评论 1 294
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,742评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,508评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,954评论 1 308
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,247评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,404评论 1 345
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,104评论 5 340
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,736评论 3 324
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,352评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,557评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,371评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,292评论 2 352

推荐阅读更多精彩内容

  • 动态调用动态库方法c/c++linuxwindows 关于动态调用动态库方法说明 一、 动态库概述 1、 动态库的...
    KINGZ1993阅读 13,910评论 0 10
  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 172,032评论 25 707
  • 动态链接,在可执行文件装载时或运行时,由操作系统的装载程序加载库。大多数操作系统将解析外部引用(比如库)作为加载过...
    小5筒阅读 5,499评论 0 3
  • 静态库与动态库的区别 首先来看什么是库,库(Library)说白了就是一段编译好的二进制代码,加上头文件就可以供别...
    吃瓜群众呀阅读 11,927评论 3 42
  • 天平再移动 终究还是天平 不会成一条线 夏天再热烈 无论有多么期待 冬天依旧遥远 绿叶依恋大树 等待亲近的一刻 不...
    水月金刀阅读 144评论 0 0