首页 > 代码库 > JNI学习心得(1)

JNI学习心得(1)

最近由于公司需求,开发Android项目,之前没接触过,了解甚少。

Android驱动层的开发流程(个人理解):(自上而下)Android APP =>JNI => HAL => Linux devices Read/Write

据我了解,大致流程就是这样,如有不对,欢迎大礼拍砖。

由于驱动层一般由C/C++开发。需要与JAVA交互需要通过JNI。

JNI负责JAVA与其他语言(C/C++)进行交互。类似于windows平台下,不同语言之间用.dll封装,进行交互。

JNI最终会生成 .so ,感觉就是类似于.dll文件。

HAL层没有充分理解,大致是这样一回事吧,不确定:Android的/dev下的设备文件,只负责对设备的硬件时序操作(如SPI、寄存器),而功能性驱动则由HAL层进行开发,说是保密性。

前奏少说了。直奔主题:

JNI—Demo 参考自:http://www.cnblogs.com/chenjiajin/archive/2012/04/12/2444188.html

本人只作归纳性:

1、安装NDK开发工具(只需下载解压就可以了,好像还有环境变量要设备,具体百度)

2、新建安卓工程

3、右击工程NEW->Folder 新建一个文件夹,一般取名JNI。该目录存放一些与C/C++的交互方法和C/C++的编译文件,源码之类。(访目录下一步会配置到)

4、配置Builder(Bulider主要是告诉eclipse用什么方法,工具,参数编译JNI目录中的文件)

  引用一下原文:

(a)Project->Properties->Builders->New,新建一个Builder。 
(b)在弹出的【Choose configuration type】对话框,选择【Program】,点击【OK】: 
(c)在弹出的【Edit Configuration】对话框中,配置选项卡【Main】。
     在“Name“中输入新builders的名称(我取名为Ndk_Builder)。
       在“Location”中输入nkd-build.cmd的路径。
      (我的是D:\AndroidDev\android-ndk-r7\ndk-build.cmd,根据各自的ndk路径设置,也可以点击“Browser File System…”来选取这个路径)。
         在“Working Diretcoty”中输入${workspace_loc:/TestNdk}(也可以点击“Browse Workspace”来选取TestNdk目录)。
 
(d)【Edit Configuration】对话框中,配置选项卡【Refresh】。
      勾选“Refresh resources upon completion”,
      勾选“The entire workspace”,
      勾选“Recuresively include sub-folders”。
 
(e)【Edit Configuration】对话框中,配置选项卡【Build options】。
      勾选“After a “Clean””,
      勾选“During manual builds”,
      勾选“During auto builds”,
      勾选“Specify working set of relevant resources”。
 
      点击“Specify Resources…”
      勾选TestNdk工程的“jni“目录(刚新建的是什么名,就选什么),点击”finish“。 
  

  点击“OK“,完成配置。
  OK,到这里Eclipse就能够自动调用NDK编译jin目录下的C/C++代码了。

5、在刚建立的Android中,添加个类,如JniClient(该类主要是存放函数定义,类似于C语言中的H文件)。内容如下:

  

package xxx.xxx.xxx;

public class JniClient {

static public native String AddStr(String strA, String strB); //该方法的实现将会在JNI目录的C里面
static public native int AddInt(int a, int b);//同上
}

 

6、进入安卓工程目录下的 bin\classes\包名..\包名...\ 中,你将会发现,第五步添加的JniClient类已经被ecliipse编译出了 JniClient.class 

  用CMD进入“安卓本工程目录\bin\classes目录” ,输入”javah 工程包名.JniClient“后回车,

  在 "本工程\bin\classes“目录下就生成了C++头文件”工程包名_JniClient.h“。”工程包名_JniClient.h“ 的文件内容如下:

 

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

