1.首先创建 DataProvider类:

package com.pl.ndkpassdata;

public class DataProvider {
static{
System.loadLibrary("passdata");//加载库
} /**
* 把两个java中的int传递给c语言, c语言处理完毕后,把相加的结果返回给java
*/
public native int add(int x,int y);
/**
* 静态方法 c语言处理减法运算 返回结果给java
*/
public static native int sub(int x,int y);
/**
* 把java中的string传递给c语言, c语言获取到java中的string之后 ,在string后面添加 一个hello 字符串
*/
public native String getHelloString(String s);
/**
* 把java中的一个int数组 传递给c语言,c语言处理完毕这个java数组
* 把int数组中的每一个元素+10,然后把结果返回给java
*/
public native int[] getIntArr(int[] iNum); }

2.创建完毕后找到它.class位置,生成JNI样式的头文件:

com_pl_ndkpassdata_DataProvider.h:

/* DO NOT EDIT THIS FILE - it is machine generated */
#include <jni.h>
/* Header for class com_pl_ndkpassdata_DataProvider */ #ifndef _Included_com_pl_ndkpassdata_DataProvider
#define _Included_com_pl_ndkpassdata_DataProvider
#ifdef __cplusplus
extern "C" {
#endif
/*
* Class: com_pl_ndkpassdata_DataProvider
* Method: add
* Signature: (II)I
*/
JNIEXPORT jint JNICALL Java_com_pl_ndkpassdata_DataProvider_add
(JNIEnv *, jobject, jint, jint); /*
* Class: com_pl_ndkpassdata_DataProvider
* Method: sub
* Signature: (II)I
*/
JNIEXPORT jint JNICALL Java_com_pl_ndkpassdata_DataProvider_sub
(JNIEnv *, jclass, jint, jint); /*
* Class: com_pl_ndkpassdata_DataProvider
* Method: getHelloString
* Signature: (Ljava/lang/String;)Ljava/lang/String;
*/
JNIEXPORT jstring JNICALL Java_com_pl_ndkpassdata_DataProvider_getHelloString
(JNIEnv *, jobject, jstring); /*
* Class: com_pl_ndkpassdata_DataProvider
* Method: getIntArr
* Signature: ([I)[I
*/
JNIEXPORT jintArray JNICALL Java_com_pl_ndkpassdata_DataProvider_getIntArr
(JNIEnv *, jobject, jintArray); #ifdef __cplusplus
}
#endif
#endif

上面是用javah -jni命令生成的。

3.  在项目的根目录创建jni目录(和src,bin等目录平级);

1.拷贝com_pl_ndkpassdata_DataProvider.h到jni目录下;

2.在jni目录下创建Android.mk文件和passdata.c文件;(关于c语言的.c和.h文件的区别是 :函数定义要放在.c中,而.h只做声明.)

下面是Android.mk的内容:

 LOCAL_PATH := $(call my-dir)

   include $(CLEAR_VARS)

   LOCAL_MODULE    := passdata
LOCAL_SRC_FILES := passdata.c
#liblog.c
LOCAL_LDLIBS += -llog
include $(BUILD_SHARED_LIBRARY)

下面是passdata.c

