首先说一下在网上查找资料时,对于调用第三方so库,有人说有两种方法:

1.    对于so库的API符合JNI格式(即使用javah指令生成的头文件中那种格式),可以在Java代码中声明它对应的native方法,直接调  用。

jni方法名为: jstringJNICALL Java_com_example_hellojni_HelloJni_stringFromJNI(JNIEnv *,jobject);(即前缀Java+包名+类名+方法名)

java中声明的native方法名:publicnative String  stringFromJNI();


2.    对于so库的API不符合JNI格式,需要自己编写c/c++源文件,在该源文件实现自己的JNI格式native函数,在JNI函数中调用第三方so库的函数,再在java中调用自己实现的JNI格式的native方法。这种方法更加灵活。

jni函数的调用请参考我的另一篇博客,这里主要介绍第三方so库的配置、加载。


一、下图是我的项目JniDemo目录:

Android引用第三方城市组件 android 调用第三方so_源文件


   

导入的两个第三方库是:libhello.so、libhello-jni.so

自己从源文件myhello.c编译生成的库是:libmyhello.so


java中调用native方法代码如下:


1. package com.example.jnidemo;  
2.   
3. import android.app.Activity;  
4. import android.os.Bundle;  
5. import android.view.Menu;  
6. import android.view.MenuItem;  
7. import android.widget.TextView;  
8.   
9. public class DemoMain extends Activity {  
10.   
11. static {  
12. "myhello");  
13. "hello");  
14. "hello-jni");  
15.     }  
16.     TextView textView;  
17.   
18. @Override  
19. protected void onCreate(Bundle savedInstanceState) {  
20. super.onCreate(savedInstanceState);  
21.         setContentView(R.layout.activity_demo_main);  
22.   
23.         textView = (TextView) findViewById(R.id.text);  
24. "." + getJNIString();  
25.         textView.setText(str);  
26.     }  
27.   
28. public native String getString();  
29.   
30. public native String getJNIString();  
31. }


布局就一个TextView组件,不再介绍。


二、源文件的编写

在使用javah生成头文件后,要在源文件中使用include“xxxx.h”引入头文件。如果头文件不在jni根目录下,还要在Android.mk中使用

       LOCAL_C_INCLUDES:=(相对于jni目录的)包含头文件的目录路径

来声明一下,否则报错找不到头文件。

然后实现自己声明的native方法,再在其中调用第三方库的函数。


具体代码如下:

[cpp]  view plain  copy

 print ?

1. #include <string.h>  
2. #include <jni.h>  
3. #include "com_example_jnidemo_DemoMain.h"  
4. #include "com_hello_hello_HelloActivity.h"  
5. #include "com_example_hellojni_HelloJni.h"  
6.   
7. jstring Java_com_example_jnidemo_DemoMain_getString(JNIEnv* env, jobject thiz) {  
8.   
9. return Java_com_hello_hello_HelloActivity_sayHello(env, thiz);//调用libhello.so中的函数  
10. }  
11.   
12. jstring Java_com_example_jnidemo_DemoMain_getJNIString(JNIEnv* env,  
13.         jobject thiz) {  
14.   
15. return Java_com_example_hellojni_HelloJni_stringFromJNI(env, thiz);}//调用libhello-jni.so中的函数



三、Android.mk,Application.mk配置

Android.mk用来配置各个模块如何编译,如下:


[plain]  view plain  copy

 print ?

