NDK配置及JNI入门教程之HelloWorld篇

> Studio编译SO包??

AndroidStudio 编译C/C++文件生成SO文件- http://blog.csdn.net/krubo1/article/details/50547681

Android SO文件的兼容和适配- http://blog.coderclock.com/2017/05/07/android/Android-so-files-compatibility-and-adaptation/?ref=myread

JNI官方规范中文版- http://blog.csdn.net/a18706739873/article/details/50574836
Java本地接口(JNI)编程指南和规范2中文版.pdf- http://download.csdn.net/download/ubuntu8888/5220038
jni官方规范- http://download.csdn.net/download/u012342012/6357989

JNI规范- http://blog.csdn.net/a345017062/article/category/1256568

NDK开发- http://blog.csdn.net/a396901990/article/category/6305704

Android 通过JNI实现守护进程- http://blog.csdn.net/yyh352091626/article/details/50542554

JNI入门教程之HelloWorld篇- http://blog.csdn.net/mingjava/article/details/180946

> NDK环境的配置.
Eclipse中配置NDK R10环境- http://blog.csdn.net/mole/article/details/42520185
AndroidStudio如何配置NDK/JNI开发环境-http://blog.csdn.net/aplixy/article/details/51429305
Android Studio NDK开发环境搭建-http://jingyan.baidu.com/article/456c463b4a52460a58314402.html


> JNI编程指南和规范的中文翻译--http://download.csdn.net/detail/a345017062/2674842
本文讲述如何使用JNI技术实现HelloWorld,目的是让读者熟悉JNI的机制并编写第一个HelloWorld程序。
    Java Native Interface(JNI)是Java语言的本地编程接口,是J2SDK的一部分。在java程序中,我们可以通过JNI实现一些用java语言不便实现的功能。通常有以下几种情况我们需要使用JNI来实现。
标准的java类库没有提供你的应用程序所需要的功能,通常这些功能是平台相关的
你希望使用一些已经有的类库或者应用程序,而他们并非用java语言编写的
程序的某些部分对速度要求比较苛刻,你选择用汇编或者c语言来实现并在java语言中调用他们
    在《java核心技术》中,作者提到JNI的时候,建议不到万不得已不要使用JNI技术,一方面它需要你掌握更多的知识才可以驾驭,一方面使用了JNI你的程序就会丧失可移植性。在本文我们跳过JNI的底层机制,读者最好先把它想象为本地代码和java代码的粘合剂。

> 开始编写HelloWord例子,在D盘下建了个Android工程
一步一步学习androidNDK编程(hello world)- http://blog.csdn.net/mockingbirds/article/details/45132551

APP监听是否被卸载及卸载反馈功能- http://blog.csdn.net/mockingbirds/article/details/46594107 

   下面我们开始编写HelloWorld程序,由于涉及到要编写c/c++代码因此我们会在开发中使用Microsoft VC++或Eclipse工具
1. 我们在硬盘上建立一个UninstallNative目录作为我们的工作目录,首先我们需要编写自己的java代码,在java代码中我们会声明native方法,代码非常简单。如下所示

    在Android工程中编写UninstallNative.java文件
package com.desaco.uninstallfeedback.native_method;
public class UninstallNative {
    //com.desaco.uninstallfeedback.native_method.UninstallNative
    static {
        System.loadLibrary("uninstall_app_feedback");
    }
    public native String userFeedback();
}
  注意我们的userFeedback()方法的声明,它有一个关键字native,表明这个方法使用java以外的语言实现。方法不包括实现,因为我们要用c/c++语言实现它。注意System.loadLibrary("uninstall_app_feedback")这句代码,它是在静态初始化块中定义的,系统用来装载hello共享库,这就是我们在后面生成的uninstall_app_feedback.dll(如果在其他的操作系统可能是其他的形式,比如uninstall_app_feedback.so)

2.切换到 D:\workpalce_fun\UninstallFeedback\src\com\desaco\uninstallfeedback\native_method目录下,打开cmd窗口(已经配置好了JDK环境)

   生成class文件命令: javac UninstallNative.java  生成UninstallNative.class文件

3.JDK自带了javah命令可以实现,也可以用eclipse中ant 脚本去生成一个.h的头文件,假设工程目录下有bin文件夹,bin存放编译好的class文件;

Android工程目录D:\workpalce_fun\UninstallFeedback\bin\classes

  生成.h文件命令: D:\workpalce_fun\UninstallFeedback\bin\classes>

javah -jni com.desaco.uninstallfeedback.native_method.UninstallNative

注意:com.desaco.uninstallfeedback.native_method是包名,UninstallNative是类名. 

  com_desaco_uninstallfeedback_native_method_UninstallNative.h文件如下:

/* DO NOT EDIT THIS FILE - it is machine generated */
#include <jni.h>
/* Header for class com_desaco_uninstallfeedback_native_method_UninstallNative */

