虽然Java虚拟机为开发人员屏蔽了底层的实现细节,使得开发人员不用考虑底层操做系统的差别性。不过在某些应用程序中,仍是免不了要直接与底层操做系统上的原生代码进行交互。今天咱们就来看一下Java对本地调用提供的支持。java
1、为何要进行本地调用windows
1.基于性能的考虑elasticsearch
Java语言从其运行速度上来讲,在大多数方面是慢于底层操做系统上原生的C和C++等语言的。这主要是因为Java虚拟机这个中间层次的存在。若是彻底用Java语言实现的性能没法达到程序的预期要求,能够选择把部分重要且耗时的代码用C或C++来实现。工具
2.基于某些特殊的需求性能
Java平台提供的标准类库的功能很强大,包括了在开发中可能遇到的大部分功能。可是仍然有一些功能没法用标准API来实现,主要是一些与底层硬件平台直接交互的功能。Java虚拟机没有把这一部分功能暴露给运行在其上的程序。若是须要这方面的功能,那么只能使用原生代码来进行开发。ui
3.与已有的使用原生代码编写的程序之间进行集成。this
若是Java程序须要与底层操做系统上由C和C++语言开发的程序进行交互,那么能够进行本地调用。操作系统
咱们平时的开发更多的状况是后边两种状况;在elasticsearch中基本上是属于第二种状况。命令行
2、使用JNI实现本地调用code
针对以上提到的各类状况,Java提供了JNI(Java Native Interface)和JNA(Java Native Access)两种方式,其中JNI的一个重要使用场景是提升程序的性能。当对程序中关键部分的性能要求比较高的时候,可使用C和C++代码来实现。
咱们先来看下怎么使用JNI来进行本地调用。
首先咱们须要有一个Java类来声明本地方法,并负责加载本地代码库。本地方法与Java接口中的方法或抽象类中的抽象方法同样,只包含方法声明,没有相关的实现。程序中的其余部分能够用正常的方法调用本地方法,好比参数传递和返回值使用等都与正常的方法相同。当虚拟机在执行本地方法时,会尝试在已经加载的本地代码库中查找本地方法的对应实现。在查找到对应的实现方法以后,虚拟机会负责进行参数传递、实际方法调用和返回值传递等工做。
public class HelloNative { static{ System.loadLibrary("greetLib"); } public static native void greeting(); }
下一步要编写实现本地方法的C/C++代码。Java提供的命令行工具根据Java源代码生成C/C++代码所需的头文件。对于本地方法,头文件中会包含相关的方法声明与其对应。
F:\source\JNI\src>javac -h .\ .\HelloNative.java
经过下边自动生成的头文件,咱们能够看到这里有不少的隐式约定,咱们只要按照这个声明进行实现便可,具体的规则不是今天的重点,不进行详述。
/* DO NOT EDIT THIS FILE - it is machine generated */ #include <jni.h> /* Header for class HelloNative */ #ifndef _Included_HelloNative #define _Included_HelloNative #ifdef __cplusplus extern "C" { #endif /* * Class: HelloNative * Method: greeting * Signature: ()V */ JNIEXPORT void JNICALL Java_HelloNative_greeting (JNIEnv *, jclass); #ifdef __cplusplus } #endif #endif
3、elasticsearch使用JNA实现本地调用
经过上边对JNI的简单了解,咱们更多的时候碰到的状况是,在编写Java程序以前,就已经有了可使用的本地代码库。这个本地代码库多是程序的一部分,也多是底层操做系统自带的。这些本地代码库的特色是在实现的时候并无考虑与Java虚拟机的集成,所以也没有使用与JNI相关的内容。在使用这样的本地代码库时,咱们就须要一个中间的本地代码库做为桥梁。这个本地代码库做为Java程序中本地方法的实现,负责实际调用时的参数类型转换和返回值传递等工做。这个过程是十分的繁琐的,Java提供了JNA来支持这种状况。
咱们知道elasticsearch启动的时候须要检测当前用户是不是root用户,这个检测是直接调用的底层操做系统的代码,咱们来看下elasticsearch是怎样使用JNA实现的。
首先elasticsearch提供了Natives类,做为调用本地方法的入口,并负责检测JNA的可用性。
static { boolean v = false; try { // load one of the main JNA classes to see if the classes are available. this does not ensure that all native // libraries are available, only the ones necessary by JNA to function Class.forName("com.sun.jna.Native"); v = true; } catch (ClassNotFoundException e) { logger.warn("JNA not found. native methods will be disabled.", e); } catch (UnsatisfiedLinkError e) { logger.warn("unable to load JNA native support library, native methods will be disabled.", e); } JNA_AVAILABLE = v; }
检测JNA是否可用,而后再调用JNANatives的对用方法
static boolean definitelyRunningAsRoot() { if (!JNA_AVAILABLE) { logger.warn("cannot check if running as root because JNA is not available"); return false; } return JNANatives.definitelyRunningAsRoot(); }
在JNANatives的definitelyRunningAsRoot中,若是是非windows系统,则调用
JNACLibrary.geteuid
/** Returns true if user is root, false if not, or if we don't know */ static boolean definitelyRunningAsRoot() { if (Constants.WINDOWS) { return false; // don't know } try { return JNACLibrary.geteuid() == 0; } catch (UnsatisfiedLinkError e) { // this will have already been logged by Kernel32Library, no need to repeat it return false; } }
elasticsearch使用JNAKernel32Library来封装对windows的Kernel32的调用,使用 JNACLibrary来封装对非windows系统的libc的调用
static { try { Native.register("c"); } catch (UnsatisfiedLinkError e) { logger.warn("unable to link C library. native methods (mlockall) will be disabled.", e); } } static native int mlockall(int flags); static native int geteuid();
4、总结