研究了N久+N次,终于在这一周解决了,感谢度娘。感觉第一次快要跑通时,心里真是万分紧张,感觉什么都不会再爱了。点下按钮,返回预期的值,OK。搞定,为此,放松了一个上午,现在来写写一些我是怎么搞通的吧。不敢保证,每个库都能这么搞,但是对一些刚入门者,应该有帮助。好的,开讲!
Android.mk文件可以说是本文最终要的一部分。我们先一步一步去讲一讲它。
首先阅读下这篇文章,预热一下,了解了解打包静态库和动态库的方法。
其次我们从NDK的samples的two-libs例子入手,分析mk的写法。(即搭建Android环境时,解压的NDK目录下的samples/two-libs)
先上代码
1)first文件:定义了一个加法的方法
//first.h #ifndef FIRST_H #define FIRST_H extern int first(int x, int y); #endif /* FIRST_H */ //first.c #include "first.h" int first(int x, int y) { return x + y; }2)second.c:这个文件是Jni的相关的文件。
//second.c #include "first.h" #include <jni.h> //添加jni头文件 //下面这个函数的意思就是com.example.twolibs包com.example.androidlibcurldemo下MainActivity.java中的add方法。 jint Java_com_example_androidlibcurldemo_MainActivity_add( JNIEnv* env, jobject this, jint x, jint y ) { return first(x, y); }这个add方法,是在我们上一步创建工程的src目录下的MainActivity.java中声明的两个native方法,说白了,second.c的作用就是,定义java中的native方法,并在实现中调用C的代码,如first.c中的first函数。相当于一个中介的样子。
#定义变量LOCAL_PATH指向mk所在的路径,如Android.mk文件在/usr/lib/curl/Android.mk #那么LOCAL_PATH代表的值就是:/usr/lib/curl #注意,这个变量没有清理,如果在后面引入其他mk文件,这个变量是可以传递到引入的mk文件中的 LOCAL_PATH:= $(call my-dir) # first lib, which will be built statically #CLEAR_VARS清理变量,因为这些变量都是静态全局的,如果不清了,下次编译时用到这些变量就会造成错误的 include $(CLEAR_VARS) #本地静态库模块名字,如果在别处导入库文件时,会用到这个名字 #此处名字的使用是libXXX.so或libXXX.a中的XXX的名字,也可以libXXX,不过,这两种在使用上还是有点区别的, #使用XXX或libXXX最后打包库后都是libXXX.a/so,在导入库模块时要使用LOCAL_LDLIBS := -lz来指明导入的是本地库 #否则会报找不到库的错误 LOCAL_MODULE := libtwolib-first LOCAL_SRC_FILES := first.c #创建静态库,注意,这个库是我们第一次创建,所以使用BUILD_STATIC_LIBRARY #如果LOCAL_SRC_FILES(即源码文件)使用的是.a则使用include $(PREBUILT_STATIC_LIBRARY),见下面代码 #如果使用的是.so文件,则使用include $(PREBUILT_SHARED_LIBRARY),PREBUILT表示使用的是预编译库里的源文件 include $(BUILD_STATIC_LIBRARY) # second lib, which will depend on and include the first one #清理变量 include $(CLEAR_VARS) #模块名称 LOCAL_MODULE := libtwolib-second #源码文件 LOCAL_SRC_FILES := second.c #导入关联的静态库,如果还有其它静态库,直接在后面添加即可,注意空格隔开 LOCAL_STATIC_LIBRARIES := libtwolib-first #最终创建动态库文件,libtwolib-second.so #如果最终创建静态库文件,是需要把SHARED改为STATIC即可,如libtwolib-first include $(BUILD_SHARED_LIBRARY)
4)Application.mk
我们在这个mk文件中只添加APP_ABI := all NDK就会为我们自动打包对应不同平台ABI的库文件,不同平台的库文件还是有区别的。
不过现在的Android手机大部分都是armeabi的,所以,如果不添加这句话,NDK默认的就会生成armeabi目录,并编译这个平台下使用的库文件。
5)编译库文件我们进入到创建的Android工程,根目录即可,打开终端,如我使用的mac,打开终端后输入:
$ cd /Users/yuxikuo/Android_WorkPlace/AndroidLibcurlDemo
$ ndk-buildAndroid NDK: WARNING: APP_PLATFORM android-19 is larger than android:minSdkVersion 8 in ./AndroidManifest.xml [armeabi] Compile thumb : twolib-second <= second.c [armeabi] Compile thumb : twolib-first <= first.c [armeabi] StaticLibrary : libtwolib-first.a [armeabi] SharedLibrary : libtwolib-second.so [armeabi] Install : libtwolib-second.so => libs/armeabi/libtwolib-second.so打包后的结果如上:Android中NDK将打包后的包文件拷贝到1)工程目录下的libs/下对应ABI目录下面,ABI表示当前机器的操作系统,一共有4类,不同的目标系统ABI有不同的打包方式和兼容性。2)同时还会在工程目录新建obj/local目录,该目录下同样根据不同的ABI拷贝对应的库文件。不过,不同的ABI目录是需要自己新建的,默认下是armeabi,因为大部分Android使用的是这个,并且它可以运行在所有ARM CPU上。
armeabi => ARMv5TE以上
armeabi-v7a => ARMv7以上
x86 => x86平台
mips => mips
由于根据不同的ABI进行打包,那么我们每次都会生成4个包,打包过程见下面代码。
而且现在其他的平台并不常见,而且armeabi对其他平台也有兼容,所以,我们这里暂时只考虑armeabi平台下库文件的打包。
Android NDK: WARNING: APP_PLATFORM android-19 is larger than android:minSdkVersion 8 in ./AndroidManifest.xml [armeabi-v7a] Compile thumb : twolib-second <= second.c [armeabi-v7a] Compile thumb : twolib-first <= first.c [armeabi-v7a] StaticLibrary : libtwolib-first.a [armeabi-v7a] SharedLibrary : libtwolib-second.so [armeabi-v7a] Install : libtwolib-second.so => libs/armeabi-v7a/libtwolib-second.so //armeabi-v7a [armeabi] Compile thumb : twolib-second <= second.c [armeabi] Compile thumb : twolib-first <= first.c [armeabi] StaticLibrary : libtwolib-first.a [armeabi] SharedLibrary : libtwolib-second.so [armeabi] Install : libtwolib-second.so => libs/armeabi/libtwolib-second.so //armeabi [x86] Compile : twolib-second <= second.c [x86] Compile : twolib-first <= first.c [x86] StaticLibrary : libtwolib-first.a [x86] SharedLibrary : libtwolib-second.so [x86] Install : libtwolib-second.so => libs/x86/libtwolib-second.so //x86 [mips] Compile : twolib-second <= second.c [mips] Compile : twolib-first <= first.c [mips] StaticLibrary : libtwolib-first.a [mips] SharedLibrary : libtwolib-second.so [mips] Install : libtwolib-second.so => libs/mips/libtwolib-second.so //mips6)在Java代码中调用。
在MainActivity.java中添加如下代码
@Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); //super.onCreate(savedInstanceState); TextView tv = new TextView(this); int x = 1000; int y = 42; // here, we dynamically load the library at runtime // before calling the native method. //把生成的库load进来,注意去掉libXXX.so中的lib和so。只保留XXX System.loadLibrary("twolib-second"); int z = add(x, y); tv.setText( "The sum of " + x + " and " + y + " is " + z ); setContentView(tv); } public native int add(int x, int y);//native函数声明,在我们的second.c文件中定义,并使用了first.c中的first函数,实现加法注意,如果出现错误或崩溃,看看second.c的函数包名是不是错误,然后看看有没有在java中声明native的add方法
将Cocos2d-x的libcurl单独打包到Android
原文地址:http://blog.csdn.net/yuxikuo_1/article/details/41746869