在使用NDK开发Android应用程序时,经常会遇到JNI文件的中头文件和函数名称不正确,则使用NDK开发应用程序时,步骤如下:
1)第一步:编写java文件
-----------------------------------------------------------------------------
package com.example.demo;
public class Hello{
public native static int add(); //static类型的方法
public native String getString(); //非static类型的方法
}
-----------------------------------------------------------------------------
2)第二步:编译java文件
-----------------------------------------------------------------------------
使用命令:javac Hello.java
3)第三步:使用javah -jni com.example.demo.Hello生成相应JNI的头文件,生成文件如下:
---------------------------------------------------------------------------------------------------
/* DO NOT EDIT THIS FILE - it is machine generated */
#include <jni.h>
/* Header for class com_example_demo_Hello */
#ifndef _Included_com_example_demo_Hello
#define _Included_com_example_demo_Hello
#ifdef __cplusplus
extern "C" {
#endif
/*
* Class: com_example_demo_Hello
* Method: add
* Signature: ()I
*/
JNIEXPORT jint JNICALL Java_com_example_demo_Hello_add
(JNIEnv *, jclass); //表明该方法在java代码中是static类型的
/*
* Class: com_example_demo_Hello
* Method: getString
* Signature: ()Ljava/lang/String;
*/
JNIEXPORT jstring JNICALL Java_com_example_demo_Hello_getString
(JNIEnv *, jobject); //表明该方法在java代码中是非static类型的
#ifdef __cplusplus
}
#endif
#endif
---------------------------------------------------------------------------------------------------
通过第三个步骤的编译结果可以看出,生成的本地方法中,第一个参数类型是JNIEnv* ,是一个指向JNIEnv的指针类型。JNIEnv是一个指向了
一个Pointer,这个Pointer指向了一张函数表,这张函数表的每一项都是JNI中的一个函数的入口。本地的方法通过查找这张表来调用某个JNI函数,来喝
JVM交互。第二个参数,如果声明的方法为static类型的,则该参数为jclass类型;如果声明的参数为非static类型,则为jobject类型。
4)编译当前的JNI文件(Android.mk)
---------------------------------------------------------------------------------------------------
LOCAL_PATH := $(call my-dir)
include $(CLEAR_VARS)
LOCAL_MODULE := hello-jni //编译为动态库的名称hello-jni.so
LOCAL_SRC_FILES := hello-jni.c //编译源文件
include $(BUILD_SHARED_LIBRARY)
----------------------------------------------------------------------------------------------------
5.Android代码下JNI编程中C和C++代码编写差别
-----------------------------------------------------------------------------------------------------------------------
android jni之C 和 C++ 函数实现的比较:
注意:jni.h头文件中对于***.c & ***.cpp采用不同的定义
jclass (JNICALL *GetObjectClass) (JNIEnv *env, jobject obj);
jclass GetObjectClass(jobject obj) {
return functions->GetObjectClass(this,obj);
}
对于***.c
1)jclass test_class = (*env)->GetObjectClass(env, obj);
2)jfieldID id_num = (*env)->GetFieldID(env, test_class, "num", "I");
对于 ***.cpp
1)jclass test_class = env->GetObjectClass(obj);
2)jfieldID id_num = env->GetFieldID(test_class, "num", "I");
在 C 中,
JNI 函数调用由“(*env)->”作前缀,目的是为了取出函数指针所引用的值。
在 C++ 中,
JNIEnv 类拥有处理函数指针查找的内联成员函数。
下面将说明这个细微的差异,其中,这两行代码访问同一函数,但每种语言都有各自的语法。
C 语法:jsize len = (*env)->GetArrayLength(env,array);
C++ 语法:jsize len =env->GetArrayLength(array);
------------------------------------------------------------------------------------------------------------------------
6.编写相应的代码步骤
1)首先编写java层代码
------------------------------------------------------------------------------------------------------------------------
package com.example.jnitest;
public class ArrayTest {
public native String getLine(String str);
}
------------------------------------------------------------------------------------------------------------------------
2)使用cygin命令编译如下:
使用上述命令可以在d/javacode/jnitest/src目录下生成com_example_jnitest_ArrayTest.h
3)将com_example_jnitest_ArrayTest.h头文件添加到.c文件中,编写JNI代码
--------------------------------------------------------------------------------------------------------------------------
#include "com_example_jnitest_ArrayTest.h"
#include<stdio.h>
JNIEXPORT jstring JNICALL Java_com_example_jnitest_ArrayTest_getLine
(JNIEnv *env, jobject clazz, jstring parameter){
char buf[128] = "初学者"; //字符缓冲
char *str;
str = (char *)(env)->GetStringUTFChars(parameter,NULL);
if(str == NULL)
{
return NULL;
}
printf("%s",str);
/**
**使用完了utf-8类型的字符后,我们需要释放由上面方法返回的字符串,这样可以释放被这些字符占用的内存空间,避免造成内存瘫痪
**/
(env)->ReleaseStringUTFChars(parameter, str);
return (env)->NewStringUTF(buf); //该方法实例化一个UTF-8编码的本地字符串为java.lang.String类型,新创建的就是java中
}
--------------------------------------------------------------------------------------------------------------------------------
4)使用cygin编译相应的JNI代码:
------------------------------------------------------------------------------------------------------------------------------------
5)在java文件中添加相应的动态库
----------------------------------------------------------------------------------------------------------------------------------
package com.example.jnitest;
public class ArrayTest {
static {
System.loadLibrary("ArrayTest");
}
public native String getLine(String str);
}
------------------------------------------------------------------------------------------------------------------------------------
6)编写Android应用程序
--------------------------------------------------------------------------------------------------------------------------------------
package com.example.jnitest;
import android.os.Bundle;
import android.app.Activity;
import android.view.Menu;
import android.widget.TextView;
public class MainActivity extends Activity {
TextView mTextView=null;
ArrayTest mArrayTest=null;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
mArrayTest =new ArrayTest();
mTextView = (TextView)findViewById(R.id.textView);
mTextView.setText(mArrayTest.getLine("happy"));
}
}