我们现在来写一个程序:helloWorld.java(我的所有写的文件都放在桌面,同个文件夹即可)
public class helloworld{
static
{
System.loadLibrary("cSayHello");
}
public static native void hello();
@SuppressWarnings("static-access")
public static void main(String[] args){
new helloworld().hello();
}
}
直接在编译器运行这段代码会出现下面错误:
Exception in thread "main" java.lang.UnsatisfiedLinkError: no cSayHello in java.library.path
at java.lang.ClassLoader.loadLibrary(ClassLoader.java:1867)
at java.lang.Runtime.loadLibrary0(Runtime.java:870)
at java.lang.System.loadLibrary(System.java:1122)
at thread.helloworld.<clinit>(helloworld.java:6)
上面的错误是说找不到cHello:no cHello in java.library.path,所以啊,这个c/c++的方法我们要自己实现,毕竟我们用的不是操作系统以及定义好的方法。
所以我们先来,使用cmd**在helloworld.java所在的目录下**使用命令行:
javac helloworld
javah helloworld
然后我们可以看到在helloworld.java所在的目录下多了两个文件,一个是helloworld.class文件,一个是helloworld.h文件。
打开helloworld.h,里面引用了jni.h这个文件,这个文件在我们安装的java目录下面的include文件下:
/* 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: ()V
*/
JNIEXPORT void JNICALL Java_helloworld_hello
(JNIEnv *, jclass);
#ifdef __cplusplus
}
#endif
#endif
我的java是转在C盘下面:
我们来写需要引入的c文件cSayHello
,我也是放在桌面,反正同一个文件夹就可以。
#include "helloworld.h"
#include <stdio.h>
JNIEXPORT void JNICALL Java_helloworld_hello(JNIEnv *env, jclass jc)
{
printf("Hello,I use JNICALL");
}
在windows系统上,需要下载安装WinGW Gcc,安装教程参考https://www.jianshu.com/p/535a3131ccd8,安装成功cmd输入:
gcc -m64 -Wl,--add-stdcall-alias -I"C:\Program Files\Java\jdk1.8.0_111\include" -I"C:\Program Files\Java\jdk1.8.0_111\include\win32" -shared -o cSayHello.dll helloworld.c
然后直接运行,就可以看到输出了:
java helloworld
最后总结一下步骤:
在java中使用native的步骤:
1.在java代码中声明native方法
2.执行javah来生成一个.h文件
3.写.cpp文件来实现native导出的方法,需要包含上面第二步产生的.h文件,同时也包含了jdk自带的jni.h
4.将第三步的.cpp文件通过gcc 编译成动态链接库文件
5.在java中使用的用System.loadLibrary()方法加载第四步产生的动态链接库文件,这个native()方法就可以在Java中被访问
一般情况下,我们jdk中声明的native方法,在编译的时候都会自动去加载动态链接库文件,而不需要我们自己去操作了。