1. LOCAL_PATH := $(call my-dir) #my-dir就是该Android.mk所在目录,本项目中即jni目录  
2. include $(CLEAR_VARS)   #清楚此行之前除了LOCAL_PATH外所有的变量,因为定义的多个模块中会有相同名称的变量,目的是避免变量赋值冲突  
3.   
4. LOCAL_MODULE    := hello-jni    #指定一个当前模块名  
5. LOCAL_SRC_FILES := libhello-jni.so  #要编译的源文件  
6. include $(PREBUILT_SHARED_LIBRARY)  #编译目标,PREBUILT_表示已经编译好的,在使用NDK编译时不会再次编译,而是直接拷贝到libs目录  
7.                                                                                 #预编译.a静态库使用 PREBUILT_STATIC_LIBRARY  
8.   
9. include $(CLEAR_VARS)  
10.   
11. LOCAL_MODULE    := hello  
12. LOCAL_SRC_FILES := libhello.so  
13. include $(PREBUILT_SHARED_LIBRARY)  
14.   
15. include $(CLEAR_VARS)  
16.   
17. LOCAL_MODULE    := myhello  #自己由源文件编译库的模块名  
18. LOCAL_SRC_FILES := myhello.c    #将被编译的源文件  
19.   
20. #【重要关键点】引入依赖的第三方so库(使用模块名引入),使用\可以引入多个(注意:\符号后没有空格或其他字符)  
21. LOCAL_SHARED_LIBRARIES := \  
22. hello-jni\  
23. hello  
24. include $(BUILD_SHARED_LIBRARY) #表示编译成.so共享库,即动态库


Application.mk用来配置目标编译ABI(应用二进制接口),如arm64-v8a、armeabi、armeabi-v7a、mips、mips64、x86、x86_64。以armeabi-v7a为例,如下:


[plain]  view plain  copy

 print ?

1. APP_ABI := armeabi-v7a  #表示 编译目标 ABI(应用二进制接口)


四、在终端使用NDK编译jni目录


如果看到所有库都install到了libs目录,没有报错,就编译成功了.

补充:下面结合我遇到过的编译错误,解析一下原因及解决手段:


      so库(编译时第三方库不会再次编译,而是直接拷贝到libs中,所以一般是在编译依赖了第三方库的自己的动态库时报的错)

(每次用NDK重新编译,最好删除之前生成的编译结果so库和obj目录)

(1)报错error:undefined reference to'Java_com_example_hellojni_HelloJni_stringFromJNI'

collect2:error: ld returned 1 exit status

    网上有人说LOCAL_ALLOW_UNDEFINED_SYMBOLS:= true就可以编译过,但这是治标不治本,运行时依然报错。

错误原因:so库在生成时,如果Application.mk声明一个变量APP-ABI:=xxx,会生成不同平台下的so库,而且编译时64位平台的so库无法在32位平台上被链接,这才报了这个解决依赖链接时找不到库中方法的问题,所以虽然Android.mk中指明了是PREBUILT_SHARED_LIBRARY的so库,但不被链接还是找不到库中API的。


解决方法:获取so库时最好要取得相应版本的库(armeabi-v7a与armeabi都是32位,一般情况下应该互相兼容,但不兼容64位的arm64-v8a)。

(2)报错 error adding symbols:File in wrong format

collect2:error: ld returned 1 exit status

(ld是链接操作)


错误原因:如果Application.mk中APP-ABI:=的目标编译平台版本为64位,而实际导入的so库版本是32位,就会不识别该so库(wrong format)。


解决方法:在Application.mk(如果没有,创建)中,把APP-ABI:=xxx的目标编译版本降低点,如armeabi-v7a、armeabi这些32位等等,使之与实际导入so库匹配

整理思路:(可以在终端中,使用$file xxx.so指令查看动态库是32位还是64位。)


接下来我通过对比不同so库与编译目标ABI来进行解析:

前提准备:在自己由源文件编译的动态库中,假设依赖调用了两个第三方so库 a.so(准备了各个ABI版本)和b.so(只有版本为armeabi的)。

1. 第三方a.so库版本arm64-v8a,(不创建Application.mk)默认目标编译版本(默认是armeabi版本):

    

    报错:Fileformat not recognized

    原因:默认的目标编译版本为32位,比第三方a.so库的64位低,识别不了a.so库。

2. 第三方a.so库版本arm64-v8a,Application.mk中目标编译版本APP-ABI:= armeabi-v7a


    报错:Fileformat not recognized

    原因:目标编译版本是32位,比第三方64位的a.so库低,识别不了64位a.so库。

3.第三方a.so库版本armeabi-v7a,Application.mk中目标编译版本APP-ABI:= arm64-v8a


    报错:erroradding symbols: File in wrong format

    原因:目标编译版本是64位比32位的第三方so库高,a.so或b.so被认为文件格式错误。

