关于Java native关键字
顾名思义被native修饰的方法是一个本地方法,简单的讲,一个被native修饰的方法就是告知Java调用一个非Java实现的方法,由于Java的理念是“Write Once, Run Anywhere”,因此高移植性也让Java丧失了对底层的控制。但有的时候必须访问系统底层权限或系统资源的时候,又或者丧失移植性来追求程序的效率是个不错的想法的时候,纯Java的实现可能很复杂或者根本不可能的时候,就需要去调用一个native修饰的本地方法。和abstract修饰的方法一样,被native修饰的方法也只有声明。没有实现体,两者唯一的区别在于abstract修饰的方法是一个抽象方法,需要由该类的非抽象子类或实现一个接口的类来提供实现,而native修饰的方法需要一个非Java语言提供实现,比如C,因此使用native关键字将使程序失去跨平台的能力。下面结合示例给出native的使用方法
native关键字的使用
native的使用其实很简单,主要可以分为以下4步:
- 在Java源码文件中声明一个被native修饰的方法。
public class Hello{
public native void helloC(); //声明使用native修饰的方法
public static void main(String[] args) {
new Hello().helloC();
}
}
- 在Java源码文件中使用静态初始化块导入一个外部动态链接库,格式是这样的:
public class Hello{
public native void helloC(); //声明使用native修饰的方法
//使用静态初始化块导入一个动态链接库
static {
//注意:loadLibrary的参数必须和动态链接库名一致(不包括后缀名)
System.loadLibrary("nativeLibrary");
}
public static void main(String[] args) {
new Hello().helloC();
}
}
这样完成Java的内部工作之后,直接进入第三步
- 使用javac -h . className.java 指令编译Java源文件,注意中间的点不能省略,它代表了在当前路径下生成编译产生的头文件,当然其实你可以把它换成你想要的路径,但为了方便,推荐直接使用点。如图:
这一步编译完成之后,文件结构如图:
说明:如果你使用的jdk版本过低,可能没有javac -h这个指令,但是你可以使用javah这个指令,在jdk11中javah已经被取消了。最简单的判断方法就是直接在cmd使用javah指令,如果提示找不到就使用javac -h。 - 打开刚才编译产生的头文件内容如下:
/* DO NOT EDIT THIS FILE - it is machine generated */
#include <jni.h>
/* Header for class Hello */
#ifndef _Included_Hello
#define _Included_Hello
#ifdef __cplusplus
extern "C" {
#endif
/*
* Class: Hello
* Method: helloC
* Signature: ()V
*/
JNIEXPORT void JNICALL Java_Hello_helloC
(JNIEnv *, jobject);
#ifdef __cplusplus
}
#endif
#endif
其中的JNIEXPORT void JNICALL Java_Hello_helloC (JNIEnv *, jobject);就是我们要实现的方法。
但是其中包含了一个jni.h头文件,jni.h头文件又包含了一个jni_md.h头文件,这两个头文件分别在jdk安装路径下的include文件夹里和include下的win32文件夹下,如图:
为了方便先把这两个文件复制到gcc编译器的include文件夹里(如果你不想这样做的话可以使用gcc的-I指令,关于该指令的使用请自行百度)。如图:
最后新建一个C++文件实现JNIEXPORT void JNICALL Java_Hello_helloC (JNIEnv *, jobject),其中需要包含javac -h 产生的头文件,例子中就是Hello.h头文件。比如我们直接向控制台打印一个Hello Cpp。实现如下:
#include <stdio.h>
#include "Hello.h"
JNIEXPORT void JNICALL Java_Hello_helloC
(JNIEnv *, jobject)
{
printf("Hello Cpp");
return;
}
完了之后使用g++ --share srcName.cpp -o linkLibraryname.dll生成动态链接库,
其中srcName代表c++源文件名,linkLibrary代表生成的动态链接库名,注意和Java源文件里静态初始化块导入的库名一致。如图:
文件夹内容多了一个.dll文件,如图:
到这里已经大功告成直接运行java程序即可,结果如下: