• 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()));
    	}
    
    }
    C:
    #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");

    中调用 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 文档 ,这个很重要 !


    工程的其他代码可在http://download.csdn.net/detail/xyz_lmn/4868265下载

    参考:



    /**
    * @author 张兴业
    *  iOS入门群:83702688
    *  android开发进阶群:241395671
    *  我的新浪微博:@张兴业TBOW
    */


  • 相关阅读:
    有个名字叫随便乱记——css3
    CSS3伸缩布局
    路政整理
    GIst
    SVN回滚版本
    你需要知道的CSS3 动画技术
    iScroll框架的使用和修改
    CSS3阴影 box-shadow的使用和技巧总结
    Javascript异步编程的4种方法
    zepto学习零碎
  • 原文地址:https://www.cnblogs.com/xyzlmn/p/3168078.html
Copyright © 2020-2023  润新知