Android 下 JNI 开发

本文介绍了解决在Android模拟器上加载.so库文件时出现的错误问题,包括如何正确配置Application.mk文件,确保应用程序能同时支持armeabi、armeabi-v7a和x86架构。

6. 确定加载.so库文件的名字没有错误, 还是报一下错误. 是模拟器问题. 把生成的arm下的.so文件,放到x86模拟器上运行就报此错误.

 

   Caused by: java.lang.UnsatisfiedLinkError: Couldn't load atguigu31: findLibrary returned null

 

 

- 解决方法:

 1、在jni目录下创建一个Application.mk文件, 声明以下内容:

 先看查看文档:/android-ndk-r9/docs/APPLICATION-MK.html

 再看/android-ndk-r9/docs/CPU-X86.html文档:

 

Generating x86 machine code is simple: just add 'x86' to your APP_ABI definition in your Application.mk file, for example:

生成的x86机器代码很简单:只需添加86你app_abi定义你的文件,例如:

 

2、具体在Application.mk文件内容如下

 

     APP_ABI := armeabi armeabi-v7a x86

 

           APP_ABI := all

       // 当前只能在x86的模拟器上运行, 因为生成的so文件是x86机器的机器码

          APP_ABI := x86

06_Android.mk详细介绍-10

 

# $是调用编译工具链中的函数, 这里的作用是获取当前文件目录.

LOCAL_PATH := $(call my-dir)

# 清空并初始化编译工具链的变量.

# 注意: 会清空所有的LOCAL_变量, 但不会清空LOCAL_PATH的变量.

include $(CLEAR_VARS)

# 编译出来文件的名字, 编译系统会在名字前加lib前缀. 如果已经写lib, 不会再添加.

# 注意: 不可以加扩展名.

LOCAL_MODULE    := atguigu

# 指定编译的源文件.如果多个源文件可以用空格连接起来

LOCAL_SRC_FILES := hello.c

# 指定编译文件类型

# 动态库(BUILD_SHARED_LIBRARY) 扩展名:.so 体积小.

# 静态库(BUILD_STATIC_LIBRARY) 扩展名:.a  体积大.

include $(BUILD_SHARED_LIBRARY)

 

 

Android.mk 的含义

 

LOCAL_PATH:=$(call my-dir)

 

LOCAL_PATH是定义源文件在哪个目录用的.

 

my-dir 是个定义的宏方法, $(call my-dir)就是调用这个叫 my-dir的宏方法,这个方法返回值就是

Android.mk文件所在的目录

 

include $(CLEAR_VARS)

 

CLEAR_BARS 变量是build system里面的一个变量

这个变量指向了所有的类似 LOCAL_XXX的变量,

执行完这一句话, 这个编译系统就把 所有的类似

LOCAL_MODULE,_SRC_FILELOCALS,LOCAL_STATIC_LIBRARIES,...这样的变量都清除掉

但是不会清除掉 LOCAL_PATH

 

LOCAL_MODULE  就是你要生成的库的名字,名字要是唯一的这个.不能有空格.

编译后系统会自动在前面加上lib的头, 比如说我们的Hello 就编译成了libHello.so

 

还有个特点就是如果你起名叫libHello 编译后ndk就不会给你的module名字前加上lib了

 

但是你最后调用的时候 还是调用Hello这个库

 

 

 

LOCAL_SRC_FILES = :Hello.c

这个是指定你要编译哪些文件

不需要指定头文件 ,引用哪些依赖, 因为编译器会自动找到这些依赖 自动编译

 

 

include $(BUILD_SHARED_LIBRARY)  BUILD_STATIC_LIBRARY

.so

 

编译后生成的库的类型,如果是静态库.a 配置include $(BUILD_STATIC_LIBRARY)

 

 

别的参数

 

LOCAL_CPP_EXTENSION := cc //指定c++文件的扩展名

LOCAL_MODULE    := ndkfoo

LOCAL_SRC_FILES := ndkfoo.cc

 

LOCAL_LDLIBS += -llog -lvmsagent -lmpnet -lmpxml -lH264Android

//指定需要加载一些别的什么库.

 

 

 

07_NDK 简便开发流程-22

1. 关联NDK: Window -> Preferences -> Android -> NDK

2. 创建Android工程:SimpleJNIDemo, 声明native方法.

3. 设置函数库名字: 右键工程 -> Android Tools -> App Native support

4. 使用javah生成.h的头文件, 并把.h文件拷贝到工程下jni文件夹中.

5. c代码提示: 右键工程 -> Properties -> C/C++ General ->  Path and Symbols

// Includes -> Add -> File system 选中以下路径.

 platforms\android-18\arch-arm\usr\include

如图:

6.#include "com_atguigu_simplejnidemo_MainActivity.h"

7. 把后缀名.cpp改成.c, 实现native方法.

8. java代码中加载动态库, 调用native方法.

9.演示自动编译和直接运行

 

 

 

编译通过讲解 Android.mk

讲解 native代码生成的so文件存放在手机的哪个目录

 

#include <string.h>

#include <jni.h>

 

jstring Java_cn_atguigu_ndk_DemoActivity_helloFromJNI(JNIEnv *env, jobject javaThis) {

 

  return (*env)->NewStringUTF(env, "Hello from native code!");

 

}

 

LOCAL_PATH := $(call my-dir)

include $(CLEAR_VARS)

# Here we give our module name and source file(s)

LOCAL_MODULE    := Hello

LOCAL_SRC_FILES := Hello.c

 

include $(BUILD_SHARED_LIBRARY)

 

 

本教程由尚硅谷教育大数据研究院出品,如需转载请注明来源。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值