#include <stdio.h>
#include <jni.h>
#include <stdlib.h>
#include "com_pl_ndkpassdata_DataProvider.h"
#include <android/log.h>
#define LOG_TAG "System.out.c"
#define LOGD(...) __android_log_print(ANDROID_LOG_DEBUG, LOG_TAG, __VA_ARGS__)
#define LOGI(...) __android_log_print(ANDROID_LOG_INFO, LOG_TAG, __VA_ARGS__) /**
* 用于把java中的String转换成c可用的指向字符的指针类型
*/
/**
* <jni.h>中的指针函数:
* jclass (*FindClass)(JNIEnv*, const char*); //拿到一个类
* jstring (*NewStringUTF)(JNIEnv*, const char*);//拿到一个String对象
* jmethodID (*GetMethodID)(JNIEnv*, jclass, const char*, const char*);//拿到类的方法
* jobject (*CallObjectMethod)(JNIEnv*, jobject, jmethodID, ...);//调用类的方法
* jsize (*GetArrayLength)(JNIEnv*, jarray);//拿到数组的长度
* jbyte* (*GetByteArrayElements)(JNIEnv*, jbyteArray, jboolean*); //转换成c可用的byte指针类型
*/
char* Jstring2CStr(JNIEnv * env,jstring s){ //JNIEnv*:java虚拟机的结构体c语言实现的指针,包含的有很多jni方法
char* rtn = NULL;
jclass classstring = (*env)->FindClass(env,"java/lang/String"); //调用java中的String类
jstring strencode = (*env)->NewStringUTF(env,"GB2312"); //拿到编码格式
//拿到方法 。参数2:指定类,参数3:方法名,参数4:(Ljava/lang/String;)是getBytes的参数;[B代表是返回值是一个byte数组
jmethodID mid = (*env)->GetMethodID(env,classstring,"getBytes","(Ljava/lang/String;)[B");
jbyteArray barr = (jbyteArray)(*env)->CallObjectMethod(env,s,mid,strencode);//相当于调用java的getBytes
//拿到byte数组的长度
jsize size = (*env)->GetArrayLength(env,barr);
jbyte* ba = (*env)->GetByteArrayElements(env,barr,JNI_FALSE);
if(size > 0){
rtn = (char*)malloc(size+1); //申请一块内存 //+"\0"
memcpy(rtn,ba,size); //拷贝字符串
rtn[size]=0; //'\0'
}
(*env)->ReleaseByteArrayElements(env,barr,ba,0);//释放内存空间
return rtn;
}
JNIEXPORT jint JNICALL Java_com_pl_ndkpassdata_DataProvider_add
(JNIEnv * env, jobject obj, jint x, jint y){
LOGD("add()x:%d,y:%d",x,y);
return x+y;
} JNIEXPORT jint JNICALL Java_com_pl_ndkpassdata_DataProvider_sub
(JNIEnv * env, jclass clazz, jint x, jint y){
LOGD("sub()x:%d,y:%d",x,y);
return x-y;
} JNIEXPORT jstring JNICALL Java_com_pl_ndkpassdata_DataProvider_getHelloString
(JNIEnv * env, jobject obj, jstring s){
char* cs = Jstring2CStr(env,s);
LOGD("cs=%s",cs);
char carr[7] = {' ','h','e','l','l','o','\0'};
strcat(cs,carr); //拼接
LOGD("new cs=%s",cs);
LOGD("end getHelloString()");
return (*env)->NewStringUTF(env,cs);
} //jsize (*GetArrayLength)(JNIEnv*, jarray); //拿到数组长度
// jint* (*GetIntArrayElements)(JNIEnv*, jintArray, jboolean*) //转换成c可用的int指针类型
JNIEXPORT jintArray JNICALL Java_com_pl_ndkpassdata_DataProvider_getIntArr
(JNIEnv * env, jobject obj, jintArray jarr){
jsize len = (*env)->GetArrayLength(env,jarr);
LOGD("len=%d",len);
jint* carr = (*env)->GetIntArrayElements(env,jarr,JNI_FALSE);
int i;
for(i=0;i<len;i++){
*(carr+i)+=10;
}
LOGD("end getIntArr()");
return jarr;
}

完成映射实现,使用Cygwin生成.so库文件.基本完成了,别忘了clean下项目。

基本到这做完了。下面是测试代码:

MainActivity.java

package com.pl.ndkpassdata;

import android.os.Bundle;
import android.app.Activity;
import android.view.View;
import android.view.View.OnClickListener;
import android.widget.Button;
import android.widget.Toast; public class MainActivity extends Activity implements OnClickListener{
private Button bt1,bt2,bt3,bt4;
private DataProvider provider;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.main);
bt1 = (Button) this.findViewById(R.id.bt1);
bt2 = (Button) this.findViewById(R.id.bt2);
bt3 = (Button) this.findViewById(R.id.bt3);
bt4 = (Button) this.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(10, 10);
Toast.makeText(this, "相加的结果:"+result, 1).show();
break; case R.id.bt2:
String str = provider.getHelloString("pengliang");
Toast.makeText(this, str, 1).show();
break;
case R.id.bt3:
int[] arr = {7,8,9,10,6};
provider.getIntArr(arr);
String arrtostr ="[";
for(int i=0;i<arr.length;i++){
arrtostr+=arr[i]+",";
}
arrtostr +="]";
Toast.makeText(this, arrtostr, 1).show();
break;
case R.id.bt4:
int subresult = DataProvider.sub(100, 24);
Toast.makeText(this, "相减的结果:"+subresult, 1).show();
break;
}
}
}

main.xml:

<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>

