java 中本地方法调用---native关键字

以java中java.lang.System类中的一个得到系统当前时间的native方法为例,对native关键字进行说明:
__________________________________________________________
public final class System {//final关键字修饰的类不能被继承
.......
/**
*返回系统当前时间(该方法依赖于底层操作系统的实现)
*/
public static native long currentTimeMillis();
.......
}
__________________________________________________________
对于上面的currentTimeMillis();声明为native 表明:该方法的具体实现是依赖于底层的操作系统平台,并非java语言实现。
所以,一个native方法就是java调用非java代码的接口,当在方法中调用一些不是由java语言写的代码或者在方法中用java语言
直接操纵计算机硬件时要声明为native方法。native 方法中除了不能和abstract关键字连用(abstract表示方法是空实现,这显然与native调用外部实现的事实不符),非native方法并无区别。
---
JAVA本地方法适用的情况
1.为了使用底层的主机平台的某个特性,而这个特性不能通过JAVA API访问
2.为了访问一个老的系统或者使用一个已有的库,而这个系统或这个库不是用JAVA编写的
3.为了加快程序的性能,而将一段时间敏感的代码作为本地方法实现。
----------------------------------------
类首次被使用到时,编译生成字节码会被加载到内存。在该字节码的入口维持着一个该类所有方法描述符的列表,这些方法描述符包含了类的详细信息。对应native方法,描述符块中会有一个指向该方法的实现的指针。这些实现在一些DLL文件(动态链接库文件dll是由java类编译后的.h文件及jni.h文件信息和具体的方法实现等编译生成的)内,当native方法被调用时,java会调用System.loadLibrary(*.dll)方法加载该dll。
其中,java通过javah(Javah:产生可以调用Java过程的C过程,或建立能被Java程序调用的C过程的头文件)生成该类的.h文件。
jni(Java Native Interface,java本地接口)
------------------------------------------下面是加载dll文件的具体实现-------
Java代码 复制代码 收藏代码
  1. /**
  2. *java.lang.System中, 加载由 libname 参数指定的系统库
  3. */
  4. public static void loadLibrary(String libname) {//调用Runtime的loadLibrary方法
  5. Runtime.getRuntime().loadLibrary0(getCallerClass(), libname);
  6. }
  7. /**
  8. *Runtime的loadLibrary方法(带有synchronized的同步方法)
  9. */
  10. synchronized void loadLibrary0(Class fromClass, String libname) {
  11. //用于判断libname所指定文件的合法性
  12. SecurityManager security = System.getSecurityManager();
  13. if (security != null) {
  14. security.checkLink(libname);
  15. }
  16. if (libname.indexOf((int)File.separatorChar) != -1) {
  17. throw new UnsatisfiedLinkError(
  18. "Directory separator should not appear in library name: " + libname);
  19. }
  20. //调用ClassLoader的loadLibrary方法
  21. ClassLoader.loadLibrary(fromClass, libname, false);
  22. }
  23. /**
  24. *ClassLoader的loadLibrary方法
  25. */
  26. // Invoked in the java.lang.Runtime class to implement load and loadLibrary.
  27. static void loadLibrary(Class fromClass, String name, boolean isAbsolute) {
  28. try {
  29. if (!DownloadManager.isJREComplete() &&
  30. !DownloadManager.isCurrentThreadDownloading()) {
  31. DownloadManager.downloadFile("bin/" + System.mapLibraryName(name));
  32. }
  33. } catch (Exception e) {
  34. throw new UnsatisfiedLinkError("Error downloading library "
  35. + name + ": " + e);
  36. }
  37. ClassLoader loader = (fromClass == null) ? null : fromClass.getClassLoader();
  38. if (sys_paths == null) {
  39. usr_paths = initializePath("java.library.path");
  40. sys_paths = initializePath("sun.boot.library.path");
  41. }
  42. }
  43. .......
  44. }
/**
    *java.lang.System中, 加载由 libname 参数指定的系统库
    */
    public static void loadLibrary(String libname) {//调用Runtime的loadLibrary方法
        Runtime.getRuntime().loadLibrary0(getCallerClass(), libname);
    }

    /**
	 *Runtime的loadLibrary方法(带有synchronized的同步方法)
	 */
    synchronized void loadLibrary0(Class fromClass, String libname) {
        //用于判断libname所指定文件的合法性
         SecurityManager security = System.getSecurityManager();
            if (security != null) {
                     security.checkLink(libname);
            }
          if (libname.indexOf((int)File.separatorChar) != -1) {
                   throw new UnsatisfiedLinkError(
                   "Directory separator should not appear in library name: " + libname);
          }
        //调用ClassLoader的loadLibrary方法
        ClassLoader.loadLibrary(fromClass, libname, false);
    }

	/**
	 *ClassLoader的loadLibrary方法
     */
     // Invoked in the java.lang.Runtime class to implement load and loadLibrary.
    static void loadLibrary(Class fromClass, String name, boolean isAbsolute) {
        try {
            if (!DownloadManager.isJREComplete() && 
             !DownloadManager.isCurrentThreadDownloading()) {
                DownloadManager.downloadFile("bin/" + System.mapLibraryName(name));
            }
        } catch (Exception e) {
            throw new UnsatisfiedLinkError("Error downloading library " 
            + name + ": " + e);
        } 

        ClassLoader loader =  (fromClass == null) ? null : fromClass.getClassLoader();
        if (sys_paths == null) {
             usr_paths = initializePath("java.library.path");
              sys_paths = initializePath("sun.boot.library.path");
        }
       }
        .......
      }
--------------------------------------------------------------------------------------------------------------------------------
另外,java.lang.System还用到了java中的另外两个比较常见的关键字:final和static。
在这里也做简单的说明:
关于final:
final修饰的类(比如System)表示该类不可被继承,所以final不能修饰interface或abstract类;
final修饰的方法不能被子类覆盖(不能被Override),不能修饰构造函数;
final修饰字段属性、参数时表示其被第一次赋值后不能被修改(但这里请注意,如果是一个对象,
不能修改的是该对象的引用,但是对象的内容还是可以修改的)
使用final的 好处:
1.防止被修改;2.final类型的方法被调用时,编译器会把它嵌入进来在一定程度上会提高了编译效率(注意是:一定程度,原因如下)。
编译器有一种被称为inline的机制,它会使你在调用final方法时,直接将方法主体插入到调用处,而不是进行例行的方法调用,例如保存断点,压栈等,这样可能会使你的程序效率有所提高,然而当你的方法主体非常庞大时,或你在多处调用此方法,那么你的调用主体代码便会迅速膨胀,可能反而会影响效率,所以你要慎用final进行方法定义。
---------
关于static:
static 可以理解为“类”级别的关键字,因此对于static不存在this的概念,当类被jvm加载时,会按照声明的先后对static成员字段进行初始化
于是,static 可用于定义全局变量,而 static final 就可以定义一个常量 
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值