这篇已经是"一步一步学习androidNDK编程"的第四篇了,在这篇中,我们将会在java中传递代码给c代码。
首先,我们新建一个android工程"ndkdata",
第一步:
同样的首先声明native方法,如下:
public native int add(int a,int b);public native String helloSir(String name);public native int[] intMethod(int[] intArray);可以看到我这里声明了三个方法,用来传递不同的数据类型给c代码
第二步:
由于我们的这些方法都是声明在MainActivity.java中的,所以我们需要首先将该MainActivity.java用javac编译为.class文件,然后运用javah,生成对应的.h头文件。
/* DO NOT EDIT THIS FILE – it is machine generated */#include <jni.h>/* Header for class com_example_ndkdata_MainActivity */#ifndef _Included_com_example_ndkdata_MainActivity#define _Included_com_example_ndkdata_MainActivity#ifdef __cplusplusextern "C" {#endif/* * Class:com_example_ndkdata_MainActivity * Method: add * Signature: (II)I */JNIEXPORT jint JNICALL Java_com_example_ndkdata_MainActivity_add (JNIEnv *, jobject, jint, jint);/* * Class:com_example_ndkdata_MainActivity * Method: helloSir * Signature: (Ljava/lang/String;)Ljava/lang/String; */JNIEXPORT jstring JNICALL Java_com_example_ndkdata_MainActivity_helloSir (JNIEnv *, jobject, jstring);/* * Class:com_example_ndkdata_MainActivity * Method: intMethod * Signature: ([I)[I */JNIEXPORT jintArray JNICALL Java_com_example_ndkdata_MainActivity_intMethod (JNIEnv *, jobject, jintArray);#ifdef __cplusplus}#endif#endif 接下来我们在ndkdata工程中,创建jni目录,然后再将刚才生成的com_example_ndkdata_MainActivity.h文件拷贝到该目录下,然后创建我们的c代码:hello.c
将com_example_ndkdata_MainActivity.h中生成的方法添加到hello.c文件中这里需要添加参数,因为默认生成的只有方法的参数类型,没有参数对象,并且引入对应的头文件,如下:
#include<stdio.h>#include<jni.h>#include "com_example_ndkdata_MainActivity.h"JNIEXPORT jint JNICALL Java_com_example_ndkdata_MainActivity_add (JNIEnv * env, jobject obj, jint x, jint y) {}JNIEXPORT jstring JNICALL Java_com_example_ndkdata_MainActivity_helloSir (JNIEnv * env, jobject obj, jstring jstr) {}JNIEXPORT jintArray JNICALL Java_com_example_ndkdata_MainActivity_intMethod (JNIEnv * env, jobject obj, jintArray arr) {}接下来就是实现我们的c代码了,如下:#include<stdio.h>#include<jni.h>#include "com_example_ndkdata_MainActivity.h"JNIEXPORT jint JNICALL Java_com_example_ndkdata_MainActivity_add (JNIEnv * env, jobject obj, jint x, jint y) { return x+y; }/** * 返回值 char* 这个代表char数组的首地址 * Jstring2CStr 把java中的jstring的类型转化成一个c语言中的char 字符串 */char* Jstring2CStr(JNIEnv* env, jstring jstr){<span style="white-space:pre"></span> char* rtn = NULL;<span style="white-space:pre"></span> jclass clsstring = (*env)->FindClass(env,"java/lang/String"); //String<span style="white-space:pre"></span> jstring strencode = (*env)->NewStringUTF(env,"GB2312"); // 得到一个java字符串 "GB2312"<span style="white-space:pre"></span> jmethodID mid = (*env)->GetMethodID(env,clsstring, "getBytes", "(Ljava/lang/String;)[B"); //[ String.getBytes("gb2312");<span style="white-space:pre"></span> jbyteArray barr= (jbyteArray)(*env)->CallObjectMethod(env,jstr,mid,strencode); // String .getByte("GB2312");<span style="white-space:pre"></span> jsize alen = (*env)->GetArrayLength(env,barr); // byte数组的长度<span style="white-space:pre"></span> jbyte* ba = (*env)->GetByteArrayElements(env,barr,JNI_FALSE);<span style="white-space:pre"></span> if(alen > 0)<span style="white-space:pre"></span> {<span style="white-space:pre"></span> rtn = (char*)malloc(alen+1); //"\0"<span style="white-space:pre"></span> memcpy(rtn,ba,alen);<span style="white-space:pre"></span> rtn[alen]=0;<span style="white-space:pre"></span> }<span style="white-space:pre"></span> (*env)->ReleaseByteArrayElements(env,barr,ba,0); //<span style="white-space:pre"></span> return rtn;}JNIEXPORT jstring JNICALL Java_com_example_ndkdata_MainActivity_helloSir (JNIEnv * env, jobject obj, jstring jstr) { //在c语言中 是没有java的String<span style="white-space:pre"></span>char* cstr = Jstring2CStr(env, jstr);<span style="white-space:pre"></span>// c语言中的字符串 都是以'\0' 作为结尾<span style="white-space:pre"></span>char arr[7]= {' ','h','e','l','l','o','\0'};<span style="white-space:pre"></span>strcat(cstr,arr);<span style="white-space:pre"></span>return (*env)->NewStringUTF(env,cstr); }JNIEXPORT jintArray JNICALL Java_com_example_ndkdata_MainActivity_intMethod (JNIEnv * env, jobject obj, jintArray arr) { }见所未见,闻所未闻。