Android JNI入门第六篇——C调用Java
本篇将介绍在JNI编程中C调用Java实现。
源码下载地址:http://download.csdn.net/detail/xyz_lmn/4868265
关键代码:
java:
- public class CCallJava {
- public static String getTime() {
- Log.d("CCallJava", "Call From C Java Static Method" +String.valueOf(System.currentTimeMillis()));
- return String.valueOf(System.currentTimeMillis());
- }
- public void sayHello(String msg) {
- Log.d("CCallJava", "Call From C Java void Method" +String.valueOf(System.currentTimeMillis()));
- }
- }
- #include "TestCCallJava.h"
- #include <android/log.h>
- extern JNIEnv* jniEnv;
- jclass TestCCallJava;
- jobject mTestCCallJava;
- jmethodID getTime;
- jmethodID sayHello;
- int GetTestCCallJavaInstance(jclass obj_class);
- /**
- * 初始化 类、对象、方法
- */
- int InitTestCCallJava() {
- __android_log_print(ANDROID_LOG_INFO, "JNIMsg", "InitTestCCallJava Begin 1" );
- if(jniEnv == NULL) {
- return 0;
- }
- if(TestCCallJava == NULL) {
- TestCCallJava = (*jniEnv)->FindClass(jniEnv,"com/trunkbow/ccalljava/CCallJava");
- if(TestCCallJava == NULL){
- return -1;
- }
- __android_log_print(ANDROID_LOG_INFO, "JNIMsg", "InitTestCCallJava Begin 2 ok" );
- }
- if (mTestCCallJava == NULL) {
- if (GetTestCCallJavaInstance(TestCCallJava) != 1) {
- (*jniEnv)->DeleteLocalRef(jniEnv, TestCCallJava);
- return -1;
- }
- __android_log_print(ANDROID_LOG_INFO, "JNIMsg", "InitTestCCallJava Begin 3 ok" );
- }
- if (getTime == NULL) {
- getTime = (*jniEnv)->GetStaticMethodID(jniEnv, TestCCallJava, "getTime","()Ljava/lang/String;");
- if (getTime == NULL) {
- (*jniEnv)->DeleteLocalRef(jniEnv, TestCCallJava);
- (*jniEnv)->DeleteLocalRef(jniEnv, mTestCCallJava);
- return -2;
- }
- __android_log_print(ANDROID_LOG_INFO, "JNIMsg", "InitTestCCallJava Begin 4 ok" );
- }
- if (sayHello == NULL) {
- sayHello = (*jniEnv)->GetMethodID(jniEnv, TestCCallJava, "sayHello","(Ljava/lang/String;)V");
- if (sayHello == NULL) {
- (*jniEnv)->DeleteLocalRef(jniEnv, TestCCallJava);
- (*jniEnv)->DeleteLocalRef(jniEnv, mTestCCallJava);
- (*jniEnv)->DeleteLocalRef(jniEnv, getTime);
- return -3;
- }
- __android_log_print(ANDROID_LOG_INFO, "JNIMsg", "InitTestCCallJava Begin 5 ok" );
- }
- __android_log_print(ANDROID_LOG_INFO, "JNIMsg", "InitTestCCallJava Begin 6" );
- return 1;
- }
- int GetTestCCallJavaInstance(jclass obj_class) {
- if(obj_class == NULL) {
- return 0;
- }
- jmethodID construction_id = (*jniEnv)->GetMethodID(jniEnv, obj_class,
- "<init>", "()V");
- if (construction_id == 0) {
- return -1;
- }
- mTestCCallJava = (*jniEnv)->NewObject(jniEnv, obj_class,
- construction_id);
- if (mTestCCallJava == NULL) {
- return -2;
- }
- return 1;
- }
- /**
- * 获取时间 ---- 调用 Java 方法
- */
- void GetTime() {
- if(TestCCallJava == NULL || getTime == NULL) {
- int result = InitTestCCallJava();
- if (result != 1) {
- return;
- }
- }
- jstring jstr = NULL;
- char* cstr = NULL;
- __android_log_print(ANDROID_LOG_INFO, "JNIMsg", "GetTime Begin" );
- jstr = (*jniEnv)->CallStaticObjectMethod(jniEnv, TestCCallJava, 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(TestCCallJava == NULL || mTestCCallJava == NULL || sayHello == NULL) {
- int result = InitTestCCallJava() ;
- 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, mTestCCallJava, sayHello,jstrMSG);
- __android_log_print(ANDROID_LOG_INFO, "JNIMsg", "SayHello End" );
- (*jniEnv)->DeleteLocalRef(jniEnv, jstrMSG);
- }
关键代码说明:
C中定义映射的类、方法、对象
- jclass TestCCallJava;
- jobject mTestCCallJava;
- jmethodID getTime;
- jmethodID sayHello;
InitTestCCallJava()方法初始化类、方法、对象
- TestCCallJava = (*jniEnv)->FindClass(jniEnv,"com/trunkbow/ccalljava/CCallJava");
初始化对象:
- mTestCCallJava = (*jniEnv)->NewObject(jniEnv, obj_class,construction_id);
初始化方法:
- getTime = (*jniEnv)->GetStaticMethodID(jniEnv, TestCCallJava, "getTime","()Ljava/lang/String;");
非静态方法:
- sayHello = (*jniEnv)->GetMethodID(jniEnv, TestCCallJava, "sayHello","(Ljava/lang/String;)V");
C 中调用 Java的 方法
调用静态方法:
- jstring jstr = NULL;
- char* cstr = NULL;
- __android_log_print(ANDROID_LOG_INFO, "JNIMsg", "GetTime Begin" );
- jstr = (*jniEnv)->CallStaticObjectMethod(jniEnv, TestCCallJava, 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" );
调用非静态方法
- jstring jstrMSG = NULL;
- jstrMSG =(*jniEnv)->NewStringUTF(jniEnv, "Hi,I'm From C");
- __android_log_print(ANDROID_LOG_INFO, "JNIMsg", "SayHello Begin" );
- (*jniEnv)->CallVoidMethod(jniEnv, mTestCCallJava, sayHello,jstrMSG);
- __android_log_print(ANDROID_LOG_INFO, "JNIMsg", "SayHello End" );
注意 GetXXXMethodID 和 CallXXXMethod 。
第一个XXX 表示的是映射方法的类型,如: 静态 跟非静态
第二个 XXX 表示 调用方法的返回值 ,如:Void,Object,等等。(调用静态方法的时候Call后面要加Static)
详细 映射方法 和 调用方法 请参考 JNI 文档 ,这个很重要 !
Android JNI入门第六篇——C调用Java的更多相关文章
- Android JNI入门第五篇——基本数据类型使用
前面讲到了java和native数据类型,这里就开始做一下使用: 第一步:新建工程 第二部:书写 java方法: public class NativeMethod { static { System ...
- Android JNI入门第三篇——jni头文件分析
一. 首先写了java文件: public class HeaderFile { private native void doVoid(); native int doShort(); native ...
- Android JNI入门第四篇——Android.mk文件分析
ndroid.mk文件是在使用NDK编译C代码时必须的文件,Android.mk文件中描述了哪些C文件将被编译且指明了如何编译.掌握Android.mk文件的编写主要是掌握其里头将要使用的一些关键字, ...
- Android jni c/c++线程通过CallVoidMethod调用java函数出现奔溃问题
最近在移植网络摄像机里的p2p库到android平台,需要用到jni,最近在c线程了调用java函数的时候 出现一个问题,假如在同一个线程调用java函数是没问题的,但在一个c线程了调用java函数就 ...
- Android JNI开发之C/C++层调用JAVA
一.从C/C++层调用JAVA层代码(无参数调用) //在c代码里面调用java代码里面的方法 // java 反射 // 1 . 找到java代码的 class文件 // jclass (*Find ...
- Android入门第六篇之ListView (一)
本文来自http://blog.csdn.net/hellogv/ ListView是一个经经常使用到的控件,ListView里面的每一个子项Item能够使一个字符串,也能够是一个组合控件.先说说Li ...
- # hadoop入门第六篇:Hive实例
前言 前面已经讲了如何部署在hadoop集群上部署hive,现在我们就做一个很小的实例去熟悉HIVE QL.使用的数据是视频播放数据包括视频编码,播放设备编码,用户账号编码等,我们在这个数据基础上 ...
- Visualforce入门第六篇_2017.3.1
Visualforce实现过滤.数据列表显示.分页功能 可以参考salesforce官网开发文档:https://trailhead.salesforce.com/modules/visualforc ...
- JavaMail入门第五篇 解析邮件
上一篇JavaMail入门第四篇 接收邮件中,控制台打印出的内容,我们无法阅读,其实,让我们自己来解析一封复杂的邮件是很不容易的,邮件里面格式.规范复杂得很.不过,我们所用的浏览器内置了解析各种数据类 ...
随机推荐
- MySQL性能优化的最佳21条经验【转载】
今天,数据库的操作越来越成为整个应用的性能瓶颈了,这点对于Web应用尤其明显.关于数据库的性能,这并不只是DBA才需要担心的事,而这更是我们程序员需要去关注的事情.当我们去设计数据库表结构,对操作数据 ...
- list组件
<?xml version="1.0"?> <!-- Simple example to demonstrate the Spark List component ...
- linux centos下安装g++
1.查看是否安装 g++ -v 2.命令直接安装 yum install gcc-c++ 3.提示你找不到g++安装包,执行下面命令 yum install gcc-c++ libstdc++-dev ...
- 1.2 eclipse使用 :working set
working set可以是相当于文件夹~~有多个project时, 分别存放在不同的 workingset下,可以方便管理 新建或编辑 working set时,需要记住选择project *可参照 ...
- 用GeneratedKeyHolder获得新建数据主键值
public User createUser(final User user) { final String sql = "insert into sys_users(username, p ...
- C# 引用参数
最近经常和同事讨论引用参数的问题,为了搞清楚,查了些资料,其中CLR via C#中讲的比较清楚,整理了下 ----摘自(CLR via C#) 在默认情况下,CLR假设所有的方法参数都是按值传递的. ...
- cURL: Learning..
CURL usage.. -v, -m, -H, -I, -s, --connect-timeout, -x, -X GET|POST, -d, -T, -o. --retry, -u curl [o ...
- 门面模式(Facade)解析
门面模式使用一个门面类来包装一些复杂的类,对外提供一个简单的访问方法. 见如下代码: class CPU { public void startup() { System.out.println(&q ...
- Loadrunner性能测试分类详(二)
一.基准测试 有基础的标准,这样能通过对比发现系统的不同点与变化. 1.可以再指定的标准下通过基准测试建立一个性能基准,这样以后当系统的环境.参数发生变化后,再进行一次相同标准下的测试,即可看出变化对 ...
- [转]程序开发基础学习二(C++ Google Style 命名规则)
无规矩不成方圆,新的岗位就需要服从团队的编码规则.很开心团队用的是Google的C++编码规则,大概看了下Google 的编码规则,正如九天翔雁说的:“Google的 C++ Style Guide远 ...