【转】基于 Android NDK 的学习之旅-----数据传输(引用数据类型)
原文网址:http://www.cnblogs.com/luxiaofeng54/archive/2011/08/20/2147086.html
基于 Android NDK 的学习之旅-----数据传输二(引用数据类型)(附源码)
基于 Android NDK 的学习之旅-----数据传输(引用数据类型)
接着上篇文章继续讲。主要关于引用类型的数据传输,本文将介绍字符串传输和自定义对象的传输。
1、主要流程
1、 String 字符串传输
a) 上层定义一个native的方法,需要一个String 参数 ,返回一个String
b) JNI对应上层的方法,打印出上层传输下来的String数据,并返回处理String数据
c) 上层 收到 native 方法 返回的 值,在UI中显示出来
2、 自定义对象的传输
a) 自定义一个对象Person
b) 上层定义一个native方法,参数Person,返回值Person
c) JNI接收对象,打印出相关信息数据
d) JNI 修改Person 对象数据,并返回到上层
e) 上层接收到数据后 在UI显示出来
2设计实现
1、 界面设计如下:
老老样子,很搓,嘿嘿
代码不在这贴出了,有需要的兄弟直接到文章结束部分下载。
2、 关键代码说明
Java 上层:
public native String transferString(String mStrMSG); public native Object transferPerson(Person mPerson); |
Person.java
package com.duicky; /** * * * @author luxiaofeng <454162034@qq.com> * */ public class Person { private String name; private int age; public Person() { name = "" ; age = 0 ; } public String getName() { return name; } public void setName(String name) { this .name = name; } public int getAge() { return age; } public void setAge( int age) { this .age = age; } @Override public String toString() { return "Person [name=" + name + ", age=" + age + "]" ; } } |
定义两个native方法, 第一个是 用来 测试传输字符串的的,第二个是用来测试传输自定义对象的。
Android.mk 文件
LOCAL_PATH := $(call my-dir) include $(CLEAR_VARS) LOCAL_C_INCLUDES := $(LOCAL_PATH)/include LOCAL_LDLIBS += -L$(SYSROOT)/usr/lib -llog LOCAL_MODULE := NDK_07 LOCAL_SRC_FILES := \ TransmissionPerson.c \ TransmissionString.c include $(BUILD_SHARED_LIBRARY) |
老样子,不说了,你懂的。 如果不懂,嘎嘎,那就请点击Android.mk 文件 简介咯
JNI 中间层
TransmissionString.c // String 传输
#include <string.h> #include <jni.h> #include <android/log.h> JNIEnv* jniEnv; //---------------------------------------------------------------- //---------------------------------------------------------------- //---------------------------------------------------------------- jstring Java_com_duicky_Transmission_transferString( JNIEnv* env,jobject thiz,jstring msg ) { if (jniEnv == NULL) { jniEnv = env; } char data[128]; memset (data, 0, sizeof (data)); char *c_msg = NULL; c_msg = ( char *)(*jniEnv)->GetStringUTFChars(jniEnv, msg, 0); __android_log_print(ANDROID_LOG_INFO, "JNIMsg" , "C JNI ---- > %s" ,c_msg); return (*jniEnv)->NewStringUTF(jniEnv, "This is send by C JNI" ); } |
TransmissionPerson.c // 自定义对象 传输
#include <string.h> #include <jni.h> #include <android/log.h> extern JNIEnv* jniEnv; jclass Person; jobject mPerson; jmethodID getName; jmethodID setName; jmethodID getAge; jmethodID setAge; jmethodID toString; int InitPerson(); void ToString(); void GetName(); void GetAge(); void SetName(); void SetAge(); //---------------------------------------------------------------- //---------------------------------------------------------------- //---------------------------------------------------------------- jobject Java_com_duicky_Transmission_transferPerson( JNIEnv* env,jobject thiz,jobject person ) { if (jniEnv == NULL) { jniEnv = env; } if (Person == NULL || getName == NULL || setName == NULL || getAge == NULL || setAge == NULL || toString == NULL) { if (1 != InitPerson()) { return NULL; } } mPerson = person; if (mPerson == NULL) { return NULL; } GetName(); GetAge(); ToString(); __android_log_print(ANDROID_LOG_INFO, "JNIMsg" , "Begin Modify mPerson .... " ); SetName(); SetAge(); ToString(); return mPerson; } //---------------------------------------------------------------- //---------------------------------------------------------------- //---------------------------------------------------------------- /** * 初始化 类、方法 */ int InitPerson() { __android_log_print(ANDROID_LOG_INFO, "JNIMsg" , "InitPerson Begin " ); if (jniEnv == NULL) { return 0; } if (Person == NULL) { Person = (*jniEnv)->FindClass(jniEnv, "com/duicky/Person" ); if (Person == NULL){ return -1; } __android_log_print(ANDROID_LOG_INFO, "JNIMsg" , "InitPerson Begin 2 ok" ); } if (getName == NULL) { getName = (*jniEnv)->GetMethodID(jniEnv, Person, "getName" , "()Ljava/lang/String;" ); if (getName == NULL) { (*jniEnv)->DeleteLocalRef(jniEnv, Person); return -2; } __android_log_print(ANDROID_LOG_INFO, "JNIMsg" , "InitPerson Begin 4 ok" ); } if (setName == NULL) { setName = (*jniEnv)->GetMethodID(jniEnv, Person, "setName" , "(Ljava/lang/String;)V" ); if (setName == NULL) { (*jniEnv)->DeleteLocalRef(jniEnv, Person); (*jniEnv)->DeleteLocalRef(jniEnv, getName); return -2; } __android_log_print(ANDROID_LOG_INFO, "JNIMsg" , "InitPerson Begin 4 ok" ); } if (getAge == NULL) { getAge = (*jniEnv)->GetMethodID(jniEnv, Person, "getAge" , "()I" ); if (getAge == NULL) { (*jniEnv)->DeleteLocalRef(jniEnv, Person); (*jniEnv)->DeleteLocalRef(jniEnv, getName); (*jniEnv)->DeleteLocalRef(jniEnv, setName); return -2; } __android_log_print(ANDROID_LOG_INFO, "JNIMsg" , "InitPerson Begin 4 ok" ); } if (setAge == NULL) { setAge = (*jniEnv)->GetMethodID(jniEnv, Person, "setAge" , "(I)V" ); if (setAge == NULL) { (*jniEnv)->DeleteLocalRef(jniEnv, Person); (*jniEnv)->DeleteLocalRef(jniEnv, getName); (*jniEnv)->DeleteLocalRef(jniEnv, setName); (*jniEnv)->DeleteLocalRef(jniEnv, getAge); return -2; } __android_log_print(ANDROID_LOG_INFO, "JNIMsg" , "InitPerson Begin 4 ok" ); } if (toString == NULL) { toString = (*jniEnv)->GetMethodID(jniEnv, Person, "toString" , "()Ljava/lang/String;" ); if (toString == NULL) { (*jniEnv)->DeleteLocalRef(jniEnv, Person); (*jniEnv)->DeleteLocalRef(jniEnv, getName); (*jniEnv)->DeleteLocalRef(jniEnv, setName); (*jniEnv)->DeleteLocalRef(jniEnv, getAge); (*jniEnv)->DeleteLocalRef(jniEnv, setAge); return -2; } __android_log_print(ANDROID_LOG_INFO, "JNIMsg" , "InitPerson Begin 4 ok" ); } __android_log_print(ANDROID_LOG_INFO, "JNIMsg" , "InitPerson End" ); return 1; } /** * GetName 对应Person的getName方法 */ void GetName() { if (Person == NULL || getName == NULL) { if (1 != InitPerson()){ return ; } } jstring jstr = NULL; char * cstr = NULL; //调用方法 jstr = (*jniEnv)->CallObjectMethod(jniEnv, mPerson, getName); cstr = ( char *) (*jniEnv)->GetStringUTFChars(jniEnv,jstr, 0); __android_log_print(ANDROID_LOG_INFO, "JNIMsg" , "getName ---- > %s" ,cstr ); //释放资源 (*jniEnv)->ReleaseStringUTFChars(jniEnv, jstr, cstr); (*jniEnv)->DeleteLocalRef(jniEnv, jstr); } /** * GetAge 对应Person的getName方法 */ void GetAge() { if (Person == NULL || getName == NULL) { if (1 != InitPerson()){ return ; } } //调用方法 jint age = (*jniEnv)->CallIntMethod(jniEnv, mPerson, getAge); __android_log_print(ANDROID_LOG_INFO, "JNIMsg" , "getAge ---- > %d" ,age ); } /** * SetName 对应Person的setName方法 */ void SetName() { if (Person == NULL || setName == NULL) { if (1 != InitPerson()){ return ; } } jstring jstr = (*jniEnv)->NewStringUTF(jniEnv, "Modify Name" ); //调用方法 (*jniEnv)->CallVoidMethod(jniEnv, mPerson, setName,jstr); (*jniEnv)->DeleteLocalRef(jniEnv, jstr); } int age = 20; /** * SetAge 对应Person的setAge方法 */ void SetAge() { if (Person == NULL || setAge == NULL) { if (1 != InitPerson()){ return ; } } //调用方法 (*jniEnv)->CallVoidMethod(jniEnv, mPerson, setAge,age++); } /** * ToString 对应 Person 的 toString 方法 , 打印出相关信息 */ void ToString() { if (Person == NULL || toString == NULL) { if (1 != InitPerson()){ return ; } } jstring jstr = NULL; char * cstr = NULL; //调用方法 jstr = (*jniEnv)->CallObjectMethod(jniEnv, mPerson, toString); cstr = ( char *) (*jniEnv)->GetStringUTFChars(jniEnv,jstr, 0); __android_log_print(ANDROID_LOG_INFO, "JNIMsg" , "C JNI toString ---- > %s" ,cstr ); (*jniEnv)->ReleaseStringUTFChars(jniEnv, jstr, cstr); (*jniEnv)->DeleteLocalRef(jniEnv, jstr); } |
3、运行结果
A、测试 String传输 : 点击 第一个按钮,看UI显示 和 LogCat 打印信息
1、 JNI 收到 Java 的信息
2、 Java 收到JNI 的信息
3、 UI 显示信息
B、测试 自定义对象传输:点击 第二个按钮,看UI显示 和 LogCat 打印信息
1、JNI 收到 Person 对象的 数据(Java 写死了发送 name:duicky ,age:10)
2、JNI 通过 setName,SetAge改变Person 后的数据
3、Java 上层 接收到的数据
4、 UI 显示信息
以上就是 Java --- JNI String和 自定义对象 传输的 小例子 , 其他 引用数据类型和Java自定义 都可以仿照上面的做法传输。
还有一种方式 :大数据传输可以通过保存到文件实现。如:上层保存数据到文件然后C JNI 层直接读取文件 或者 C JNI 层保存数据到文件 由 上层读取 。
有不理解的兄弟请留言,个人技术有限,有讲错的地方请大牛们指出,讲的不够全面的请多多包涵,谢谢,
点击下载源码 数据传输二
本文出自 duicky博客 , 转载请注明出处
http://www.cnblogs.com/luxiaofeng54/archive/2011/08/20/2147086.html
【转】基于 Android NDK 的学习之旅-----数据传输(引用数据类型)的更多相关文章
- 基于 Android NDK 的学习之旅-----数据传输(引用数据类型)
接着上篇文章继续讲.主要关于引用类型的数据传输,本文将介绍字符串传输和自定义对象的传输. 1.主要流程 1. String 字符串传输 a) 上层定义一个native的方法,需要一个 ...
- 基于 Android NDK 的学习之旅-----环境搭建
工欲善其事 必先利其器 , 下面介绍下 Eclipse SDK NDK Cygwin CDT 集成开发环境的搭建. 1.Android 开发环境搭建 Android开发环境搭建不是重点,相信看此文章的 ...
- 基于 Android NDK 的学习之旅-----序言
前些日子做了个Android项目, 引擎层 用C的, 准备写这个系类的文章,借此跟朋友来分享下我NDK开放的经验以及自己知识的总结和备忘.希望能给需要这方面资料的朋友提供一定的帮助. 主要涉及到: ...
- 于 Android NDK 的学习之旅-----数据传输(基本数据类型和数组传输)
之前的一些文章都有涉及到上层和中间层的数据传输,简单来说,也就是参数和返回值的使用.因为中间层要做的最多的也就是数据传输与转换,下面来介绍下这方面的知识. 数据传输可分为 基本数据类型传输 和 引用数 ...
- 基于 Android NDK 的学习之旅-----HelloWorld
Hello World作为所有编程语言的起始阶段,占据着无法改变的地位,所有中/英/法/德/美……版本的编程教材中,hello world总是作为第一个TEST记录于书本之中,所有的编程第一步就在于此 ...
- 基于 Android NDK 的学习之旅-----JNI LOG 打印
程序都是调出来的. 下面我介绍下JNI层的log打印方法的使用,类似与Android sdk提供的log 1.Android 应用层 MainActivity.java 主要功能代码 a) ...
- 基于 Android NDK 的学习之旅-----Java 调用C
随便谈谈为什么要Java调用C 吧: 我认为: 1. 有些公司开发Android项目的时候, 许多组件功能可能是C中已经实现了,所以我们没必要同样的功能又在java中实现一遍.例如我之前做的一个项目 ...
- 基于 Android NDK 的学习之旅----- C调用Java
许多成熟的C引擎要移植到Android 平台上使用 , 一般都会 提供 一些接口, 让Android sdk 和 jdk 实现. 下文将会介绍 C 如何 通过 JNI 层调用 Java 的静态和非静态 ...
- 基于 Android NDK 的学习之旅-----Android.mk 介绍
一个Android.mk file用来向编译系统描述你的源代码.具体来说:该文件是GNU Makefile的一小部分,会被编译系统解析一次或多次.你可以在每一个Android.mk file中定义一个 ...
随机推荐
- 一次优化web项目的经历记录(三)
一次优化web项目的经历记录 这段时间以来的总结与反思 前言:最近很长一段时间没有更新博客了,忙于一堆子项目的开发,严重拖慢了学习与思考的进程. 开水倒满了需要提早放下杯子,晚了就会烫手,这段时间以来 ...
- Linux网络基础
为了防止无良网站的爬虫抓取文章,特此标识,转载请注明文章出处.LaplaceDemon/SJQ. http://www.cnblogs.com/shijiaqi1066/p/3840284.html ...
- 9.22 noip模拟试题
水灾(sliker.cpp/c/pas) 1000MS 64MB 大雨应经下了几天雨,却还是没有停的样子.土豪CCY刚从外地赚完1e元回来,知道不久除了自己别墅,其他的地方都将会被洪水淹没. CCY ...
- SQL语句优化(分享)
一.操作符优化 1.IN 操作符 用IN写出来的SQL的优点是比较容易写及清晰易懂,这比较适合现代软件开发的风格.但是用IN的SQL性能总是比较低的,从Oracle执行的步骤来分析用IN的SQL与不用 ...
- copy file using FileReader/Writer.
The code below demonstates copying file using 'FileReader' and 'FileWriter'. class CopyV2 extends Ti ...
- Linux下安装Nginx1.9.3-0303(本人亲手实践)
Linux下安装Nginx1.9.3 Linux操作系统 Oel 5.8 64bit 最新版Nginx: 1.9.3 最近同事让我帮忙搞 ngix,两天时间 安装.配置搞定了.继续 Nginx 1.9 ...
- HDOJ 2036
错误代码: #include<stdio.h>#include<math.h>int main(){ int x[102],y[102]; int i,n; float s,a ...
- Singleton 模式
个人认为 Singleton 模式是设计模式中最为简单.最为常见.最容易实现,也是最应该熟悉和掌握的模式.且不说公司企业在招聘的时候为了考察员工对设计的了解和把握,考的最多的就是 Singleton ...
- overflow第一次觉得你有点可恶
今天用css做下拉菜单,因为不需要做手机自适应,再手机里看起来工整一点就行,可是列表中最后一个li的宽度撑开了父div,导致看起来很糟糕,所以给父元素加overflow:hidden:但是下拉列表也被 ...
- 关于浮动float属性和position:absolute属性的区别
最近返回头看了很多书籍,一直在纠结float属性和absolute绝对定位的区别和使用的情况,给大家分享一下自己的心得和体会吧. 1,float属性 float属性意义是让元素拜托独占一行的霸道总裁, ...