下载安卓APP箭头
箭头给我发消息

客服QQ:3315713922

浅析Java中对象的传递和使用

作者:课课家教育     来源: http://www.kokojia.com点击数:731发布时间: 2016-02-29 20:03:18

标签: java程序java程序员java语法

大神带你学编程,欢迎选课

  在前例中,我们将一个字串传递给固有方法。事实上,亦可将自己创建的java对象传递给固有方法。

  在我们的固有方法内部,可访问已收到的那些对象的字段及方法。

  为传递对象,声明固有方法时要采用原始的java语法。如下例所示,MyJavaClass有一个public(公共)字段,以及一个public方法。UseObjects类声明了一个固有方法,用于接收MyJavaClass类的一个对象。为调查固有方法是否能控制自己的自变量,我们设置了自变量的public字段,调用固有方法,然后打印出public字段的值。

浅析Java中对象的传递和使用_java程序_java程序员_课课家

  class MyJavaClass {

  public void divByTwo() { aValue /= 2; }

  public int aValue;

  }

  public class UseObjects {

  public static void main(String [] args) {

  UseObjects app = new UseObjects();

  MyJavaClass anObj = new MyJavaClass();

  anObj.aValue = 2;

  app.changeObject(anObj);

  System.out.println("Java: " + anObj.aValue);

  }

  private native void

  changeObject(MyJavaClass obj);

  static {

  System.loadLibrary("UseObjImpl");

  }

  }

  编译好代码,并将.class文件传递给javah后,就可以实现固有方法。在下面这个例子中,一旦取得字段和方法ID,就会通过JNI函数访问它们。

  JNIEXPORT void JNICALL

  Java_UseObjects_changeObject(

  JNIEnv * env, jobject jThis, jobject obj) {

  jclass cls;

  jfieldID fid;

  jmethodID mid;

  int value;

  cls = env->GetObjectClass(obj);

  fid = env->GetFieldID(cls,

  "aValue", "I");

  mid = env->GetMethodID(cls,

  "divByTwo", "()V");

  value = env->GetIntField(obj, fid);

  printf("Native: %d\\\\n", value);

  env->SetIntField(obj, fid, 6);

  env->CallVoidMethod(obj, mid);

  value = env->GetIntField(obj, fid);

  printf("Native: %d\\\\n", value);

  }

  除第一个自变量外,C++函数会接收一个jobject,它代表Java对象引用“固有”的那一面——那个引用是我们从Java代码里传递的。我们简单地读取aValue,把它打印出来,改变这个值,调用对象的divByTwo()方法,再将值重新打印一遍。

  为访问一个字段或方法,首先必须获取它的标识符。利用适当的JNI函数,可方便地取得类对象、元素名以及签名信息。这些函数会返回一个标识符,利用它可访问对应的元素。尽管这一方式显得有些曲折,但我们的固有方法确实对Java对象的内部布局一无所知。因此,它必须通过由JVM返回的索引访问字段和方法。这样一来,不同的JVM就可实现不同的内部对象布局,同时不会对固有方法造成影响。

  若运行java程序,就会发现从Java那一侧传来的对象是由我们的固有方法处理的。但传递的到底是什么呢?是指针,还是Java引用?而且垃圾收集器在固有方法调用期间又在做什么呢?

  垃圾收集器会在固有方法执行期间持续运行,但在一次固有方法调用期间,我们的对象可保证不会被当作“垃圾”收集去。为确保这一点,事先创建了“局部引用”,并在固有方法调用之后立即清除。由于它们的“生命期”与调用过程息息相关,所以能够保证对象在固有方法调用期间的有效性。

  由于这些引用会在每次函数调用的时候创建和破坏,所以不可在static变量中制作固有方法的局部副本(本地拷贝)。若希望一个引用在函数存在期间持续有效,就需要一个全局引用。全局引用不是由JVM创建的,但通过调用特定的JNI函数,程序员可将局部引用扩展为全局引用。创建一个全局引用时,需对引用对象的“生存时间”负责。全局引用(以及它引用的对象)会一直留在内存里,直到用特定的JNI函数明确释放了这个引用。它类似于C的malloc()和free()。

赞(15)
踩(3)
分享到:
华为认证网络工程师 HCIE直播课视频教程