#ifndef _Included_com_ndk_test_JniClient
#define _Included_com_ndk_test_JniClient
#ifdef __cplusplus
extern "C" {
#endif
/*
 * Class:     com_ndk_test_JniClient
 * Method:    AddStr
 * Signature: (Ljava/lang/String;Ljava/lang/String;)Ljava/lang/String;
 */
JNIEXPORT jstring JNICALL Java_com_ndk_test_JniClient_AddStr
  (JNIEnv *, jclass, jstring, jstring);

/*
 * Class:     com_ndk_test_JniClient
 * Method:    AddInt
 * Signature: (II)I
 */
JNIEXPORT jint JNICALL Java_com_ndk_test_JniClient_AddInt
  (JNIEnv *, jclass, jint, jint);

#ifdef __cplusplus
}
#endif
#endif

 

这些主要是为了等会告诉c文件,需要去实现这些封装的函数。

 

7、在jni目录下新建一个Android.mk文件(建议在eclipse中建立,如果在文件夹中建立,需要在eclipse中导入),其内容如下(详细的语法以后再另外解释):
LOCAL_PATH := $(call my-dir)
include $(CLEAR_VARS)
LOCAL_MODULE := TestNdk
LOCAL_SRC_FILES := com_ndk_test_JniClient.c
include $(BUILD_SHARED_LIBRARY)

8、将 ”包名_JniClient.h“拷贝到TestNdk工程的jni目录下, 然后新建一个"包名_JniClient.c"文件(建议在eclipse中建立,如果在文件夹中建立,需要在eclipse中导入),

     完成头文件中函数的实现,其内容如下(本来 包名_JniClient_AddStr 是想完成字符串相加的功能的,但数据转换有点问题,想先写完本文档,后续再研究jni数据类型的问题,所以只简单的返回一个字符串。):
#include "com_ndk_test_JniClient.h"
#include <stdlib.h>
#include <stdio.h>

#ifdef __cplusplus   
extern "C"  
{   
#endif  
/*
 * Class:     com_ndk_test_JniClient
 * Method:    AddStr
 * Signature: (Ljava/lang/String;Ljava/lang/String;)Ljava/lang/String;
 */
JNIEXPORT jstring JNICALL Java_com_ndk_test_JniClient_AddStr
  (JNIEnv *env, jclass arg, jstring instringA, jstring instringB)
{
    jstring str = (*env)->NewStringUTF(env, "HelloWorld from JNI !");
    return str;       
}

/*
* Class:     com_ndk_test_JniClient
* Method:    AddInt
* Signature: (II)I
*/
JNIEXPORT jint JNICALL Java_com_ndk_test_JniClient_AddInt
  (JNIEnv *env, jclass arg, jint a, jint b)
{
    return a + b;
}

#ifdef __cplusplus   
}   
#endif

编辑 包名_JniClient.c 并保存后,可以看到 安卓本工程下的obj/local/armeabi目录下将自动生成 "lib工程名.so"库。

 

9、在TestNdkActivity.java中完成对JniClient.java中函数的调用:
package com.ndk.test;

import android.app.Activity;
import android.os.Bundle;
import android.widget.TextView;

public class TestNdkActivity extends Activity {
    static {
        System.loadLibrary("TestNdk");
    }

    /** Called when the activity is first created. */
    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
//        setContentView(R.layout.main);
        
        String str = JniClient.AddStr("prefix", "suffix");
        
        
        int iSum = JniClient.AddInt(5, 2);        
        String strSum = "5 + 7 = " + iSum;
        
        TextView tv1 = new TextView(this);
        tv1.setText(str);
        setContentView(tv1);
    }
}


10、运行TestNdk工程,在模拟器中可以看到界面输出来自com_ndk_test_JniClient.c 文件中的“HelloWorld from JNI ! “。

OK,NDK实例到此完成。后续就可以深入的学习NDK/JNI了,比如C/C++与Java的数据类型转换,Android.mk文件的编写格式等。

 

 

核心过程是: 建ANDROID工程=>建JNI目录=>配置编译方式(builder)=>建JNI函数定义类(JNIClient)=>定义类中添加需要在c中实现的函数原形=>

      使用javah编译JNIClient.class文件,生成 h文件 =>复制h文件到JNI目录 => 在JNI目录中(eclipse中)建立Android.mk文件 => 

      建立与 c文件实现 JNIClient中的定义的函数=>其他java类的调用方法 JNIClient.方法名(函数名)