4. 第三方a.so库版本armeabi-v7a,Application.mk中目标编译版本APP-ABI:= armeabi-v7a


    结果:版本匹配,NDK编译正常,armeabi-v7a兼容armeabi版本的b.so,app运行正常

5. 第三方a.so库版本armeabi,Application.mk中目标编译版本APP-ABI:= armeabi-v7a


    结果:版本匹配,NDK编译正常,armeabi-v7a与armeabi互相兼容,app运行正常

6. 第三方a.so库版本arm64-v8a,Application.mk中目标编译版本APP-ABI:= arm64-v8a


    报错:erroradding symbols: File in wrong format

    原因:目标编译版本高于b.so,所以在解决32位的b.so的依赖时报错,但64位的a.so编译正常。


总结:

    调用第三方so库时要先查看文件ABI版本,根据32或64位的相应ABI版本去定义Application.mk中目标编译版本APP-ABI。当然最好都是同一种版本,避免出现不识别、不兼容。

五、运行APP


补充:

我的demo是可以直接运行并调用第三方so库的,但在实际项目中还是遇到了loadLibrary()找不到so库的问题,报了下面的异常:


Couldn't load CloudService from loader dalvik.system.PathClassLoader[

DexPathList[[zip file "/data/app/com.example.demo-1.apk"],

nativeLibraryDirectories=[/data/app-lib/com.example.cameraframedatademo-1,

/vendor/lib,

/system/lib]]]: findLibrary returned null


解决方法: 既然是从DexPathList、nativeLibraryDirectories路径中找不到so库文件,那就手动push到对应目录下。我这里是把所有第三方库都adb push 到了设备的system/lib

目录下(如果需要且支持64位so库,请push到system/lib64目录下),这样项目调用最终会在system/lib目录下找到so库(由于push到了系统文件夹下,其他应用也可以调用

喽)。


demo项目源码下载地址


简介

Android日常的开发过程中有的项目需要引入第三方的库,有时候大家可能会在libs文件夹下看到

mips、armeabi、armeabi-v7a和x86这四个文件夹。那么这三个文件夹下面的包是干什么用的?

这三个包下面存放的用C编译的本地库文件(各类『.so』文件)。

mips、armeabi、armeabi-v7a和x86都表示CPU的类型。一般的手机或平板都是用arm的cpu。不同的cpu的特性不一样,armeabi就是针对普通的或旧的

arm v5 cpu,armeabi-v7a是针对有浮点运算或高级扩展功能的arm v7 cpu。

mips、armeabi、armeabi-v7a和x86到底是什么

mips:MIPS是世界上很流行的一种RISC处理器。MIPS的意思是“无内部互锁流水级的微处理器”(Microprocessor without interlocked piped stages),

其机制是尽量利用软件办法避免流水线中的数据相关问题。

armeabi:默认选项,将创建以基于 ARM* v5TE 的设备为目标的库。 具有这种目标的浮点运算使用软件浮点运算。 使用此 ABI (二进制接口)

创建的二进制代码将可以在所有 ARM* 设备上运行。所以armeabi通用性很强。但是速度慢

armeabi-v7a:创建支持基于 ARM* v7 的设备的库,并将使用硬件 FPU 指令。armeabi-v7a是针对有浮点运算或高级扩展功能的arm v7 cpu。

x86:支持基于硬件的浮点运算的 IA-32 指令集。x86是可以兼容armeabi平台运行的,无论是armeabi-v7a还是armeabi,同时带来的也是性能上的损耗,

另外需要指出的是,打包出的x86的so,总会比armeabi平台的体积更小。

总结

如果项目只包含了 armeabi,那么在所有Android设备都可以运行; 如果项目只包含了 armeabi-v7a,除armeabi架构的设备外都可以运行; 

如果项目只包含了 x86,那么armeabi架构和armeabi-v7a的Android设备是无法运行的; 如果同时包含了 armeabi, armeabi-v7a和x86,

所有设备都可以运行,程序在运行的时候去加载不同平台对应的so,这是较为完美的一种解决方案,同时也会导致包变大。