ndk开发中利用java与c之间互相传递数据

1389人阅读 评论(0) 收藏 举报
分类:

1、DataProvider

package com.njupt.ndk_passdata;

public class DataProvider {

	public native int add(int x,int y);
	public static native int sub(int x,int y);
	
	public native String sayHelloInC(String s);
	public native int[] intMethod(int[] iNum);
	public native byte[] byteMethod(byte[] iByte);
	
}


2、MainActivity

package com.njupt.ndk_passdata;

import android.os.Bundle;
import android.app.Activity;
import android.view.Menu;
import android.view.View;
import android.view.View.OnClickListener;
import android.widget.Button;
import android.widget.Toast;

public class MainActivity extends Activity implements OnClickListener{

	static{
		System.loadLibrary("Hello");
	}
	
	private Button bt1;
	private Button bt2;
	private Button bt3;
	private Button bt4;
	private DataProvider provider;
	
	@Override
	protected void onCreate(Bundle savedInstanceState) {
		super.onCreate(savedInstanceState);
		setContentView(R.layout.main);
		
		bt1 = (Button) findViewById(R.id.bt1);
		bt2 = (Button) findViewById(R.id.bt2);
		bt3 = (Button) findViewById(R.id.bt3);
		bt4 = (Button) findViewById(R.id.bt4);
		
		bt1.setOnClickListener(this);
		bt2.setOnClickListener(this);
		bt3.setOnClickListener(this);
		bt4.setOnClickListener(this);
		provider = new DataProvider();
	}

	public void onClick(View v){
		switch (v.getId()) {
		case R.id.bt1:
			int result = provider.add(3, 5);
			Toast.makeText(this, "相加的结果: " + result, 1).show();
			break;

		case R.id.bt2:
			String str = provider.sayHelloInC("zhangsan ");
			Toast.makeText(this, str, 1).show();
			break;
			
		case R.id.bt3:
			int[] arr = {1,2,3,4,5};
			provider.intMethod(arr);
			
			for(int i = 0 ; i < arr.length ; ++i){
				System.out.println("java " + arr[i]);
			}
			
			break;
		case R.id.bt4:
			int subresult = DataProvider.sub(5, 3);
			Toast.makeText(this, "相见的结果是: " + subresult, 1).show();
		    
			break;
		default:
			break;
		}
	}
	@Override
	public boolean onCreateOptionsMenu(Menu menu) {
		// Inflate the menu; this adds items to the action bar if it is present.
		getMenuInflater().inflate(R.menu.main, menu);
		return true;
	}

}

.h文件的获取可以通过以下手段得到:

切换到工程的src目录下,运行javah命令得到

3、Hello.c

#include <stdio.h>
#include <jni.h>
#include "com_njupt_ndk_passdata_DataProvider.h"

#include <android/log.h>//include  D:\android-ndk-r7b\platforms\android-8\arch-arm\usr\include\android下的log.h这个目录
#define LOG_TAG "System.out"
#define LOGD(...) __android_log_print(ANDROID_LOG_DEBUG, LOG_TAG, __VA_ARGS__)
#define LOGI(...) __android_log_print(ANDROID_LOG_INFO, LOG_TAG, __VA_ARGS__)

/**
 * 返回值 char* 这个代表char数组的首地址
 *  Jstring2CStr 把java中的jstring的类型转化成一个c语言中的char 字符串
 */
char* Jstring2CStr(JNIEnv* env, jstring jstr) {
	char* rtn = NULL;
	jclass clsstring = (*env)->FindClass(env, "java/lang/String"); //String
	jstring strencode = (*env)->NewStringUTF(env, "GB2312"); // 得到一个java字符串 "GB2312"
	jmethodID mid = (*env)->GetMethodID(env, clsstring, "getBytes",
			"(Ljava/lang/String;)[B"); //[ String.getBytes("gb2312");
	jbyteArray barr = (jbyteArray)(*env)->CallObjectMethod(env, jstr, mid,
			strencode); // String .getByte("GB2312");
	jsize alen = (*env)->GetArrayLength(env, barr); // byte数组的长度
	jbyte* ba = (*env)->GetByteArrayElements(env, barr, JNI_FALSE);
	if (alen > 0) {
		rtn = (char*) malloc(alen + 1);         //"\0"
		memcpy(rtn, ba, alen);
		rtn[alen] = 0;
	}
	(*env)->ReleaseByteArrayElements(env, barr, ba, 0);  //
	return rtn;
}