[置顶] JNI之java传递数据给c语言的更多相关文章

  1. Android NDK 开发(四)java传递数据到C【转】

    转载请注明出处:http://blog.csdn.net/allen315410/article/details/41845701 前面几篇文章介绍了Android NDK开发的简单概念.常见错误及处 ...

  2. [置顶] think in java interview-高级开发人员面试宝典(二)

    从现在开始,以样题的方式一一列出各种面试题以及点评,考虑到我在前文中说的,对于一些大型的外资型公司,你将会面临全程英语面试,因此我在文章中也会出现许多全英语样题. 这些题目来自于各个真实的公司,公司名 ...

  3. [置顶] NS2中对TCP数据包和ACK包的TCP Sink类的主要实现代码详尽剖析--吐血放送

    NS2中对TCP数据包和ACK包的TCP Sink类的主要实现代码详尽剖析,限于个人水平,如有错误请留言指出! TcpSink类的recv()方法: void TcpSink::recv(Packet ...

  4. 【Redis】解析Redis和Java传递数据

    在Java中使用Redis之前需要导入 jedis.jar 包,由于Redis是基于key-value进行数据存储,java中的数据存储到Redis中有许多方式,这里笔者介绍采用JSON字符串和对象序 ...

  5. [置顶] 我的 Java 后端书架 (2016 年暖冬版)

    转自:  http://calvin1978.blogcn.com/articles/bookshelf16.html 我的 Java 后端书架 (2016 年暖冬版) 本书架主要针对 Java 后端 ...

  6. [置顶] DataGridView控件---绑定数据方法

             DataGridView控件是在windows应用程中显示数据最好的方式,它只需要几行简短的代码就可以把数据显示给用户,同时又支持增.删.改操作.今天将自己总结的增加数据的方法总结分 ...

  7. [置顶] think in java interview-高级开发人员面试宝典(八)

    面经出了7套,收到许多读者的Email,有许多人说了,这些基础知识是不是为了后面进一步的”通向架构师的道路“做准备的? 对的,你们没有猜错,就是这样的,我一直在酝酿后面的”通向架构师的道路“如何开章. ...

  8. [置顶] think in java interview-高级开发人员面试宝典(一)

    “生死六重门” 无论你是在职,非在职,高级工程师,工程师,架构师,如果你正在面试阶段,请看完此文! 相信这篇文章对你的职业生涯和阶值观会造成重大的改变! 如果你是一名PM或者是管理者正在物色合适的开发 ...

  9. [置顶] think in java interview-高级开发人员面试宝典代码示例

    下载资源地址为: http://download.csdn.net/detail/lifetragedy/6379755 这是think in java interview中的代码示例,包括JAVA基 ...

随机推荐

  1. Android学习笔记(十五)——碎片的生命周期(附源代码)

    碎片的生命周期 点击下载源代码 与活动类似.碎片具有自己的生命周期.理解了碎片的生命周期后.我们能够在碎片被销毁时正确地保存事实上例,在碎片被重建时将其还原到前一个状态. 1.使用上一篇的项目Frag ...

  2. 说说Xcode4中xib绑定的原理

    最开始的是时候始终没有弄明白xib文件中的绑定关系.经过一周的开发体验终于有一些理解与收获. Xib文件就是MVC模式中的View这个层的界面显示布局的信息.即类似Asp.net的aspx文件或者ja ...

  3. Port 8081 already in use, packager is either not running or not running correctly

    运行 react_native 时发生这个错误,解决办法 关掉端口8081对应的进程 1.打开终端,输入命令:lsof -i:8081 2.此时提示: COMMAND   PID    USER   ...

  4. 新到的Mac配置Java开发环境

    今天Mac到手,需要配置一些用到的开发环境,在这里做一些纪录. 1. 下载Eclipse,地址:http://www.eclipse.org/downloads/,因为个人需求,所以下载的是Java ...

  5. BZOJ 1642: [Usaco2007 Nov]Milking Time 挤奶时间( dp )

    水dp 先按开始时间排序 , 然后dp. dp( i ) 表示前 i 个时间段选第 i 个时间段的最优答案 , 则 dp( i ) = max( dp( j ) ) + w_i ( 0 < j ...

  6. BZOJ 1231: [Usaco2008 Nov]mixup2 混乱的奶牛( dp )

    状压dp dp( x , S ) 表示最后一个是 x , 当前选的奶牛集合为 S , 则状态转移方程 : dp( x , S ) =  Σ dp( i , S - { i } )  ( i ∈ S , ...

  7. linux ln 命令(转载)

    ln是linux中又一个非常重要命令,它的功能是为某一个文件在另外一个位置建立一个同不的链接,这个命令最常用的参数是-s,具体用法是:ln –s 源文件 目标文件. 当我们需要在不同的目录,用到相同的 ...

  8. php定时自动执行 需启动第一次

    1 2 3 4 5 6 7 8 9 10 11 12 <? ignore_user_abort(); //即使Client断开(如关掉浏览器),PHP脚本也可以继续执行. set_time_li ...

  9. werkzeug中reloader的实现

    在用flask开发时,如果把use_reloader设为True(debug设为True也能实现),那当你修改了app代码或调用环境发生改变时,服务器会自动重启,如下 * Detected chang ...

  10. shell实例浅谈之六文件特定行打印的多种方法

    一.问题 Sed和AWK在处理文件方面有很强的优势,还有head和tail等文件处理工具的使用,grep也可实现文本的搜索.上述命令都可以在后面直接加文件名,不需要在前面使用cat添加管道,cat会影 ...