Java静态java main调用非静态态问题

> C透过JNI 层调用Java的静态和非静态方法
C透过JNI 层调用Java的静态和非静态方法
panweihua_710 & &
发布时间: & &
浏览:6 & &
回复:0 & &
悬赏:0.0希赛币
C通过JNI 层调用Java的静态和非静态方法
1、主要流程
  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 方法
2、设计实现
  1、界面设计如下:
  老样子,很搓,不过实用,嘿嘿
  代码不在这贴出了,有需要的兄弟直接到文章结束部分下载。
  2、   关键代码说明
  C中定义映射的类、方法、对象
  jclass TestP
  jobject mTestP
  jmethodID getT
  jmethodID sayH
  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/S");
  非静态:
  sayHello = (*jniEnv)-&GetMethodID(jniEnv, TestProvider, "sayHello","(Ljava/lang/S)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.
  * @author luxiaofeng $<$
  public class TestProvider {
  public static 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());
  public void 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);
  4、   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 
  LOCAL_SRC_FILES := \
  CToJava.c \
  Provider.c
  include $(BUILD_SHARED_LIBRARY)
  老样子,不说了,你懂的。 如果不懂,嘎嘎,那就请点击Android.mk 文件 简介
  5、   JNI文件夹下文件
  Provider.h
  #include &string.h&
  #include &jni.h&
  void GetTime() ;
  void SayHello();
  Provider.c
  #include "Provider.h"
  #include &android/log.h&
  extern JNIEnv* jniE
  jclass TestP
  jobject mTestP
  jmethodID getT
  jmethodID sayH
  int GetProviderInstance(jclass obj_class);
  * 初始化 类、对象、方法
  int InitProvider() {
  __android_log_print(ANDROID_LOG_INFO, "JNIMsg", "InitProvider Begin
  if(jniEnv == NULL) {
  return 0;
  if(TestProvider == NULL) {
  TestProvider = (*jniEnv)-&FindClass(jniEnv,"com/duicky/TestProvider");
  if(TestProvider == NULL){
  return -1;
  __android_log_print(ANDROID_LOG_INFO, "JNIMsg", "InitProvider Begin
  if (mTestProvider == NULL) {
  if (GetProviderInstance(TestProvider) != 1) {
  (*jniEnv)-&DeleteLocalRef(jniEnv, TestProvider);
  return -1;
  __android_log_print(ANDROID_LOG_INFO, "JNIMsg", "InitProvider Begin
  if (getTime == NULL) {
  getTime = (*jniEnv)-&GetStaticMethodID(jniEnv, TestProvider, "getTime","()Ljava/lang/S");
  if (getTime == NULL) {
  (*jniEnv)-&DeleteLocalRef(jniEnv, TestProvider);
  (*jniEnv)-&DeleteLocalRef(jniEnv, mTestProvider);
  return -2;
  __android_log_print(ANDROID_LOG_INFO, "JNIMsg", "InitProvider Begin
  if (sayHello == NULL) {
  sayHello = (*jniEnv)-&GetMethodID(jniEnv, TestProvider, "sayHello","(Ljava/lang/S)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
  __android_log_print(ANDROID_LOG_INFO, "JNIMsg", "InitProvider Begin
  return 1;
  int GetProviderInstance(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;
  mTestProvider = (*jniEnv)-&NewObject(jniEnv, obj_class,
  construction_id);
  if (mTestProvider == NULL) {
  return -2;
  return 1;
  * 获取时间 ---- 调用 Java 方法
  void GetTime() {
  if(TestProvider == NULL || getTime == NULL) {
  int result = InitProvider();
  if (result != 1) {
  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) {
  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* jniE
Java 中 声明的native getTime 方法的实现
  void Java_com_duicky_MainActivity_getTime(JNIEnv* env, jobject thiz)
  if(jniEnv == NULL) {
  jniEnv =
  GetTime();
Java 中 声明的native sayHello 方法的实现
  void Java_com_duicky_MainActivity_sayHello(JNIEnv* env, jobject thiz)
  if (jniEnv == NULL) {
  jniEnv =
  SayHello();
3、运行效果
  1、点击 “C调用java静态方法”按钮
  C成功调用了Java中的getTime 方法,通过C方法打印出上层调用得到的时间,并且上层成功吐司出调用信息出来。
  2、点击 “C调用java非静态方法”按钮
  C成功调用了sayHello 方法, 并成功接收到 C 传递的参数,和 吐司出相对应的信息
4、C调用Java注意点
  a) C 映射java 方法时 对应的签名
  getTime = (*jniEnv)-&GetStaticMethodID(jniEnv, TestProvider, "getTime","()Ljava/lang/S");
  故事情节还没发展这么快,下一章才会专门介绍下这个签名的使用
  b)映射方法的时候需要区别静态和非静态GetStaticMethodID,GetMethodID
  c)调用的时候也需要区分CallStaticObjectMethod,CallVoidMethod 而且还需要区分返回值类型
  有不理解的兄弟请留言,个人技术有限,有讲错的地方请大牛们指出,讲的不够全面的请多多包涵,谢谢,
  点击下载源码 C调用Java例子
本问题标题:
本问题地址:
温馨提示:本问题已经关闭,不能解答。
暂无合适的专家
&&&&&&&&&&&&&&&
希赛网 版权所有 & &&&&增值电信业务经营许可证湘B2-下次自动登录
现在的位置:
& 综合 & 正文
java 静态(static)方法与非静态方法
1、静态方法中可以引用静态方法和静态变量,不可以引用非静态方法与变量。
这很好理解:因为静态方法不需要实例化类,可以直接用(类名.方法名)的方式调用。 假设静态方法可以操作非静态方法与变量,那么对于(类名.方法名)的这种方法调用方式, 那么其中的非静态变量就没有被实例化。(我们知道静态变量是不需要实例化的)。
老师上课的时候讲因为静态方法中没有this指针。
2、非静态方法中可以引用静态方法和静态变量,也可以引用非静态方法与变量。
3、子类不能重写父类的静态方法,但可以声明与该静态方法相同的方法,从而将父类的静态方法隐藏。
这让我联想起了记下的一个关于private 方法的经验:
public class ClassA {
private void method1(){
System.out.println("called ClassA's method1!!!");
static void method2(){
System.out.println("called ClassA's method2!!!");
class SubClassA extends ClassA{
void method1(){
System.out.println("called Sub's method1!!!");
static void method2(){
System.out.println("called Sub's method1!!!");
public static void main(String args[]){
SubClassA a=new SubClassA();
a.method1();
//SubClassA类中无法调用父类的method1,因为该方法为私有(private)
a.method2();
//该方法也是对父类中的静态方法的覆盖
&&&&推荐文章:
【上篇】【下篇】

我要回帖

更多关于 java 静态 非静态 的文章

 

随机推荐