#ifndef _Included_com_desaco_uninstallfeedback_native_method_UninstallNative
#define _Included_com_desaco_uninstallfeedback_native_method_UninstallNative
#ifdef __cplusplus
extern "C" {
#endif
/*
 * Class:     com_desaco_uninstallfeedback_native_method_UninstallNative
 * Method:    displayHelloWorld
 * Signature: ()V
 */
JNIEXPORT void JNICALL Java_com_desaco_uninstallfeedback_native_1method_UninstallNative_displayHelloWorld
  (JNIEnv *, jobject);

/*
 * Class:     com_desaco_uninstallfeedback_native_method_UninstallNative
 * Method:    userFeedback
 * Signature: ()Ljava/lang/String;
 */
JNIEXPORT jstring JNICALL Java_com_desaco_uninstallfeedback_native_1method_UninstallNative_userFeedback
  (JNIEnv *, jobject);

#ifdef __cplusplus
}
#endif
#endif

    JNI是一门复杂,有趣且富有挑战性的技术,我们在后面的几篇文章中将说明如何在java语言和Native语言之间进行交互。

> 编译SO库

eclipse中android NDK开发环境的搭建及编译SO库 - http://blog.csdn.net/wangzhen209/article/details/50312633
Eclipse下用NDK编译生成so文件- http://blog.csdn.net/devilnov/article/details/41543975
在Eclipse+SDK IDE中,配置Window-->Preferences-->Android-->NDK,在NDK Location中输入之前android-ndk-r10d的全路径

> NDK环境配置
配置NDK加入到path环境变量,如:path: d:/ide/android-ndk-r10
然后在CMD窗口中输入ndk-build  -version,检测是否安装成功

(添加环境变量  NDK_ROOT 为:   D:\Android\android-ndk-r10d
在环境变量 PATH 下追加 :%NDK_ROOT%;)

在CMD窗口中,切换到工程的JNI目录下,输入命令ndk-build

android应用监听自身卸载,弹出用户反馈调查- https://github.com/janstk/LaoUtils  
https://github.com/wqycsu/uninstalldemo

Android5.0以下可以适用:http://download.csdn.net/download/a378881925/8373409


AndroidStudio 编译C/C++文件生成SO文件- http://blog.csdn.net/krubo1/article/details/50547681

》》》》》》》》》》》》》》》》》》》》》》》》》》》》》》》》》》》》

Android JNI 使用的数据结构JNINativeMethod详解

Andoird 中使用了一种不同传统Java JNI的方式来定义其native的函数。其中很重要的区别是Andorid使用了一种Java 和 C 函数的映射表数组,并在其中描述了函数的参数和返回值。这个数组的类型是JNINativeMethod,定义如下:

typedef struct {
const char* name;
const char* signature;
void* fnPtr;
} JNINativeMethod;

第一个变量name是Java中函数的名字。

第二个变量signature,用字符串是描述了函数的参数和返回值

第三个变量fnPtr是函数指针,指向C函数。

其中比较难以理解的是第二个参数,例如

"()V"

"(II)V"

"(Ljava/lang/String;Ljava/lang/String;)V"

实际上这些字符是与函数的参数类型一一对应的。

"()" 中的字符表示参数,后面的则代表返回值。例如"()V" 就表示void Func();

"(II)V" 表示 void Func(int, int);

具体的每一个字符的对应关系如下

字符   别名          Java类型        C++本地类型        字节(bit)  

V      void            void           
Z       jboolean     boolean       unsigned char    8, unsigned  
I        jint              int            long             32 
J       jlong            long           __int64          64
D      jdouble       double          double            64
F      jfloat            float           float             32
B      jbyte            byte           signed char     8
C      jchar           char            unsigned short   16, unsigned
S      jshort          short           
short            16

数组则以"["开始,用两个字符表示

[I       jintArray      int[]
[F     jfloatArray    float[]
[B     jbyteArray    byte[]
[C    jcharArray    char[]
[S    jshortArray   short[]
[D    jdoubleArray double[]
[J     jlongArray     long[]
[Z    jbooleanArray boolean[]

上面的都是基本类型。如果Java函数的参数是class,则以"L"开头,以";"结尾中间是用"/" 隔开的包及类名。而其对应的C函数名的参数则为jobject. 一个例外是String类,其对应的类为jstring

Ljava/lang/String; String jstring
Ljava/net/Socket; Socket jobject

如果JAVA函数位于一个嵌入类,则用$作为类名间的分隔符。

例如 "(Ljava/lang/String;Landroid/os/FileUtils$FileStatus;)Z"


Android JNI 数据类型转换:http://www.cnblogs.com/likwo/archive/2012/05/22/2512483.html

Android之NDK开发:http://www.cnblogs.com/devinzhang/archive/2012/02/29/2373729.html

你可能感兴趣的:(NDK配置及JNI入门教程之HelloWorld篇)