java把数据传递给c代码
下午弄了个jni关于java语言调用c语言的例子,虽然对于jni的相关步骤一经能熟练掌握,但是由于对于c了解的太少,因此遇到非常之多的问题,无奈只有做好详细笔记,待日后详细理解了。真是不好搞呀!!!!!!!
1、先来看本节的目标代码如下:(目的:练习java把数据传递给c代码)
package com.luochuang.fromjava;
public class DataProvider
{
/**
* 传递两个java的int数据给c语言,c语言处理完之后把数据返回
*
* @param a
* @param b
* @return
*/
public native int add(int a, int b);
/**
* java语言直接传递JAva中的字符串给C语言,C语言处理字符串之后返回新的字符串
*
* @param s
* @return
*/
public native String sayHelloInC(String s);
/**
* java传递int类型的数据给c语言,c语言处理数组的每个元素,把每个元素的值加上10, 再把数据返回给java方法
*
* @param Num
* @return
*/
public native int[] intMethod(int[] Num);
}
2、生成1方法中的java头文件。(来到该工程的bin/classes/目录下利用javah命令生成1类对应的头文件)
/* DO NOT EDIT THIS FILE - it is machine generated */
#include
/* Header for class com_luochuang_fromjava_DataProvider */
#ifndef _Included_com_luochuang_fromjava_DataProvider
#define _Included_com_luochuang_fromjava_DataProvider
#ifdef __cplusplus
extern "C" {
#endif
/*
* Class: com_luochuang_fromjava_DataProvider
* Method: add
* Signature: (II)I
*/
JNIEXPORT jint JNICALL Java_com_luochuang_fromjava_DataProvider_add
(JNIEnv *, jobject, jint, jint);
/*
* Class: com_luochuang_fromjava_DataProvider
* Method: sayHelloInC
* Signature: (Ljava/lang/String;)Ljava/lang/String;
*/
JNIEXPORT jstring JNICALL Java_com_luochuang_fromjava_DataProvider_sayHelloInC
(JNIEnv *, jobject, jstring);
/*
* Class: com_luochuang_fromjava_DataProvider
* Method: intMethod
* Signature: ([I)[I
*/
JNIEXPORT jintArray JNICALL Java_com_luochuang_fromjava_DataProvider_intMethod
(JNIEnv *, jobject, jintArray);
#ifdef __cplusplus
}
#endif
#endif
3、把这个头文件拷贝到自己新建的jni目录下,并新建hello.c文件。
#include "com_luochuang_fromjava_DataProvider.h"
#include
#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__)
//这里实现第一个相加的方法
JNIEXPORT jint JNICALL Java_com_luochuang_fromjava_DataProvider_add(
JNIEnv * env, jobject obj, jint x, jint y) {
//java中的int占四个字节,c语言也是4个字节,因此可以相互转化
LOGD("x=%d",x);
LOGD("y=%d",y);
int result = x + y;
LOGI("result=%d",result);
return x + y;
}
关于这个方法的介绍:
jstring Java_cn_itcast_ndk_DemoActivity_helloFromJNI(JNIEnv *env, jobject javaThis) {
return (*env)->NewStringUTF(env, "Hello from native code!");
}
JNIEnv 类型代表了java环境 通过JNIEnv* 指针,就可以对java端的代码进行操作.
创建java类的对象,调用java对象的方法
获取java对象的属性 等等.
jobject是个什么
jobject obj 就是当前方法所在的类代表的对象.
#include
#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__)
这几句话的意思是在logcat中打印语句。(这里我也不太懂,书上就是这么写的,对c了解太少无奈啊)
4、声明android.mk方法(android.mk是Android操作系统编译的时候使用的编译规则文件)
LOCAL_PATH := $(call my-dir)
include $(CLEAR_VARS)
LOCAL_MODULE := Hello
LOCAL_SRC_FILES := Hello.c
#把一个函数库引入到交叉编译的系统里面
LOCAL_LDLIBS += -llog;
include $(BUILD_SHARED_LIBRARY)
这里的参数的意思是:
LOCAL_PATH必须位于Android.mk文件的最开始。它是用来定位源文件的位置,$(call my-dir)的作用就是返回当前目录的路径。
include $(CLEAR_VARS)的作用是清除一些变量的值,但是LOCAL_PATH除外。
LOCAL_MODULE是用来指定当前待编译模块的名称,在示例中的模块名称为hello-jni
LOCAL_SRC_FILES是用来指定参与编译的源代码文件。这里只编译hell0-jin.c
include $(BUILD_SHARED_LIBRARY)是用来指示将当前模块编译为共享库,前缀为lib,后缀为.so。
还有另外一个BUILD_STATIC_LIBRARY,是用来指示将当前模块编译为静态库的,前缀为.a,后缀为.a。
这就是一个最简单的Android.mk的结构。可能还有另外一点需要介绍:
LOCAL_C_INCLUDES:=include 这个是用来指定在编译时即将使用的c头文件的位置,以当前目录为起点。
具体可以参考这里:android ndk 之Android.mk编写
5、来到当前文件的根目录下利用ndk-build命令编译。
6、来到当前的Activity。
package com.luochuang.fromjava;
import android.app.Activity;
import android.os.Bundle;
import android.view.View;
import android.widget.Toast;
public class HelloFromJavaActivity extends Activity
{
DataProvider privider;
static
{
// 用来装载库文件,不论是JNI库文件还是非JNI库文件。在任何本地方法被调用之前必须先用这个两个方法之一把相应的JNI库文件装载。System.loadLibrary
// 参数为库文件名,不包含库文件的扩展名。
System.loadLibrary("Hello");
}
@Override
public void onCreate(Bundle savedInstanceState)
{
super.onCreate(savedInstanceState);
setContentView(R.layout.main);
privider = new DataProvider();
}
public void onClick(View view)
{
switch (view.getId())
{
case R.id.cfromjava:
int resulst = privider.add(10, 20);
Toast.makeText(HelloFromJavaActivity.this, "调用结果" + resulst, 0)
.show();
break;
}
}
}
最后吐司显式结果。