Android NDK开发轻松入门-程序员宅基地

技术标签: 项目构建  脚本  java  NDK  jni  android 高手进阶教程  android  工具  

简介:


AndroidNDK是能使Android应用开发者把从c/c++编译而来的本地代码嵌入到应用包中的一系列工具的组合。


注意:

AndroidNDK只能用于Android1.5及以上版本中。


I. Android NDK 目标:

Android虚拟机允许你的应用在源码中通过JNI调用本地代码(c/c++)中实现的方法们。概括起来,这表示:

--你的应用的java源代码中要声明一个或多个方法,这些方法前面需有'native'关键字,这表明它们被本地代码实现。如:

  1. native byte[] loadFile(String filePath);  

--你必须提供本地的共享库(.so),库中包含这些方法的实现。这个库将会打包你的应用的.apk中。这个库的命名必须符合标准的Unix命名规则,也就是:lib<something>.so这种形式。并且还要包含一个标准的JNI入口。例如:

libFileLoader.so

--你的应用必须显式的加载本地库。例如,要在应用启动时加载,只需简单地在代码中增加如下语句:

    static{  
      
      System.loadLibrary("FileLoader");  
      
    }  

注意写库的名字时你不需要使用'lib'前缀和'.so'后缀。


AndroidNDK只是AndroidSDK的一个组件,它帮助你:

--产生JNI兼容的共享库,此库能运行于跑在ARMCPU上的Android1.5及以上系统。

--把共享库考贝到你的应用的项目中的合适的位置,并最终把它们添加到你的.apks中。

--在后续的NDK版本中,我们打算提供通过远程gdb调试本地代码的工具并尽量多地提供源码和符号信息。


AndroidNDK还提供了:

--一系列跨平台的编译工具(编译器,链接器,等等),它们可以在Linux,OS X Windows (使用Cygwin)上产生ARM上的二进制程序码。

--一系列的头文件,包含了Android系统所支持的稳定的本地API们,这保证了你所用的接口在后续所有版本中依旧被支持。

    重点注意:记住大多数系统库并没有固定死并且可能在未来的版本中会发生重大变化,甚至被删除,但是”稳定的API们”是不变的。

--一个构建系统,使得开发者只需写少量编译文件描述哪些源文件需被构建即可。构建系统可处理所有编译工具链/平台/CPU/ABI细节。还有,后续的NDK更新中可以在添加更多的编译工具链,平台,系统接口的同时不需开发者的项目构建文件发生变化。


II.Android NDK 不想做的:

使用NDK编写运行于android设备上的通用代码并不好。你的应用依然应该主要使用Java编写,这样才能正确处理Android系统的事件来避免"应用无反应"对话框的出现或处理应用的生命周期。


注意,无论如何还是可以通过本地代码编写小巧的应用的,这个应用上只是带有一个很小的用于启动/关闭应用的java包装


JNI的深刻理解是非常必要的。因为在这个环境中的很多操作需要开发者做出一些特殊的处理,而它们在典型的一般代码(java)中是不需要的。这些特殊处包括:

--不能通过指针直接使用VM对象的内容。例如,你无法安全地获取一个指向 java 16位字符数组字符串对象的的指针然后在一个循环中迭代它的每一项。

--当本地代码想要在不同的JNI调用之间保存VM对象的句柄时,需要对句柄进行明确的引用管理。


NDK仅仅为android系统所支持的本地API和库中的一小部分些提供了头文件。然而一个典型的Android系统映像包含了很多本地共享库,但这些应被看做是实现的细节,这些实现可能在平台更新或发布时发生彻底的变化。

如果一个Android系统的库没有被NDK的头文件明确支持,那么应用不应依赖于它。否则可能在下一次系统升级后出现杯具。

选中的系统库将逐渐地被添加到稳定版的NDKAPI中。


III.NDK 开发实践:

下面是一个对使用NDK开发本地代码过程的粗略的概览:

1/将你的本地代码源码放在路径$PROJECT/jni/下。

2/写一个文件:$PROJECT/jni/Android.mk,来描述你的源文件们。

