有时候会遇到用 native 修饰的函数或者变量,如:
public native int hashCode();
public native int native_GetSection(int nEngineSailInfo);
为什么有个native呢?查阅资料作个简单的总结:
一、认识 native 即 JNI,Java Native Interface
凡是一种语言,都希望是纯的。比如解决某一个方案都喜欢就单单这个语言来写即可。Java平台有个用户和本地C代码进行互操作的API,称为Java Native Interface (Java本地接口)。
二、用 Java 调用 C 的“Hello,JNI”
我们需要按照下班方便的步骤进行:
1、创建一个Java类,里面包含着一个 native 的方法和加载库的方法 loadLibrary。HelloNative.java 代码如下:
public class HelloNative
{
static
{
System.loadLibrary("HelloNative");
}
public static native void sayHello();
@SuppressWarnings("static-access")
public static void main(String[] args)
{
new HelloNative().sayHello();
}
}
声明native方法:如果你想将一个方法做为一个本地方法的话,那么你就必须声明该方法为native的,并且不能实现。其中方法的参数和返回值在后面讲述。
Load动态库:System.loadLibrary("hello"):加载动态库(我们可以这样理解:我们的方法 displayHelloWorld()没有实现,但是我们在下面就直接使用了,所以必须在使用之前对它进行初始化)这里一般是以static块进行加载的。同时需要注意的是System.loadLibrary();的参数“hello”是动态库的名字。
native 关键字告诉编译器(其实是JVM)调用的是该方法在外部定义,这里指的是C。如果大家直接运行这个代码, JVM会告之:“A Java Exception has occurred.”控制台输出如下:
这是程序使用它的时候,虚拟机说不知道如何找到sayHello。下面当然是动手实际实现这个方法。
2、运行javah,得到包含该方法的C声明头文件.h
就得到了下面的 HelloNative.h文件 :
/* 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: sayHello
* Signature: ()V
*/
JNIEXPORT void JNICALL Java_HelloNative_sayHello
(JNIEnv *, jclass);
#ifdef __cplusplus
}
#endif
#endif
这里我们可以这样理解:这个h文件相当于我们在java里面的接口,这里声明了一个JNIEXPORT void JNICALL Java_HelloNative_sayHello(JNIEnv *, jclass); 方法,然后在我们的本地方法里面实现这个方法,也就是说我们在编写C/C++程序的时候所使用的方法名必须和这里的一致。
3、根据头文件,写C实现本地方法。
这里我们简单地实现这个sayHello方法如下:
#include "HelloNative.h"
#include <stdio.h>
JNIEXPORT void JNICALL Java_HelloNative_sayHello
{
printf("Hello,JNI");
}
4、生成dll共享库,然后Java程序load库,调用即可。
在Windows上,MinGW GCC 运行如下
-m64表示生成dll库是64位的。然后运行 HelloNative:
终于成功地可以看到控制台打印如下: