`
dbp_cn
  • 浏览: 85003 次
  • 性别: Icon_minigender_1
  • 来自: 广州
社区版块
存档分类
最新评论

java Object类源代码详解 及native method.

    博客分类:
  • java
 
阅读更多
  1. package java.lang;   
  2. public class Object {   
  3.     
  4.    /* 一个本地方法,具体是用C(C++)在DLL中实现的,然后通过JNI调用。*/    
  5.     private static native void registerNatives();   
  6.   /* 对象初始化时自动调用此方法*/  
  7.     static {   
  8.         registerNatives();   
  9.     }   
  10.    /* 返回此 Object 的运行时类。*/  
  11.     public final native Class<?> getClass();   
  12.   
  13. /*  
  14. hashCode 的常规协定是:  
  15. 1.在 Java 应用程序执行期间,在对同一对象多次调用 hashCode 方法时,必须一致地返回相同的整数,前提是将对象进行 equals 比较时所用的信息没有被修改。从某一应用程序的一次执行到同一应用程序的另一次执行,该整数无需保持一致。   
  16. 2.如果根据 equals(Object) 方法,两个对象是相等的,那么对这两个对象中的每个对象调用 hashCode 方法都必须生成相同的整数结果。   
  17. 3.如果根据 equals(java.lang.Object) 方法,两个对象不相等,那么对这两个对象中的任一对象上调用 hashCode 方法不 要求一定生成不同的整数结果。但是,程序员应该意识到,为不相等的对象生成不同整数结果可以提高哈希表的性能。  
  18. */  
  19.   
  20.     public native int hashCode();   
  21.   
  22.   
  23.     public boolean equals(Object obj) {   
  24.     return (this == obj);   
  25.     }   
  26.   
  27.     /*本地CLONE方法,用于对象的复制。*/  
  28.     protected native Object clone() throws CloneNotSupportedException;   
  29.   
  30.     /*返回该对象的字符串表示。非常重要的方法*/  
  31.     public String toString() {   
  32.     return getClass().getName() + "@" + Integer.toHexString(hashCode());   
  33.     }   
  34.   
  35.    /*唤醒在此对象监视器上等待的单个线程。*/  
  36.     public final native void notify();   
  37.   
  38.    /*唤醒在此对象监视器上等待的所有线程。*/  
  39.     public final native void notifyAll();   
  40.   
  41.   
  42. /*在其他线程调用此对象的 notify() 方法或 notifyAll() 方法前,导致当前线程等待。换句话说,此方法的行为就好像它仅执行 wait(0) 调用一样。   
  43. 当前线程必须拥有此对象监视器。该线程发布对此监视器的所有权并等待,直到其他线程通过调用 notify 方法,或 notifyAll 方法通知在此对象的监视器上等待的线程醒来。然后该线程将等到重新获得对监视器的所有权后才能继续执行。*/  
  44.     public final void wait() throws InterruptedException {   
  45.     wait(0);   
  46.     }   
  47.   
  48.   
  49.   
  50.    /*在其他线程调用此对象的 notify() 方法或 notifyAll() 方法,或者超过指定的时间量前,导致当前线程等待。*/  
  51.     public final native void wait(long timeout) throws InterruptedException;   
  52.   
  53.     /* 在其他线程调用此对象的 notify() 方法或 notifyAll() 方法,或者其他某个线程中断当前线程,或者已超过某个实际时间量前,导致当前线程等待。*/  
  54.     public final void wait(long timeout, int nanos) throws InterruptedException {   
  55.         if (timeout < 0) {   
  56.             throw new IllegalArgumentException("timeout value is negative");   
  57.         }   
  58.   
  59.         if (nanos < 0 || nanos > 999999) {   
  60.             throw new IllegalArgumentException(   
  61.                 "nanosecond timeout value out of range");   
  62.         }   
  63.   
  64.     if (nanos >= 500000 || (nanos != 0 && timeout == 0)) {   
  65.         timeout++;   
  66.     }   
  67.   
  68.     wait(timeout);   
  69.     }   
  70.   
  71.     /*当垃圾回收器确定不存在对该对象的更多引用时,由对象的垃圾回收器调用此方法。*/  
  72.     protected void finalize() throws Throwable { }   
  73. }  

1.volatile:为了获得最佳速度,java允许线程保存共享成员变量的私有拷贝,而且只当线程进入或者离开同步代码块时才与共享成员变量的原始值对比,volatile修饰的成员变量在每次被线程访问时,都强迫从共享内存中重读该成员变量的值。而且,当成员变量发生变化时,强迫线程将变化值回写到共享内存。这样在任何时刻,两个不同的线程总是看到某个成员变量的同一个值,volatile关键字就是提示VM:对于这个成员变量不能保存它的私有拷贝,而应直接与共享成员变量交互。
使用建议:在两个或者更多的线程访问的成员变量上使用volatile。当要访问的变量已在synchronized代码块中,或者为常量时,不必使用。

2.transient:transient声明一个实例变量,当对象存储时,它的值不需要维持

3.native:native是方法修饰符,native是由另外一种语言实现的本地方法,

一. 什么是Native Method(转载自javaeye某位大侠的文章)
   简单地讲,一个Native Method就是一个java调用非java代码的接口。一个Native Method是这样一个java的方法:该方法的实现由非java语言实现,比如C。这个特征并非java所特有,很多其它的编程语言都有这一机制,比如在C++中,你可以用extern "C"告知C++编译器去调用一个C的函数。
   "A native method is a Java method whose implementation is provided by non-java code."
   在定义一个native method时,并不提供实现体(有些像定义一个java interface),因为其实现体是由非java语言在外面实现的。,下面给了一个示例:    
    public class IHaveNatives
    {
      native public void Native1( int x ) ;
      native static public long Native2() ;
      native synchronized private float Native3( Object o ) ;
      native void Native4( int[] ary ) throws Exception ;
    } 
    这些方法的声明描述了一些非java代码在这些java代码里看起来像什么样子(view).
    标识符native可以与所有其它的java标识符连用,但是abstract除外。这是合理的,因为native暗示这些方法是有实现体的,只不过这些实现体是非java的,但是abstract却显然的指明这些方法无实现体。native与其它java标识符连用时,其意义同非Native Method并无差别,比如native static表明这个方法可以在不产生类的实例时直接调用,这非常方便,比如当你想用一个native method去调用一个C的类库时。上面的第三个方法用到了native synchronized,JVM在进入这个方法的实现体之前会执行同步锁机制(就像java的多线程。)
    一个native method方法可以返回任何java类型,包括非基本类型,而且同样可以进行异常控制。这些方法的实现体可以制一个异常并且将其抛出,这一点与java的方法非常相似。当一个native method接收到一些非基本类型时如Object或一个整型数组时,这个方法可以访问这非些基本型的内部,但是这将使这个native方法依赖于你所访问的java类的实现。有一点要牢牢记住:我们可以在一个native method的本地实现中访问所有的java特性,但是这要依赖于你所访问的java特性的实现,而且这样做远远不如在java语言中使用那些特性方便和容易。
    native method的存在并不会对其他类调用这些本地方法产生任何影响,实际上调用这些方法的其他类甚至不知道它所调用的是一个本地方法。JVM将控制调用本地方法的所有细节。需要注意当我们将一个本地方法声明为final的情况。用java实现的方法体在被编译时可能会因为内联而产生效率上的提升。但是一个native final方法是否也能获得这样的好处却是值得怀疑的,但是这只是一个代码优化方面的问题,对功能实现没有影响。
    如果一个含有本地方法的类被继承,子类会继承这个本地方法并且可以用java语言重写这个方法(这个似乎看起来有些奇怪),同样的如果一个本地方法被fianl标识,它被继承后不能被重写。
   本地方法非常有用,因为它有效地扩充了jvm.事实上,我们所写的java代码已经用到了本地方法,在sun的java的并发(多线程)的机制实现中,许多与操作系统的接触点都用到了本地方法,这使得java程序能够超越java运行时的界限。有了本地方法,java程序可以做任何应用层次的任务。


二.为什么要使用Native Method
   java使用起来非常方便,然而有些层次的任务用java实现起来不容易,或者我们对程序的效率很在意时,问题就来了。
   与java环境外交互:
   有时java应用需要与java外面的环境交互。这是本地方法存在的主要原因,你可以想想java需要与一些底层系统如操作系统或某些硬件交换信息时的情况。本地方法正是这样一种交流机制:它为我们提供了一个非常简洁的接口,而且我们无需去了解java应用之外的繁琐的细节。
   与操作系统交互:
   JVM支持着java语言本身和运行时库,它是java程序赖以生存的平台,它由一个解释器(解释字节码)和一些连接到本地代码的库组成。然而不管怎 样,它毕竟不是一个完整的系统,它经常依赖于一些底层(underneath在下面的)系统的支持。这些底层系统常常是强大的操作系统。通过使用本地方法,我们得以用java实现了jre的与底层系统的交互,甚至JVM的一些部分就是用C写的,还有,如果我们要使用一些java语言本身没有提供封装的操作系统的特性时,我们也需要使用本地方法。
    Sun's Java
    Sun的解释器是用C实现的,这使得它能像一些普通的C一样与外部交互。jre大部分是用java实现的,它也通过一些本地方法与外界交互。例如:类java.lang.Thread 的 setPriority()方法是用java实现的,但是它实现调用的是该类里的本地方法setPriority0()。这个本地方法是用C实现的,并被植入JVM内部,在Windows 95的平台上,这个本地方法最终将调用Win32 SetPriority() API。这是一个本地方法的具体实现由JVM直接提供,更多的情况是本地方法由外部的动态链接库(external dynamic link library)提供,然后被JVM调用。


三.JVM怎样使Native Method跑起来:
    我们知道,当一个类第一次被使用到时,这个类的字节码会被加载到内存,并且只会回载一次。在这个被加载的字节码的入口维持着一个该类所有方法描述符的list,这些方法描述符包含这样一些信息:方法代码存于何处,它有哪些参数,方法的描述符(public之类)等等。
    如果一个方法描述符内有native,这个描述符块将有一个指向该方法的实现的指针。这些实现在一些DLL文件内,但是它们会被操作系统加载到java程序的地址空间。当一个带有本地方法的类被加载时,其相关的DLL并未被加载,因此指向方法实现的指针并不会被设置。当本地方法被调用之前,这些DLL才会被加载,这是通过调用java.system.loadLibrary()实现的。
   
   最后需要提示的是,使用本地方法是有开销的,它丧失了java的很多好处。如果别无选择,我们可以选择使用本地方法。

 

转 自 :http://blog.csdn.net/sjw890821sjw/article/details/8058843

分享到:
评论

相关推荐

    JNI详解技术.ppt

    - 在C/C++环境中,按照生成的头文件编写源代码,实现native方法。这通常涉及引入jni.h头文件和相应的平台定义文件,如jni_md.h。 - 编译C/C++代码为动态链接库(DLL)或共享库。 - 将编译好的库添加到系统路径,...

    深入java虚拟机

    Java虚拟机(JVM)是Java程序的核心运行环境,它将Java源代码编译成字节码,然后解释执行这些字节码,使得Java程序可以在不同的操作系统上无缝运行,实现了“一次编写,到处运行”的目标。 1. **什么是Java虚拟机**...

    详解Java线程堆栈

    下面是一个Java源代码示例,演示了如何使用线程堆栈分析来定位问题: ```java package org.ccgogoing.study.stacktrace; public class MyTest { Object obj1 = new Object(); Object obj2 = new Object(); ...

    Java虚拟机(JVM)面试宝典1.pdf

    1. **编写源代码**:使用IDE(集成开发环境)编写Java源代码,文件扩展名为`.java`。 2. **编译**:使用`javac`命令将源代码编译成字节码文件,即`.class`文件。 3. **加载与执行**:类加载器将`.class`文件加载到...

    jni官方文档

    2. **生成头文件**:使用`javah`工具根据Java源代码生成相应的C/C++头文件。 3. **实现本地方法**:在C/C++代码中实现头文件中定义的方法。 4. **编译本地代码**:使用C/C++编译器将本地代码编译成动态链接库(DLL或...

    JAVA WORD TO HTML轻松

    #### 2.2 代码详解 - **创建Word应用对象**:使用`ActiveXComponent`创建一个指向Word应用的对象。 - **设置可见性**:将Word应用的可见属性设置为`false`,这样在后台运行时不会显示任何界面。 - **打开文档**:...

    eclipse工程,从linux(ubuntu)到windows的迁移

    - 将Linux上的Eclipse工程完整备份至安全位置,包括但不限于所有源代码文件、配置文件等。 3. **依赖管理**: - 清理项目依赖,确保所有第三方库(如JAR包)都有对应的Windows版本,并记录好其路径信息。 #### ...

    软件编程词汇表(一些编程词汇的解释)

    源文件通常以`.java`为扩展名,每个源文件可以包含一个公共类(与文件同名)和其他类。Java编译器使用源文件生成字节码文件(.class文件)。 #### 42. Java 虚拟机 (Java Virtual Machine, JVM) Java虚拟机是一个...

    NET程序员面试题汇总

    - **机制好处**:通过使用属性,开发人员可以为框架添加元数据,而无需修改框架的源代码。这使得框架更加灵活且易于扩展。 #### 2. WebService概念与XML结合 - **WebService**:Web服务是一种允许不同应用程序之间...

Global site tag (gtag.js) - Google Analytics