3/(可选的)在文件$PROJECT/jni/Application.mk中描述你的项目的更多细节。尽管你不需要从头写,但你可以处理多CPU问题以及改写编译/链接选项。(更多细节请观docs/APPLICATION-MK.html )

4/在你的项目路径下或其任何子路径下运行"$NDK/ndk-build"来编译你的本地代码。

最后一步将在编译成功时复制你的应用所需共享库到你项目的跟路径下。然后你可以用跟以前一样的方式产生最终的.apk文件。


下面,是一些更多的细节:


III.1/配置NDK:

以前的发行版需要你运行'build/host-setup.sh'脚本来配置你的NDK。但是这一步从第4(NDK r4)开始被移除了。


III.2/放置 C C++源码:

将你的本地源码放在以下路径下:

$PROJECT/jni/

$PROJECT对应你的android应用项目的路径。

你可以随意组织jni下的内容,路径名和路径结构不会影响到最终产生的应用包。所以你无需使用类似于com.<mycompany>.<myproject>这样的名字。

注意 CC++源码都是被支持的。默认C++文件扩展名是'.cpp',但是其它的扩展名也可以被处理。(见文档docs/ANDROID-MK.html).

也可以通过调整文件Android.mk的内容,把你的源码存放在其它路径下。


III.3/写一个Android.mk构建脚本:

一个 Android.mk文件是一个很小的构建脚本。你编写它以描述你给NDK构建器的源码文件们。它的语法在docs/ANDROID-MK.html中有详细描述。


NDK简单的将你的原文件组织到多个"模块"中,每个模块可以是以下的任意一种:

-一个静态库

-一个共享库


你可以在一个Android.mk中定义多个模块,或写多个Android.mk文件,每个文件只对应一个模块。

注意,一个Android.mk文件可能被构建系统分析多遍,所以不要假设某个变量没有被定义。默认下,NDK将寻找下面的构建脚本:

$PROJECT/jni/Android.mk

如果你想在子路径下定义Android.mk文件,你应该在顶层的Android.mk中包含它们。有个函数可以做到这个功能:

include$(call all-subdir-makefiles)

这将会包含当前构建路径的所有子路径下的Android.mk文件们。


III.4/写一个Application.mk构建文件(可选):


Android.mk描述你要构建的模块们,而Application.mk文件描述你的应用自身。看文档docs/APPLICATION-MK.html来了解这个文件允许你做什么。这个文件主要包含:

-你的应用所需要模块的准确列表。

-产生的机器码所对应的CPU架构。

-可选的信息,像你要构建release还是debug,特殊的CC++编译参数以及其它需要应用到所有模块的构建选项。


这个文件是可选的:默认情况下,NDK将构建在Android.mk中列出的所有模块的并且默认面向CPUABI (armeabi).

有两种方法使用一个Application.mk:

-将它放在$PROJECT/jni/Application.mk位置,那么它会被'ndk-build'脚本自动使用。

-将它放在$NDK/apps/<name>/Application.mk$NDK代表你的NDK安装路径。之后,在NDK路径下运行"make APP=<name>"

这是在NDKr4之前的办法。出于兼容的原因,当前还是被支持的,但是我们强烈鼓励你使用第一种方法。因为它简单并且不用改动NDK安装路径下的路径树结构。


III.5/调用NDK构建系统:


使用NDK构建机器代码的最佳方式是使用'ndk-build'脚本。你也可以使用另一个老的方式--依赖于创建'$NDK/apps'子目录的方式。

不论哪种方式,编译成功后,那些编译出的“裸体”(不带有调试信息的)二进制模块都将被复制到你应用项目所在的路径下(注意非“裸体”的二进制模块会保留以提供调试能力。没有必要把非“裸体”模块复制到设备上去)


1:使用'ndk-build'命令:


'ndk-build'脚本可以在NDK安装所在的顶级目录下找到,可以直接你的应用项目的目录(也就是你的AndroidManifest.xml所在的目录)或任何子目录下调用。

例如:

cd $PROJECT

$NDK/ndk-build


这将启动NDK构建脚本,脚本将自动探测你的开发系统和应用项目文件来决定构建什么东西。

例如:

ndk-build

ndk-build clean -->清空所编译出的二进制文件们。

