jdk中规定了native method,每一个native method背后对应有一个动态链接库来支持它,在windows系统上,就是dll后缀的文件。
native method是需要通过c/c++来实现的。
本示例所需的运行环境:
java中调用动态链接库可以通过System.loadLibrary("Hello")来加载指定动态链接库。假定,我们可以编写这样一个java文件,定义一个hello的本地方法,然后在main函数中调用。
public class Hello{
public native void hello(String name);
static {
System.loadLibrary("Hello");
}
public static void main(String[] args) {
new Hello().hello("jni.");
}
}
可以通过javac Hello.java编译这个文件,这时候会生成Hello.class文件。接着,利用javah通过javah -jni Hello文件反向生成Hello.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: hello
* Signature: (Ljava/lang/String;)V
*/
JNIEXPORT void JNICALL Java_Hello_hello
(JNIEnv *, jobject, jstring);
#ifdef __cplusplus
}
#endif
#endif
接下来编写Hello.c实现Hello.h中定义的方法。
#include <jni.h>
#include "Hello.h"
#include <stdio.h>
JNIEXPORT void JNICALL Java_Hello_hello
(JNIEnv *env, jobject obj, jstring name)
{
printf("hello,jni.\n");
}
可以通过命令cl编译动态链接库,生成Hello.dll,接着可以运行java Hello,打印hello,jni。如下所示:
运行java程序。
以上过程均在命令行下就可以运行,不需要ide编辑器,运行cl指令,需要在开始->Visual Studio 2017/适用于VS 2017的 x64本机工具命令,然后切换到java文件目录即可,这就是前面提到的运行环境需要vs2017的原因。
通过cl命令的时候,需要指定包含jni.h所在的目录,以及包含jni_md.h所在的目录,所以使用-I(i大写)参数,表示include,因为使用了vs2017自带命令行工具,他会自动设置LIB目录环境变量,所以不用担心libcmt.lib找不到的问题。
这个示例,其实走了一些弯路,就是使用了vc6.0++自带的cl命令编译,结果是个32位的dll文件,生成动态库没有问题,运行java程序报错:
Can't load IA 32-bit .dll on a AMD 64-bit platform
如果你的系统是32位的,那么可以使用vc6.0++自带的cl来生成动态链接库,否则,千万不要尝试,最好直接使用vs2017自带的cl编译生成,记得切换到x64的环境上。
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)