JNIEXPORT jint JNICALL Java_com_njupt_ndk_1passdata_DataProvider_add(
		JNIEnv * env, jobject obj, jint x, jint y) {

	return x+y;
}

/**
 * 代表的是调用c代码 的class类
 * jclass DataProvider  类
 */
JNIEXPORT jint JNICALL Java_com_njupt_ndk_1passdata_DataProvider_sub(JNIEnv * env,
		jclass clazz, jint x, jint y) {//特别要注意静态方法签名的第二个参数..
	return x-y;
}

JNIEXPORT jstring JNICALL Java_com_njupt_ndk_1passdata_DataProvider_sayHelloInC(
		JNIEnv * env, jobject obj, jstring jstr) {

	//在c语言中 是没有java的String
	char* cstr = Jstring2CStr(env,jstr);//将java风格的字符串转换成c风格的字符串

	// c语言中的字符串 都是以'/0' 作为结尾.不加的话这个字符串的最后一位是任意字符..
	char arr[] = {' ','h','e','l','l','o','\0'};
	strcat(cstr,arr);

	return (*env)->NewStringUTF(env,cstr);//将c风格的字符转换成java风格的字符串
}

/**env java 虚拟机 结构体c实现的指针 包含的有很多jni方法
 *jobject obj 代表的是调用这个c代码的java对象 代表的是DataProider的对象
 */

JNIEXPORT jintArray JNICALL Java_com_njupt_ndk_1passdata_DataProvider_intMethod(
		JNIEnv * env, jobject obj, jintArray arr) {

	/**
	 * 要操作数组的必要条件:
	 * 1、知道数组的长度
	 * 2、操作数组中的每一个元素
	 */
	int len = (*env)->GetArrayLength(env,arr);//返还这个数组的长度..

	LOGD("shuzu len =%d",len);
	jint* intarr = (*env)->GetIntArrayElements(env,arr,0);
	int i;
	for(i = 0 ; i < len ; ++i){
		LOGD("intarr[%d]=%d", i, intarr[i]);
		intarr[i] += 10;
	}

	return arr;
}

JNIEXPORT jbyteArray JNICALL Java_com_njupt_ndk_1passdata_DataProvider_byteMethod(
		JNIEnv * env, jobject obj, jbyteArray bytearr) {

}


4、Android.mk

 LOCAL_PATH := $(call my-dir)

   include $(CLEAR_VARS)

   LOCAL_MODULE    := Hello   
   LOCAL_SRC_FILES := Hello.c
   LOCAL_LDLIBS += -llog
   
   include $(BUILD_SHARED_LIBRARY)


5、main.xml

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="fill_parent"
    android:layout_height="fill_parent"
    android:orientation="vertical" >

    <Button
        android:id="@+id/bt1"
        android:layout_width="fill_parent"
        android:layout_height="wrap_content"
        android:text="传递两个int给c代码" />

    <Button
        android:id="@+id/bt2"
        android:layout_width="fill_parent"
        android:layout_height="wrap_content"
        android:text="传递string给c代码" />

    <Button
        android:id="@+id/bt3"
        android:layout_width="fill_parent"
        android:layout_height="wrap_content"
        android:text="传递int数组给c代码" />

        <Button
        android:id="@+id/bt4"
        android:layout_width="fill_parent"
        android:layout_height="wrap_content"
        android:text="调用静态native方法" />
</LinearLayout>

这是运行程序,你会看到以下结果:


1
1

查看评论
* 以上用户言论只代表其个人观点,不代表CSDN网站的观点或立场
    个人资料
    • 访问:1889851次
    • 积分:32482
    • 等级:
    • 排名:第148名
    • 原创:1199篇
    • 转载:278篇
    • 译文:3篇
    • 评论:181条
    学习经历
    毕业学校:南京邮电大学本科
    成长的地方:创新工场、美团点评
    职位:游戏研发工程师、Android研发工程师
    努力奋斗、快乐生活
    Android课程交流答疑QQ: 1019648568
    博客专栏
    最新评论