Android之NDK编程(JNI)
转自:http://www.cnblogs.com/xw022/archive/2011/08/18/2144621.html
NDK编程入门--C回调JAVA方法
一、主要流程
1、 新建一个测试类TestProvider.java
a) 该类提供了2个方法
b) 一个静态的方法,一个非静态的方法
2、 JNI中新建Provider.c
a) 该文件中需要把Java中的类TestProvider映射到C中
b) 把TestProvider的两个方法映射到C中
c) 新建TestProvider 对象
d) 调用两个方法
3、 Android 上层 调用 JNI层
4、 JNI层调用C层
5、 C 层调用 Java 方法
二、设计实现
1. 关键代码说明
C中定义映射的类、方法、对象
jclass TestProvider;
jobject mTestProvider;
jmethodID getTime;
jmethodID sayHello;
C 中映射 类
TestProvider = (*jniEnv)->FindClass(jniEnv,"com/duicky/TestProvider");
C中新建对象
jmethodID construction_id = (*jniEnv)->GetMethodID(jniEnv, TestProvider,"<init>", "()V");
TestProvider mTestProvider = (*jniEnv)->NewObject(jniEnv, TestProvider,construction_id);
C 中映射方法
静态:
getTime = (*jniEnv)->GetStaticMethodID(jniEnv, TestProvider, "getTime","()Ljava/lang/String;");
非静态:
sayHello = (*jniEnv)->GetMethodID(jniEnv, TestProvider, "sayHello","(Ljava/lang/String;)V");
C 中调用 Java的 方法
静态:
(*jniEnv)->CallStaticObjectMethod(jniEnv, TestProvider, getTime);
非静态:
(*jniEnv)->CallVoidMethod(jniEnv, mTestProvider, sayHello,jstrMSG);
注意 GetXXXMethodID 和 CallXXXMethod 。
第一个XXX 表示的是映射方法的类型,如: 静态 跟非静态
第二个 XXX 表示 调用方法的返回值 ,如:Void,Object,等等。(调用静态方法的时候Call后面要加Static)
详细 映射方法 和 调用方法 请参考JNI文档
3.Java 上层 关键代码
TestProvider.Java 的两个方法
package com.duicky; publicclass TestProvider { publicstatic String getTime() {
LogUtils.printWithSystemOut( "Call From C Java Static Method" );
LogUtils.toastMessage(MainActivity.mContext, "Call From C Java Static Method" );
return String.valueOf(System.currentTimeMillis());
} publicvoid sayHello(String msg) {
LogUtils.printWithSystemOut("Call From C Java Not Static Method :"+ msg);
LogUtils.toastMessage(MainActivity.mContext, "Call From C Java Not Static Method :"+ msg);
} }
3、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_04
LOCAL_SRC_FILES := \
CToJava.c \
Provider.c include $(BUILD_SHARED_LIBRARY)
4、 JNI文件夹下文件
provider.h
#include <string.h>
#include <jni.h> void GetTime() ;
void SayHello();
Provider.c
#include "Provider.h"
#include <android/log.h> extern JNIEnv* jniEnv; jclass TestProvider;
jobject mTestProvider;
jmethodID getTime;
jmethodID sayHello; int GetProviderInstance(jclass obj_class); /**
* 初始化 类、对象、方法
*/
int InitProvider() { __android_log_print(ANDROID_LOG_INFO, "JNIMsg", "InitProvider Begin 1" ); if(jniEnv == NULL) {
return0;
} if(TestProvider == NULL) {
TestProvider = (*jniEnv)->FindClass(jniEnv,"com/duicky/TestProvider");
if(TestProvider == NULL){
return-1;
}
__android_log_print(ANDROID_LOG_INFO, "JNIMsg", "InitProvider Begin 2 ok" );
} if (mTestProvider == NULL) {
if (GetProviderInstance(TestProvider) !=1) {
(*jniEnv)->DeleteLocalRef(jniEnv, TestProvider);
return-1;
}
__android_log_print(ANDROID_LOG_INFO, "JNIMsg", "InitProvider Begin 3 ok" );
} if (getTime == NULL) {
getTime = (*jniEnv)->GetStaticMethodID(jniEnv, TestProvider, "getTime","()Ljava/lang/String;");
if (getTime == NULL) {
(*jniEnv)->DeleteLocalRef(jniEnv, TestProvider);
(*jniEnv)->DeleteLocalRef(jniEnv, mTestProvider);
return-2;
}
__android_log_print(ANDROID_LOG_INFO, "JNIMsg", "InitProvider Begin 4 ok" );
} if (sayHello == NULL) {
sayHello = (*jniEnv)->GetMethodID(jniEnv, TestProvider, "sayHello","(Ljava/lang/String;)V");
if (sayHello == NULL) {
(*jniEnv)->DeleteLocalRef(jniEnv, TestProvider);
(*jniEnv)->DeleteLocalRef(jniEnv, mTestProvider);
(*jniEnv)->DeleteLocalRef(jniEnv, getTime);
return-3;
}
__android_log_print(ANDROID_LOG_INFO, "JNIMsg", "InitProvider Begin 5 ok" );
} __android_log_print(ANDROID_LOG_INFO, "JNIMsg", "InitProvider Begin 6" );
return1; } int GetProviderInstance(jclass obj_class) { if(obj_class == NULL) {
return0;
} jmethodID construction_id = (*jniEnv)->GetMethodID(jniEnv, obj_class,
"<init>", "()V"); if (construction_id ==0) {
return-1;
} mTestProvider = (*jniEnv)->NewObject(jniEnv, obj_class,
construction_id); if (mTestProvider == NULL) {
return-2;
} return1;
} /**
* 获取时间 ---- 调用 Java 方法
*/
void GetTime() {
if(TestProvider == NULL || getTime == NULL) {
int result = InitProvider();
if (result !=1) {
return;
}
} jstring jstr = NULL;
char* cstr = NULL;
__android_log_print(ANDROID_LOG_INFO, "JNIMsg", "GetTime Begin" );
jstr = (*jniEnv)->CallStaticObjectMethod(jniEnv, TestProvider, getTime);
cstr = (char*) (*jniEnv)->GetStringUTFChars(jniEnv,jstr, 0);
__android_log_print(ANDROID_LOG_INFO, "JNIMsg", "Success Get Time from Java , Value = %s",cstr );
__android_log_print(ANDROID_LOG_INFO, "JNIMsg", "GetTime End" ); (*jniEnv)->ReleaseStringUTFChars(jniEnv, jstr, cstr);
(*jniEnv)->DeleteLocalRef(jniEnv, jstr);
} /**
* SayHello ---- 调用 Java 方法
*/
void SayHello() {
if(TestProvider == NULL || mTestProvider == NULL || sayHello == NULL) {
int result = InitProvider() ;
if(result !=1) {
return;
}
} jstring jstrMSG = NULL;
jstrMSG =(*jniEnv)->NewStringUTF(jniEnv, "Hi,I'm From C");
__android_log_print(ANDROID_LOG_INFO, "JNIMsg", "SayHello Begin" );
(*jniEnv)->CallVoidMethod(jniEnv, mTestProvider, sayHello,jstrMSG);
__android_log_print(ANDROID_LOG_INFO, "JNIMsg", "SayHello End" ); (*jniEnv)->DeleteLocalRef(jniEnv, jstrMSG);
}
CToJava.c
#include <string.h>
#include <android/log.h>
#include <jni.h>
#include "Provider.h" JNIEnv* jniEnv; /**
* Java 中 声明的native getTime 方法的实现
*/
void Java_com_duicky_MainActivity_getTime(JNIEnv* env, jobject thiz)
{ if(jniEnv == NULL) {
jniEnv = env;
} GetTime();
} /**
* Java 中 声明的native sayHello 方法的实现
*/
void Java_com_duicky_MainActivity_sayHello(JNIEnv* env, jobject thiz)
{
if (jniEnv == NULL) {
jniEnv = env;
} SayHello();
}
三.Java 方法映射到C中的签名
签名是由两部分组成,"()" 里面代表的是方法的参数,后面外面的部分代表的是该方法的返回值
public int test3(int i) { return i;} (I)I
基本数据类型对应关系如表:
其实仔细看看发现就是对应java类型的首字母拉, Boolean 比较特殊, 对应的是 Z , Long 对应J
引用数据类型:比较麻烦点,以“L”开头,以“;”结束,中间对应的是该类型的路径
如:String : Ljava/lang/String;
Object: Ljava/lang/Object;
自定义类 Cat 对应 package com.duicky;
Cat : Lcom/duicky/Cat;
数组表示: 数组表示的时候以“[” 为标志,一个“[”表示一维数组
如:int [] :[I
Long[][] : [[J
Object[][][] : [[[Ljava/lang/Object;
字符 Java类型 C类型
V void void
Z jboolean boolean
I jint int
J jlong long
D jdouble double
F jfloat float
B jbyte byte
C jchar char
S jshort short
数组则以"["开始,用两个字符表示
[I jintArray int[]
[F jfloatArray float[]
[B jbyteArray byte[]
[C jcharArray char[]
[S jshortArray short[]
[D jdoubleArray double[]
[J jlongArray long[]
[Z jbooleanArray boolean[]
输入命令: javap –s 加上你要查看方法签名的 类 名
如: javap –s Test 结果就显示出我们想要的签名了。、
四、C调用Java注意点
- 映射java 方法时 对应的签名
getTime = (*jniEnv)->GetStaticMethodID(jniEnv, TestProvider, "getTime","()Ljava/lang/String;");
映射方法的时候需要区别静态和非静态GetStaticMethodID,GetMethodID
调用的时候也需要区分CallStaticObjectMethod,CallVoidMethod 而且还需要区分返回值类型
Android之NDK编程(JNI)的更多相关文章
- Android Studio NDK编程初探
继上一篇学习了如何使用NDK编译FFMPEG后,接下来就是要学习如何在Android Studio中使用了. 经过参考和一系列的摸索,记录下具体步骤. 创建C++ Support的Android St ...
- Android Studio Ndk 编程
如今开发Android程序基本都已经从Eclipse转到了Android Studio了, 近期项目需求, 须要用到ndk编程, 于是就折腾了一下. 开发环境 Android Studio 1.5.1 ...
- Android Studio NDK编程-环境搭建及Hello!
一,下载 安装android-ndk开发包 NDK各个版本链接二,新建项目NDKDemo,选择空Activity就可以:(注:Android studio 2.2,可通过SDK Tools 添加LLD ...
- Android Studio NDK编程-环境搭建及Hello!
一,下载 安装android-ndk开发包 NDK各个版本链接二,新建项目NDKDemo,选择空Activity就可以:(注:Android studio 2.2,可通过SDK Tools 添加LLD ...
- [转]Android通过NDK调用JNI,使用opencv做本地c++代码开发配置方法
原文地址:http://blog.csdn.net/watkinsong/article/details/9849973 有一种方式不需要自己配置所有的Sun JDK, Android SDK以及ND ...
- Android平台NDK编程
转自:http://blog.csdn.net/wangbin_jxust/article/details/37389383 之前在进行cocos2dx开发时,已经详细介绍了如何将win32的c++代 ...
- Android Studio NDK开发-JNI调用Java方法
相对于NDK来说SDK里面有更多API可以调用,有时候我们在做NDK开发的时候,需要在JNI直接Java中的方法和变量,比如callback,系统信息等.... 如何在JNI中调用Java方法呢?就需 ...
- NDK编程jni学习入门,声明native方法,使其作为java与c的交互接口
首先,新建工程,简历一个jave类,在其中声明native方法,关键字为native,表面这个方法是从java以为的语言实现. 其次,要实用javac编译此java文件(javac是jdk中的命令,需 ...
- 两分钟学会Android平台NDK编程(无须Eclipse和cygwin,可使用命令行打包多个so)
版权声明:本文为博主原创文章,未经博主同意不得转载. https://blog.csdn.net/wangbin_jxust/article/details/37389383 之前在进行cocos2d ...
随机推荐
- yabeblog.me 关于Tomcat7部署 一台机器部署两个项目,一个用域名访问,一个用IP访问
该内容来自 http://www.yabeBlog.me,转载请说明出处. 1.使用IP访问的项目放在Tomcat7 的webapps目录下面:比如:AAA 2.使用域名访问的项目放在Tomcat7的 ...
- 博主教你制作类似9patch效果的iOS图片拉伸
下面张图片,本来是设计来做按钮背景的: button.png,尺寸为:24x60 现在我们把它用作为按钮背景,按钮尺寸是150x50: // 得到view的尺寸 CGSize viewSize = ...
- verilogHDL设计中的同步时序逻辑
引用自夏宇闻教授 1.同步时序逻辑: 是指表示状态的寄存器组的值只能在唯一确定的触发条件发生改变. 只能由时钟的正跳变沿或者负跳变沿触发的状态机就是一例,always@(posedge clk). 1 ...
- C++中的运算符优先级
1 () [] . ->2 ! ~ -(负号) ++ -- &(取变量地址)* (type)(强制类型) sizeof 3 * / % 4 + ...
- 配置ASP.NET Nhibernate
web.config:配置sql server数据库 <configuration> <configSections> <!--NHibernate Section--& ...
- C#系统缓存全解析
原文:http://blog.csdn.net/wyxhd2008/article/details/8076105 目录(?)[-] 系统缓存的概述 页面输出缓存 页面局部缓存 文件缓存依赖 数据库缓 ...
- Java 集合类(一)
今天我们先讲一下Collection: Collection和Collections的区别: java.util.Collection是一种java集合接口,它提供了对集合对象的基本操作通用接口方法, ...
- git@oschina.net源代码管理使用日记
git的优势: 1 可以创建分支: 2 版本控制是基于每一次提交的,而不需要考虑每次提交了多少个文件. 下载: 下载网址为:http://git-scm.com/download,根据您的操作系统选择 ...
- SVN--VisualSVN server 服务端和 TortoiseSVN客户端的基础使用
前言 在上一文http://www.cnblogs.com/wql025/p/5177699.html中,我们讲到了使用SVN的第一步,即下载.安装SVN的服务端软件--VisualSVN serve ...
- Mongo:将查询结果转换为自定义类
1.自定义类 public class MyClass { public string Name { get; set; } public int Corners { get; set; } } 2. ...