【转】Android下编译jni库的二种方法(含示例)
原文网址:http://blog.sina.com.cn/s/blog_3e3fcadd01011384.html
总结如下:两种方法是:
1)使用Android源码中的Make系统
2)使用NDK(从NDK r5开始)
---------------------------------
源码要求: 如果Android工程的顶层目录为my_android_project_name:
1) my_android_project_name/jni/Android.mk文件的示例内容如下(以生成libjni_mouse.so的C代码为例):
LOCAL_PATH:= $(call my-dir)
include $(CLEAR_VARS)
LOCAL_PRELINK_MODULE := false
LOCAL_MODULE := libjni_mouse
LOCAL_SRC_FILES := jni_mouse.c
LOCAL_C_INCLUDES := $(JNI_H_INCLUDE)
LOCAL_LDLIBS += -L$(SYSROOT)/usr/lib -llog
LOCAL_SHARED_LIBRARIES := libcutils
include $(BUILD_SHARED_LIBRARY)
2)源文件的部分内容(仅供参考):
#include <stdio.h>
#include <unistd.h>
#include <errno.h>
#include <linux/input.h>
#include <linux/uinput.h>
#include <jni.h> // located in $JDK/include
#include <cutils/log.h> // for Android Make system: $Android_SRC/system/core/include/cutils/log.h
//#include <android/log.h> // for NDK compiler
#define LOG_DBG(tag,fmt,args...) __android_log_print(ANDROID_LOG_DEBUG,tag,fmt,##args)
#define LOG_ERR(tag,fmt,args...) __android_log_print(ANDROID_LOG_ERROR,tag,fmt,##args)
#ifdef ___DEBUG
#define DBG(fmt, args...) __android_log_print(ANDROID_LOG_DEBUG, TAG, fmt, ##args)
#else
#define DBG(fmt, args...)
#endif //___DEBUG
#define OUT(fmt, args...) __android_log_print(ANDROID_LOG_ERROR, TAG, fmt, ##args)
#define JNIREG_CLASS "com/my_organization/my_pkg/JniMouse"
static JNINativeMethod JniMouseMethods[] = {
{ "nativeOpenMouse", "()I", (void*) openMouse },
{ "nativeCloseMouse", "()V", (void*) closeMouse },
{ "nativeMoveCursor", "(II)V", (void*) moveCursor },
...
};
static int registerNativeMethods(JNIEnv* env, const char* className, JNINativeMethod* methods,
int numMethods)
{
jclass clazz;
// C++ way: clazz= env->FindClass(className);
clazz = (*env)->FindClass(env, className);
if (clazz == NULL) {
LOG_ERR(TAG, "Native registration error: unable to find class\n");
return JNI_FALSE;
}
// C++ way: if (env->RegisterNatives(clazz, methods, numMethods) < 0)
if ((*env)->RegisterNatives(env, clazz, methods, numMethods) < 0) {
LOG_ERR(TAG, "RegisterNatives failed\n");
return JNI_FALSE;
}
return JNI_TRUE;
}
static int registerNatives(JNIEnv* env)
{
if (!registerNativeMethods(env, JNIREG_CLASS, JniMouseMethods,
sizeof(JniMouseMethods) / sizeof(JniMouseMethods[0])))
return JNI_FALSE;
return JNI_TRUE;
}
jint JNI_OnLoad(JavaVM* vm, void* reserved)
{
JNIEnv* env = NULL;
jint result = -1;
// this is for C++: if (vm->GetEnv((void**) &env, JNI_VERSION_1_4) != JNI_OK)
if ((*vm)->GetEnv(vm, (void**) &env, JNI_VERSION_1_4) != JNI_OK) {
LOG_ERR(TAG, "ERROR: GetEnv failed\n");
goto bail;
}
assert(env != NULL);
if (!registerNatives(env)) {
LOG_ERR(TAG, "ERROR: jni_mouse native registration failed\n");
goto bail;
}
result = JNI_VERSION_1_4;
bail:
return result;
}
---------------------------------
编译过程具体如下:
1)使用Android源码中的Make系统:
$ cd $ANDROID_SRC # (e.g. /u_data/android/sourcecode/and22)
$ . build/envsetup.sh
# 这样就会生成许多shell工具,实际上是shell函数。
$ lunch MY_LUNCH_NUMBER # 如果你的项目需要用到PLATFORM相关的内容(共享库/java库等)
# 把Android工程的顶层目录符号链接(或者拷贝)到 $ANDROID_SRC/packages/apps,如
$ ln -fs /abs/path/to/my_android_project_name $ANDROID_SRC/packages/apps
$ mmm packages/apps/my_android_project_name
# 如果没有错误,就会生成以下文件(取决于你使用的PRODUCT名字):
# $ANDROID_SRC/out/target/product/$PRODUCT_NAME/system/lib/libjni_mouse.so
# 最后拷贝生成的库文件到Android工程的libs/armeabi/目录下:
$ mkdir -p $ANDROID_SRC/packages/apps/my_android_project_name/libs/armeabi
$ cp $ANDROID_SRC/out/target/product/$PRODUCT_NAME/system/lib/libjni_mouse.so \
$ANDROID_SRC/packages/apps/my_android_project_name/libs/armeabi
# 这样,在eclipse中正常编译/生成.apk文件时,就会把生成的jni库自动包含进.apk文件中,这可以使用unzip命令来验证:
# cp my.apk /tmp/tmp/my.zip; cd /tmp/tmp; unzip my.zip; ls -l lib/armeabi/lib*.*
2)使用NDK(from NDK r5)
从NDK r5开始,NDK正式作为独立的工具来编译jni的代码,但它往往缺少某些头文件,如<linux/uinput.h>。
0)准备环境变量,如:
$ export NDK=/u_data/android/sourcecode/android-ndk-r6b
$ export CC=$NDK/toolchains/arm-linux-androideabi-4.4.3/prebuilt/linux-x86/bin/arm-linux-androideabi-gcc
$ export STRIP=$NDK/toolchains/arm-linux-androideabi-4.4.3/prebuilt/linux-x86/bin/arm-linux-androideabi-strip
$ export SYSROOT=$NDK/platforms/android-8/arch-arm # 要看你使用的目标Android的版本 :)
1) 对于缺少的头文件,可以借用PC上的x86下的同名文件,如:
cd $NDK/platforms/android-8/arch-arm/usr/include/linux
ln -fs ../../../../arch-x86/usr/include/linux/uinput.h .
2) 编译出可执行文件(如果有main函数),如:
$ $CC --sysroot=$SYSROOT -o my_jni_mouse_exe jni_mouse.c my_main_func.c
#注: '--sysroot=dir':Use dir as the logical root directory for headers and libraries.
3) 编译出共享库(如果没有main函数),如:
$ $CC --sysroot=$SYSROOT -fPIC -o jni_mouse.o -c jni_mouse.c # optional '-o ...'
$ $CC --sysroot=$SYSROOT -shared -o libjni_mouse.so jni_mouse.o
# 再strip一下,以便减少其大小(直接覆盖原来的库文件):
$ $STRIP libjni_mouse.so
4)(可选)链接如上刚生成的共享库:
$ $CC --sysroot=$SYSROOT -L. -ljni_mouse -o my_hello my_hello.c
<EOF>
【转】Android下编译jni库的二种方法(含示例)的更多相关文章
- 【转】Android下编译jni库的二种方法(含示例) -- 不错
原文网址:http://blog.sina.com.cn/s/blog_3e3fcadd01011384.html 总结如下:两种方法是:1)使用Android源码中的Make系统2)使用NDK(从N ...
- Android Studio导入第三方库的三种方法
叨叨在前 今天在项目中使用一个图片选择器的第三方框架——GalleryFinal,想要导入源码,以便于修改,于是上完查找了一下方法,想到之前用到过其他导入第三方库的方法,现在做个小总结,以防忘记. A ...
- 在Android源码下编译jni所需要知道的事~
以下只是自己的一些总结,欢迎讨论 通过NDK编译jni网上有很多例子,在这我只总结在Android源码下编译 1.android源码环境下编译so包,编出来的.so的包前面不会自动给添加lib,NDK ...
- 手把手图文并茂教你用Android Studio编译FFmpeg库并移植
转载请把头部出处链接和尾部二维码一起转载,本文出自逆流的鱼yuiop:http://blog.csdn.net/hejjunlin/article/details/52661331 之前曾写过一篇&l ...
- android ndk-build 编译静态库libxx.a 以及Android studio openssl 静态库配置(cmake)
android ndk-build 编译静态库libxx.a 需求场景: 目前有安卓编码好的现在的openssl的两个.a,我们需要调用openssl的函数,并把功能再封装成.a; 这样使用时,在an ...
- Android下查看共享库依赖项
Android下查看共享库依赖项 [时间:2017-02] [状态:Open] [关键词:android,共享库依赖项,so,ndk,objdump,readelf] 起因 近期在处理Android下 ...
- 在Ubuntu下编译Assimp库
在Ubuntu下编译Assimp库 怎样在Ubuntu下编译Assimp库?这是我曾经编译成功后做的笔记,供參考. 1.去以下的站点去下载Assimp库: http://assimp.sourcefo ...
- NDK下编译JNI
NDK环境下编译JNI 下载demo.tar.gz然后解压 弄个套路 1.编辑build.sh设置好NDK目录 2.把cpp文件放到code下面 运行sh build.sh即可
- 痞子衡嵌入式:MCUXpresso IDE下设置代码编译优化等级的几种方法
大家好,我是痞子衡,是正经搞技术的痞子.今天痞子衡给大家分享的是MCUXpresso IDE下设置代码编译优化等级的几种方法. 最近公司芯片设计团队正在开发一款全新的基于 Cortex-M33 内核的 ...
随机推荐
- (Data structure)Implement Trie && Add and Search Word
Implement Trie (Prefix Tree) Implement a trie with insert, search, and startsWith methods. Note:You ...
- 【hihoCoder第十七周】最近公共祖先·三
之前就写的是离线算法.思路就是先序一遍树,记录层数,然后高效RMQ就好.ST和线段树都能过. 以后有时间将之前的在线算法补上. #include <bits/stdc++.h> using ...
- global.asax?app.config?webconfig??
一.Global.asax 1.global.asax是什么? 一个文本文件,至于它包含写什么内容?顾名思义,global 肯定是掌管一个应用程序(application)的全局性的东西,例如应用程序 ...
- Quartz定时调度CronTrigger时间配置格式说明与实例
1. CronTrigger时间格式配置说明 CronTrigger配置格式: 格式: [秒] [分] [小时] [日] [月] [周] [年] 序号 说明 是否必填 允许填写的值 允许的通配符 ...
- JAVA获取oracle中sequences的最后一个值
项目中,用到一个序列作单号,框架用的是ssh,在dao层去拿的时候,运行时报错为dual is not mapped,[select *.nextval nextvalue from dual] 后来 ...
- [转]notifyDataSetChanged() 动态更新ListView
有时候我们需要修改已经生成的列表,添加或者修改数据,notifyDataSetChanged()可以在修改适配器绑定的数组后,不用重新刷新Activity,通知Activity更新ListView.今 ...
- PyInstaller打包Python脚本为exe
1.PyInstaller-3.1.1 百度云链接 http://pan.baidu.com/s/1jHYWin8 密码 oapl 2.安装最新版本的 pywin32-217.win32-py2 ...
- CentOS系统、Jdk、Tomcat安装实战
CentOS系统.Jdk.Tomcat安装实战 第一次接触Liunx系统,都说J2EE系统在Li ...
- Android Dependencies小差号引起的问题
问题是由于Android Dependencies小差号引起的,下午一搞Android的哥们在群里说最近导入的几个工程每个都是Android Dependencies报错,小差号,我先说解决方法: 方 ...
- Cocos2d-x3.0游戏实例之《别救我》第二篇——创建物理世界
这篇我要给大家介绍两个知识点: 1. 创建游戏物理世界 2. 没了(小若:我噗) 害怕了?不用操心.这太简单了~! 笨木头花心贡献.啥?花心?不呢.是用心~ 转载请注明,原文地址:http://www ...