一、Jni开发:jni的使用
一、Jni是什么?
JNI是Java Native Interface的缩写,中文为JAVA本地调用。从Java1.1开始,Java Native Interface(JNI)标准成为java平台的一部分,它允许Java代码和其他语言写的代码进行交互。JNI一开始是为了本地已编译语言,尤其是C和C++而设计的,但是它并不妨碍你使用其他语言,只要调用约定受支持就可以了。使用java与本地已编译的代码交互,通常会丧失平台可移植性。但是,有些情况下这样做是可以接受的,甚至是必须的,比如,使用一些旧的库,与硬件、操作系统进行交互,或者为了提高程序的性能。JNI标准至少保证本地代码能工作在任何Java 虚拟机实现下。 标准的java类库可能不支持你的程序所需的特性。 JNI·或许你已经有了一个用其他语言写成的库或程序,而你希望在java程序中使用它。你可能需要用底层语言实现一个小型的时间敏感代码,比如汇编,然后在你的java程序中调用这些功能。
二、使用步骤
我的开发环境是:win10企业版64位,JDK1.8,Eclipse,Visual Studio 2015。
由于此处并没有涉及到android开发,所以此处使用的是Eclipse来写JAVA程序。C语言程序是使用Visual Studio 2015来完成的。
以下开发都是在此开发环境下完成的。
1.编写native方法
public class JniTest {
//新建一个工程,声明一个方法,该方法会从C语言代码中获取一个字符串
public native static String getStringFromC();
public static void main(String[] args) {
//把从C语言中获取的字符串打印出来
System.out.println(getStringFromC());
}
}
2. javah命令,生成.h头文件
1.在windows命令行中(快捷键:WIN+X,C)
2.定位到JAVA工程文件夹下
3.输入javah 全类名
4.如果成功执行可在工程目录,src文件夹下找到生成的头文件
C的函数名称:Java_完整类名_函数名,此处不可随意修改,也没必要修改。
生成的头文件:com_relengxing_JniTest.h
3.复制.h头文件到C/Cpp工程中
我们需要根据生成的头文件编写相应C文件,还有生成动态库。这些操作我们在Visual Studio中完成。
先新建一个VS工程,把上一步生成的头文件复制到VS工程目录下。在解决方案下右键头文件把生成的头文件添加进来。
4.复制jni.h和jni_md.h文件到C/CPP工程中
寻找jni.h和jni_md.h文件,把这两个文件也放进VS的工程中,和刚刚生成的头文件放一起,添加现有项进解决方案的头文件。(我是在安装的JDK的目录下寻找到的)。
把头文件中的#include<jni.h>改成#include "jni.h"
添加后如下图:
VS解决方案目录
5.实现.h头文件中声明的函数
新建一个C的源文件,写法如下:
#include "com_relengxing_JniTest.h"
JNIEXPORT jstring JNICALL Java_com_relengxing_JniTest_getStringFromC(JNIEnv *env, jclass cla) {
return (*env)->NewStringUTF(env, "String From C");
}
简单的说就是把刚刚生成的头文件添加进来,然后把头文件里面的声明实现一下。
6.生成dll文件
生成动态链接库,windows下动态链接库为.dll结尾。linux下动态链接库为.so结尾。
生成步骤如下:
-
首先配置配置管理器。根据自己要使用的平台选择。由于本机是x64的电脑,所以此处选择x64。还可以选择x86和ARM。
配置管理器.png
2.在解决方案处右键,然后选择“属性”。把配置类型选择为动态库(.dll),确定。然后“生成”-“生成解决方案”。
属性页.png3.生成成功后,在VS工程目录下可以寻找到生成的.dll文件
VS工程文件夹7.配置dll文件所在目录到环境变量
此处我把.dll文件复制出来,放在一个专门的文件夹下,防止以后频繁的添加环境变量。
添加环境变量的方法如下:“此电脑”-“属性”-“高级系统设置”-“高级”-“环境变量”
8.重启Eclipse
由于修改了环境变量,所以需要重启Eclipse
9.加载动态链接库
public class JniTest {
//此处加载动态链接库
static{
System.loadLibrary("JniTest");
}
public native static String getStringFromC();
public static void main(String[] args) {
System.out.println(getStringFromC());
}
}
运行该工程
结果:
运行结果
三、补充
1.动态链接库和静态链接库
动态链接库(Dynamic Link Library,缩写为DLL)是一个可以被其它应用程序共享的程序模块,其中封装了一些可以被共享的例程和资源。动态链接库文件的扩展名一般是dll,也有可能是drv、sys和fon,它和可执行文件(exe)非常类似,区别在于DLL中虽然包含了可执行代码却不能单独执行,而应由Windows应用程序直接或间接调用。
动态链接是相对于静态链接而言的。所谓静态链接是指把要调用的函数或者过程链接到可执行文件中,成为可执行文件的一部分。换句话说,函数和过程的代码就在程序的exe文件中,该文件包含了运行时所需的全部代码。当多个程序都调用相同函数时,内存中就会存在这个函数的多个拷贝,这样就浪费了宝贵的内存资源。而动态链接所调用的函数代码并没有被拷贝到应用程序的可执行文件中去,而是仅仅在其中加入了所调用函数的描述信息(往往是一些重定位信息)。仅当应用程序被装入内存开始运行时,在Windows的管理下,才在应用程序与相应的DLL之间建立链接关系。当要执行所调用DLL中的函数时,根据链接产生的重定位信息,Windows才转去执行DLL中相应的函数代码。
一般情况下,如果一个应用程序使用了动态链接库,Win32系统保证内存中只有DLL的一份复制品,这是通过内存映射文件实现的。DLL首先被调入Win32系统的全局堆栈,然后映射到调用这个DLL的进程地址空间。在Win32系统中,每个进程拥有自己的32位线性地址空间,如果一个DLL被多个进程调用,每个进程都会收到该DLL的一份映像。