一、JNI简介
JNI是Java Native Interface的简称,也就是java本地接口。它提供了java和其他语言通信(主要是C&C++)的若干API。由于项目中出现了以下两种情况:
a.需要将一个java实现的系统与一个非java实现的系统整合起来。
b. Java代码中需要用到某种算法,不过该算法是封装在动态链接库文件(DLL)当中的。
对于上述的两种情况,如果不使用JNI的话,那就会变得非常棘手了。而且就算找到其他的解决方案,也是费时费力,这也是我们考虑使用JNI的原因。
1.1 java端
我们的开发环境是 jdk + eclipse+tomcat6.0+VC++ 6.0,先在eclipse里建立一个web工程,并让其调用该工程下的java对象。其中native方法就是我们说的 java native interface。它提供了一个调用的接口,然后用C或者C++去实现。
Java代码
package com.helloworld;
public classMyJavaBean
{
static{
System.loadLibrary("HelloEnd");
}//生成动态文件HelloEnd.dll再添加的,生成com_helloworld_MyJavaBean.h时并不存在
public nativeintplss(intd1,intd2);
}
这里的native本地方法有一个两个参数,会将其到后台去。这样java本地方法已经完成,现在来介绍下javah这个命令,javah是一个专门为JNI生成头文件的一个命令。在windows控制台输入:
javah -jni com.helloworld.MyJavaBean
命令执行完之后在工程的根目录下就会发现com_helloworld_MyJavaBean.h 这个头文件。到这里java部分算是基本完成了,接下来我们来编写后端的C++代码。
1.2 C++端
打开VC++首先新建一个Win32 Dynamic-Link library工程,之后选择An empty DLL project空工程。在这里C++的工程名是HelloEnd,把刚刚生成的那个头文件拷贝到这个工程的根目录里。
然后我们新建一个C++ source file来实现上面那个native方法。
#include"com_helloworld_MyJavaBean.h"
#include"jni.h"
JNIEXPORT jint JNICALLJava_com_helloworld_MyJavaBean_plss
(JNIEnv * env, jobject obj, jint d1, jint d2)
{
return d1+d2;
}
接着将jdk中的jni_md.h文件,放入到工程根目录,F7编译成功。在Debug目录里会发现生成了HelloEnd.dll这个文件。
这个时候后端的C++代码也已经完成,接下来的任务就是怎么把他们连接在一起了,要让前端的java程序“认识并找到”这个动态链接库,就必须把这个DLL放在windows path环境变量下面。复制我们工程的Debug目录,我这里是C:/Program Files/MicrosoftVisual Studio/MyProjects/HelloEnd/Debug这个目录,把这个目录配置到Uservariable的Path下面。重启eclipse,让eclipse在启动的时候重新读取这个path变量。
其实我们上面做的那些不是让JVM能找到那些DLL文件,接下来我们让java代码“认识”这个动态链接库。System.loadLibrary("HelloEnd");加入这句到静态初始化块里。这样我们的代码就能认识并加载这个动态链接库文件了。接下来编写测试代码。
1.3 web端
我们在页面NewFile.jsp上输入需要求和的两个整数,然后界面跳转到submit.jsp页面,在这个页面中我们将这两个整数作为参数传递给java对象MyJavaBean中的plss()函数,并在页面给出结果。
图1.3 NewFile页面
以下是submit.jsp页面的body部分代码:
<body>
两数之和是:
<%
String d1=request.getParameter("d1");
String d2=request.getParameter("d2");
int d3=Integer.valueOf(d1);
int d4=Integer.valueOf(d2);
MyJavaBean bean=new MyJavaBean();
int d5=bean.plss(d3,d4);
out.println(d5);
%>
</body>