Date: 2014-03-14
Title: Compile Android Native Binary And Library
Published: true
Type: post
Tags: Andoird, Develope


在Android平台上,通过NDK可以编译NativeC程序,生成原生的NativeCode。从生成的代码的使用方式来看,主要有三种形式:Native Executable Binary, Shared Dynamic Library, 以及Static Link Library。接下来将分别介绍这三种类型的二进制文件的用途和生成时的MakeFile的设置。

1. Static Link Library

1.1 用途

顾名思义,就是静态链接库的意思。静态库编译生成后是以*.a的文件形式存在;主要用于生成其他链接库或可执行程序;在生成时静态链接库的代码会被链接到目标程序中去,因此,目标程序在使用工程中无需在使用静态链接库。同时由于静态链接库的部分或全部代码被连接到目标程序中,从而使得目标程序体积变大;使用了同一静态链接库的不同程序均已经连接了各自所需的目标代码,程序间不会共享代码。

在使用时,会用到声明了静态链接库中函数的头文件。

1.2 生成方式

使用NDK再带的ndk-build生成时,要求代码需放在./jni/目录下;同时在jni下创建Android.mk文件;根据需要创建Application.mk

一个典型的用于生成静态链接库的Android.mk内容如下:

LOCAL_PATH := $(call my-dir)
include $(CLEAR_VARS)
LOCAL_MODULE := mystaticLibrary
LOCAL_SRC_FILES := mystaticLibrary.cpp
LOCAL_INCLUDES := $(LOCAL_PATH)/jni include $(BUILD_STATIC_LIBRARY)

由于生产的是静态库,所以还需要一个Application.mk的文件:

APP_MODULES     := mystaticLibrary
#APP_PLATFORM := android-8

其中声明应用平台的这一行是可选项。

之后在当前目录使用$(NDK_PATH)/ndk-build命令,正常情况即可生成相应的模块,一般位于./libs/armeabi/目录下.

2. Shared Dynamic Library

2.1 用途

动态链接库,通常不会在编译时将库中代码直接连接到目标程序中。在目标程序载入内存或使用dlopen动态加载时才会映射到进程内存;由于文件是单独存在的,故需要随其它程序或模块一起分发;当然,文件独立存在性,也使得其具备不同模块或目标程序可以共享一个共同的库文件,就是有了共享库的含义。

如前所说,存在两种使用方式:

  • 编译目标程序时声明了所使用的动态链接库,那么在目标程序启动时,动态链接库将会被载入内存;
  • 目标程序在使用时通过dlopen主动装载链接库,那么这种情况,链接库的装载时机取决于开发者。

下面简单介绍第2种情况的使用流程:

typedef bool (void * _MY_FUNC)(int)
...
//*使用`dlopen`*将库载入内存,获得句柄
void * libHandle = dlopen("mySharedLibrary.so",RTLD_LAZY);
...
//获取导出函数
_MY_FUNC myFunc = (_MY_FUNC)dlsym(libHandle, "MyTestFunc");
...
//调用导出函数
bool bRet = myFunc(2);
...
//关闭链接库
dlclose(libHandle);

2.2 生成动态链接共享库

2.2.1 通常情况的so生成

与生成静态链接库的要求一样,需要将源代码和库放在当前目录下的jni目录下,并在jni目录创建Android.mk文件。一般情况其内容如下:

include $(CLEAR_VARS)

LOCAL_MODULE        :=  myLibrary
LOCAL_C_INCLUDES := $(LOCAL_PATH)/jni
LOCAL_SRC_FILES := myLibrary.cpp include $(BUILD_SHARED_LIBRARY)

之后ndk-build即可。

2.2.2 适用于jni接口的so的生成

生成适用于jni调用的so,则需要用先用javah生成包含函数声明的头文件:javah <包名>.<类名>。例如:

javah com.my.package.myActivity

其它过程和2.2.1 通常情况下的so生成中的方法一致。

2.2.3 生成过程中静态连接其他静态库

生成so时,可能会需要链接其他一些静态库,这是Android.mk如下所示:

LOCAL_PATH := $(call my-dir)
###include myStaticLibrary lib as a prebuilt lib###
include $(CLEAR_VARS) LOCAL_MODULE := mystaticLibrary
LOCAL_SRC_FILES := ./../../mystaticLibrary/obj/local/armeabi/libmystaticLibrary.a
LOCAL_EXPORT_C_INCLUDES := $(LOCAL_PATH)/../mystaticLibrary/jni
include $(PREBUILT_STATIC_LIBRARY) ### build ndk lib###
include $(CLEAR_VARS) LOCAL_MODULE := myLib
LOCAL_C_INCLUDES := $(LOCAL_PATH)/jni \
$(LOCAL_PATH)/../../mystaticLibrary/jni
LOCAL_SRC_FILES := myLib.cpp LOCAL_LDLIBS := -llog
LOCAL_STATIC_LIBRARIES := mystaticLibrary include $(BUILD_SHARED_LIBRARY)

其中LOCAL_SRC_FILES指定的静态库需要提前已经编译好。这里也需要Application.mk

3. Native Executable Binary

这里说的就是原生的可执行文件。Android平台本身就有不少原生的可执行程序,如ps、cd等命令对应的bin。

其生产方法和so生成类似,链接静态库也类似。

### compile NDK Executable ###
LOCAL_PATH := $(call my-dir)
include $(CLEAR_VARS) LOCAL_MODULE := native_run
LOCAL_SRC_FILES := native_run.cpp
LOCAL_INCLUDE := $(LOCAL_PATH)/jni
include $(BUILD_EXECUTABLE)