ndk-build -B V=1 -->强制完全重新编译,并显示命令


默认下,脚本希望看到一个可选的$PROJECT/jni/Application.mk和一个必须的$PROJECT/jni/Android.mk

成功后,会把产生的二进制模块(即共享库)复制到你的项目树中的合适位置。你可以在之后使用’ant’命令或ADP插件来重建完整的应用包。

关于这个脚本的更完整的说明和可用的选项,见docs/NDK-BUILD.html


2:使用$NDK/apps/<name>/Application.mk:


这种构建方式是NDKr4以及正前版本的唯一选择,当前还支持,仅仅是为了兼容的原因。我们强烈建议你麻溜地移植为使用'ndk-build'命令的方式,因为我们可能很快就把这种方式抛弃掉。

用它需要这样做:

1.在你的NDK安装目录下(不是你的应用路径)创建一个子目录,叫:$NDK/apps/<name>/<name>是一个任意的名字用来向NDK构建系统描述你的应用(不能有空格)

2.写一个$4NDK/apps/<name>/Application.mk,需在里面定义一个APP_PROJECT_PATH指向你的应用项目目录。

3.在命令行中,进入到NDK安装路径,然后调用顶层的GNUMakefile,如下:

cd$NDK

makeAPP=<name>

结果同第一种方法一样,除了一些中间产物放在$NDK/out/apps/<name>/下之外。


IV.重建你的应用包:

使用NDK产生二进制文件后,你需是用通常的方式重建的你应用包文件(.apk),即使用'ant'命令或ADTEclipse插件。

你的新.apk中将被嵌入共享库文件,然后在安装到设备时被系统自动分离出来。


V.对调试的支持:

NDK提供了一个辅助脚本,叫做'ndk-gdb',来十分轻松地为你的应用启动一个调试会话。


本地调试只能在运行Android2.2或更高系统的设备上执行。并且不需要特殊用户权限。

更多信息,请对docs/NDK-GDB.html。简要来说,本地调试分以下几步:

1.确定你的应用是可调试的(AndroidManifest.xml中设置android:debuggable"true")

2.使用'ndk-build'构建你的共享库,然后构建你的应用然后安装到设备或模拟器上。

3.运行你的应用。

4.在你的应用项目目录下运行'ndk-gdb'


你将看到gdb提示符出现。然后扒着GDB手册笨笨的调吧。


来源:http://blog.csdn.net/nkmnkm/article/details/7235764



版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
本文链接:https://blog.csdn.net/easyer2012/article/details/7990556

智能推荐

c# 调用c++ lib静态库_c#调用lib-程序员宅基地

文章浏览阅读2w次,点赞7次,收藏51次。四个步骤1.创建C++ Win32项目动态库dll 2.在Win32项目动态库中添加 外部依赖项 lib头文件和lib库3.导出C接口4.c#调用c++动态库开始你的表演...①创建一个空白的解决方案,在解决方案中添加 Visual C++ , Win32 项目空白解决方案的创建:添加Visual C++ , Win32 项目这......_c#调用lib

deepin/ubuntu安装苹方字体-程序员宅基地

文章浏览阅读4.6k次。苹方字体是苹果系统上的黑体,挺好看的。注重颜值的网站都会使用,例如知乎:font-family: -apple-system, BlinkMacSystemFont, Helvetica Neue, PingFang SC, Microsoft YaHei, Source Han Sans SC, Noto Sans CJK SC, W..._ubuntu pingfang

html表单常见操作汇总_html表单的处理程序有那些-程序员宅基地

文章浏览阅读159次。表单表单概述表单标签表单域按钮控件demo表单标签表单标签基本语法结构<form action="处理数据程序的url地址“ method=”get|post“ name="表单名称”></form><!--action,当提交表单时,向何处发送表单中的数据,地址可以是相对地址也可以是绝对地址--><!--method将表单中的数据传送给服务器处理,get方式直接显示在url地址中,数据可以被缓存,且长度有限制;而post方式数据隐藏传输,_html表单的处理程序有那些

PHP设置谷歌验证器(Google Authenticator)实现操作二步验证_php otp 验证器-程序员宅基地

