前言:
在查看 Thread.java 文件時,發現有一個 方法比較特殊
private native void start0();
概念:
native關鍵字說明其修飾的方法是一個原生態方法,方法對應的實現不是在當前文件,而是在用其他語言(如C和C++)實現的文件中。Java語言本身不能對操作系統底層進行訪問和操作,但是可以通過JNI接口調用其他語言來實現對底層的訪問。
JNI是Java本機接口(Java Native Interface),是一個本機編程接口,它是Java軟件開發工具箱(java Software Development Kit,SDK)的一部分。JNI允許Java代碼使用以其他語言編寫的代碼和代碼庫。Invocation API(JNI的一部分)可以用來將Java虛擬機(JVM)嵌入到本機應用程序中,從而允許程序員從本機代碼內部調用Java代碼。
native用法:
1.編寫帶有native聲明的方法的Java類(java文件)
2.使用javac命令編譯編寫的Java類(class文件)
3.使用javah -jni ****來生成后綴名為.h的頭文件(.h的文件)
4.使用其他語言(C、C++)實現本地方法
5.將本地方法編寫的文件生成動態鏈接庫(dll文件)
例子:
1.java文件
class HelloWorld{
public native void hello(String name);
static{
System.loadLibrary("hello");
}
public static void main(String[] args){
new HelloWorld().hello("jni");
}
}
2.javac命令編譯
javac HelloWorld.java
3.生成.h文件 (注意 javah HelloWorld 后邊不跟.class)
javah -jni HelloWorld
文件內容:
/* DO NOT EDIT THIS FILE - it is machine generated */
#include <jni.h>
/* Header for class HelloWorld */
#ifndef _Included_HelloWorld
#define _Included_HelloWorld
#ifdef __cplusplus
extern "C" {
#endif
/*
* Class: HelloWorld
* Method: hello
* Signature: (Ljava/lang/String;)V
*/
JNIEXPORT void JNICALL Java_HelloWorld_hello
(JNIEnv *, jobject, jstring);
#ifdef __cplusplus
}
#endif
#endif
4.實現本地方法
#include <jni.h>
#include "HelloWorld.h"
#include <stdio.h>
JNIEXPORT void JNICALL Java_HelloWorld_hello(JNIEnv *env,jobject obj, jstring name){
const char *str;
str = (*env)->GetStringUTFChars(env, name, NULL);
if (str == NULL) {
return;
}
printf("Hello World! %s \n", str );
return;
}
5.生成動態鏈接庫(二選一)
cl -I%java_home%\include -I%java_home%\include\win32 -LD HelloWorldImp.c -Fehello.dll
gcc -m64 -Wl,--add-stdcall-alias -I"C:\Program Files\Java\jdk1.8.0_131\include" -I"C:\Program Files\Java\jdk1.8.0_131\include\win32" -shared -o hello.dll hello.c
注意:生成的dll文件名在選項-Fe后面配置,這里是hello,因為在HelloWorld.java文件中我們loadLibary的時候使用的名字是hello。當然這里修改之后那里也需要修改。另外需要將-I%java_home%\include -I%java_home%\include\win32參數加上,因為在第四步里面編寫本地方法的時候引入了jni.h文件。