一步一步学习androidNDK编程(java给c传递数据)

         这篇已经是"一步一步学习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头文件。

  将该MainActivity.java用javac编译为.class文件:这个就是运行该android工程即可在bin/classes目录下生成对应的.class文件。

  命令行下进入bin/classes所在的目录:执行javah com.example.ndkdata.MainActivity即可在该目录下生成对应的com_example_ndkdata_MainActivity.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 __cplusplus
extern "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) {
    
}


        编写完了hello.c之后,需要编写Android.mk,内容如下:

#交叉编译编译c/c++代码所依赖的配置文件

#获取当前Android.mk的路径  
LOCAL_PATH := $(call my-dir)

#变量初始化操作
include $(CLEAR_VARS)

#libhello.so 其实生成的libhello.so就是在我们这个模块的名称前面加上lib后边加上.so
LOCAL_MODULE    := hello
LOCAL_SRC_FILES := hello.c

include $(BUILD_SHARED_LIBRARY)
      现在就需要在cygwine的命令行下进入该工程ndkdata的根目录下,执行"ndk-build"命令生成对应的可直接运行的二进制文件,注意每次在执行"ndk-build"之后最好将android工程clean一下。

      在“ndk-build”成功之后,需要在java代码中引入这些库文件:

static {
        System.loadLibrary("hello");
}

      注意这里的"hello",就是我们在Android.mk文件中写的模块名,即LOCAL_MODULE的值。

      现在,就可以执行声明的native方法了,在java代码中直接调用即可,是不是很简单呢。

源码下载








郑重声明:本站内容如果来自互联网及其他传播媒体,其版权均属原媒体及文章作者所有。转载目的在于传递更多信息及用于网络分享,并不代表本站赞同其观点和对其真实性负责,也不构成任何其他建议。