文章浏览阅读1.2k次。使用说明:开启Google的登陆二步验证(即Google Authenticator服务)后用户登陆时需要输入额外由手机客户端生成的一次性密码。实现Google Authenticator功能需要服务器端和客户端的支持。服务器端负责密钥的生成、验证一次性密码是否正确。客户端记录密钥后生成一次性密码。下载谷歌验证类库文件放到项目合适位置(我这边放在项目Vender下面)https://github.com/PHPGangsta/GoogleAuthenticatorPHP代码示例://引入谷_php otp 验证器

【Python】matplotlib.plot画图横坐标混乱及间隔处理_matplotlib更改横轴间距-程序员宅基地

文章浏览阅读4.3k次,点赞5次,收藏11次。matplotlib.plot画图横坐标混乱及间隔处理_matplotlib更改横轴间距

docker — 容器存储_docker 保存容器-程序员宅基地

文章浏览阅读2.2k次。①Storage driver 处理各镜像层及容器层的处理细节,实现了多层数据的堆叠,为用户 提供了多层数据合并后的统一视图②所有 Storage driver 都使用可堆叠图像层和写时复制(CoW)策略③docker info 命令可查看当系统上的 storage driver主要用于测试目的,不建议用于生成环境。_docker 保存容器

随便推点

网络拓扑结构_网络拓扑csdn-程序员宅基地

文章浏览阅读834次,点赞27次,收藏13次。网络拓扑结构是指计算机网络中各组件(如计算机、服务器、打印机、路由器、交换机等设备)及其连接线路在物理布局或逻辑构型上的排列形式。这种布局不仅描述了设备间的实际物理连接方式,也决定了数据在网络中流动的路径和方式。不同的网络拓扑结构影响着网络的性能、可靠性、可扩展性及管理维护的难易程度。_网络拓扑csdn

JS重写Date函数,兼容IOS系统_date.prototype 将所有 ios-程序员宅基地

文章浏览阅读1.8k次,点赞5次,收藏8次。IOS系统Date的坑要创建一个指定时间的new Date对象时,通常的做法是:new Date("2020-09-21 11:11:00")这行代码在 PC 端和安卓端都是正常的,而在 iOS 端则会提示 Invalid Date 无效日期。在IOS年月日中间的横岗许换成斜杠,也就是new Date("2020/09/21 11:11:00")通常为了兼容IOS的这个坑,需要做一些额外的特殊处理,笔者在开发的时候经常会忘了兼容IOS系统。所以就想试着重写Date函数,一劳永逸,避免每次ne_date.prototype 将所有 ios

如何将EXCEL表导入plsql数据库中-程序员宅基地

文章浏览阅读5.3k次。方法一:用PLSQL Developer工具。 1 在PLSQL Developer的sql window里输入select * from test for update; 2 按F8执行 3 打开锁, 再按一下加号. 鼠标点到第一列的列头,使全列成选中状态,然后粘贴,最后commit提交即可。(前提..._excel导入pl/sql

Git常用命令速查手册-程序员宅基地

文章浏览阅读83次。Git常用命令速查手册1、初始化仓库git init2、将文件添加到仓库git add 文件名 # 将工作区的某个文件添加到暂存区 git add -u # 添加所有被tracked文件中被修改或删除的文件信息到暂存区,不处理untracked的文件git add -A # 添加所有被tracked文件中被修改或删除的文件信息到暂存区,包括untracked的文件...

分享119个ASP.NET源码总有一个是你想要的_千博二手车源码v2023 build 1120-程序员宅基地

文章浏览阅读202次。分享119个ASP.NET源码总有一个是你想要的_千博二手车源码v2023 build 1120

【C++缺省函数】 空类默认产生的6个类成员函数_空类默认产生哪些类成员函数-程序员宅基地

文章浏览阅读1.8k次。版权声明:转载请注明出处 http://blog.csdn.net/irean_lau。目录(?)[+]1、缺省构造函数。2、缺省拷贝构造函数。3、 缺省析构函数。4、缺省赋值运算符。5、缺省取址运算符。6、 缺省取址运算符 const。[cpp] view plain copy_空类默认产生哪些类成员函数

推荐文章

热门文章

相关标签