参考

  1. 使用(单个)静态库生成动态库
  2. stackoverfollow, How to link any library in ndk application
  3. Android NDK about Library (static library , share library and 3rd party library)

Android NDK生成共享库和静态库的更多相关文章

  1. Android NDK生成及连接静态库与动态库

    对于Android应用开发,大部分情况下我们使用Java就能完整地实现一个应用.但是在某些情况下,我们需要借助C/C++来写JNI本地代码.比如,在使用跨平台的第三方库的时候:为了提升密集计算性能的时 ...

  2. Android:JNI与NDK(二)交叉编译与动态库,静态库

    欢迎关注公众号,第一时间获取最新文章: 本篇目录 一.前言 本篇主要以window开发环境为背景介绍一下NDK开发中需要掌握的交叉编译等基础知识,选window系统主要是照顾大多数读者,mac ,li ...

  3. 关于Android NDK中调用第三方的动态库

    因为最近在整合Android 上RTSP播放器的网络库,因需要调用自己编译的网络库,调用一直出现问题,开始时是直接在Android.mk 中加入LOCAL_SHARED_LIBRARIES := li ...

  4. gcc编译工具生成动态库和静态库

    一. 库的分类 1.1. 静态库(.a) 1.1.1. 静态库的代码在编译过程中已经被载入可执行程序,因此体积比较大.所以生成的可执行文件就不受库的影响了,即使库被删除了,程序依然可以成功运行. 1. ...

  5. VS2008 动态库和静态库的生成和加载

    第一:动态库和静态库的生成: 1) 新建一个生成dll工程: 文件->新建->项目->Win32->Win32控制台应用程序 输入项目名称:dllTest ,项目路径:D:\V ...

  6. gcc编译工具生成动态库和静态库之一----介绍

     1.库的分类 根据链接时期的不同,库又有静态库和动态库之分. 静态库是在链接阶段被链接的(好像是废话,但事实就是这样),所以生成的可执行文件就不受库的影响了,即使库被删除了,程序依然可以成功运行. ...

  7. 如何使用GCC生成动态库和静态库

    根据链接时期的不同,库又有静态库和动态库之分.静态库是在链接阶段被链接的,所以生成的可执行文件就不受库的影响,即使库被删除,程序依然可以成功运行.而动态库是在程序执行的时候被链接的.程序执行完,库仍需 ...

  8. Linux下动态库和静态库的生成和使用

    1.准备头文件和源文件 hello.h #ifndef HELLO_H #define HELLO_H void hello(const char *name): #endif hello.c #in ...

  9. 【C++】如何使用GCC生成动态库和静态库

    一.静态库和动态库的定义及区别 程序编译的四个过程: 1.预处理  展开头文件/宏替换/去掉注释/条件编译(.i后缀) 2.编译    检查语法,生成汇编(.s后缀) 3.汇编    汇编代码转换成机 ...

随机推荐

  1. spring in action学习笔记一:DI(Dependency Injection)依赖注入之CI(Constructor Injection)构造器注入

    一:这里先说一下DI(Dependency Injection)依赖注入有种表现形式:一种是CI(Constructor Injection)构造方法注入,另一种是SI(Set Injection) ...

  2. Netapp exportfs NFS Config CLI Guide

    Netapp exportfs NFS Config CLI Guide A quick and simple Netapp NFS configuration guide with commands ...

  3. Linux下目录的权限详解

    在Linux文件系统模型中,每个文件都有一组9个权限位用来控制谁能够读写和执行该文件的内容.普通文件大家都了解,这里说说目录的情况. 对于目录来说,执行位的作用是控制能否进入或者通过该目录,而不是控制 ...

  4. shell脚本——项目1

    案例名称:系统初始化 背景:10台已装有linux系统的服务器 需求: 1.设置时区同步 2.禁用selinux 3.清空防火墙策略 4.历史命令显示操作时间 5.禁止root远程登录 6.禁止定时任 ...

  5. nutch 2.1安装问题集锦

    参照官方文档http://nlp.solutions.asia/?p=180 中间碰到的问题,解决方法参考 http://blog.javachen.com/2014/05/20/nutch-intr ...

  6. 【IDEA】IDEA集成Tomcat7插件运行项目

    Maven已经是Java的项目管理标配,如何在JavaEE开发使用Maven调用Web应用,是很多同学关心的问题.本文将介绍,Maven如何介绍Tomcat插件. Maven Tomcat插件现在主要 ...

  7. v4l2驱动文档之——streaming IO【转】

    转自:http://blog.csdn.net/zoe6553/article/details/17715441 v4l2驱动编写篇第六B--流输入输出 在本系列文章的上一期中,我们讨论了如何通过re ...

  8. V4L2驱动的移植与应用(二+三)【转】

    转自:http://blog.chinaunix.net/uid-10747583-id-298489.html 原文地址:http://blog.csdn.net/wxzking/archive/2 ...

  9. UVALIVE 3972 March of the Penguins

    最大流建图比较容易第一次Dicnc抄了下别人的版 存一下以后方便查 #include <map> #include <set> #include <list> #i ...

  10. Scala学习随笔——深入类和对象

    函数化对象(又称方程化对象)指的是所定义的类或对象不包含任何可以修改的状态. 本篇随笔就是着重记录函数化对象.定义了一个有理数类定义的几个不同版本,以介绍 Scala 类定义的几个特性:类参数和构造函 ...