`

Java Native Interface

阅读更多
JNI全名是Java Native Interface,通过JNI技术可以实现Java和其他编程语言的互相调用。这里我们使用的是Java和C的互相调用,Java提供本地接口,C实现该本地接口。

我使用的是RHEL 5,主要是为了测试一下在Linux平台下,了解JNI技术是如何实现的。通过一个HelloWorld实例,具体过程在下面讲解。

首先,实现的是Java本地接口Hello.java,代码如下所示:

class HelloWorld {

    
public native void sayHello();

    
static {
        System.loadLibrary(
"HelloWorld");
    }


    
public static void main(String[] args) {
        (
new HelloWorld()).sayHello();
    }

}


其中,方法声明为native,其实HelloWorld类就相当于一个接口,是为其他编程语言声明的接口。System.loadLibrary("HelloWorld");语句是一个static块,也就是在该HelloWorld类加载的时候进行执行。其中,该语句实现了加载本地的动态连接库(DLL),在Linux平台下,动态连接库文件是以.so作为扩展名的,也就是标准对象(Standard Object)。

对该本地接口类进行编译:

[root@localhost jni]# javac HelloWorld.java

接着,通过编译的HelloWorld.class文件,生成C语言的头文件,执行命令:

[root@localhost jni]# javah -jni HelloWorld

可以看到,在当前目录下生成一个HelloWorld.h文件,该文件就是C的接口文件,为使用C实现Java接口中定义的方法,可以发现在HelloWorld.h中有一个方法声明:

/**//* DO NOT EDIT THIS FILE - it is machine generated */

#ifndef __HelloWorld__
#define __HelloWorld__

#include 
<jni.h>

#ifdef __cplusplus
extern "C"
{
#endif

JNIEXPORT 
void JNICALL Java_HelloWorld_sayHello (JNIEnv *env, jobject);

#ifdef __cplusplus
}

#endif

#endif /* __HelloWorld__ */

然后,用C实现该方法,在HelloWorld.c文件中,代码如下:

#include <jni.h>
#include 
"HelloWorld.h"
#include 
<stdio.h>

JNIEXPORT 
void JNICALL Java_HelloWorld_sayHello (JNIEnv *env, jobject obj) {
    printf(
"Hello,the World!!!");
}


这里,方法签名为Java_HelloWorld_sayHello (JNIEnv *env, jobject obj),添加了形参obj,否则无法通过编译。

接下来,生成动态连接库libHelloWorld.so,执行命令:

[root@localhost jni]# gcc -fPIC -shared -o libHelloWorld.so HelloWorld.c

可以在当前目录下看到libHelloWorld.so,动态连接库文件名称以lib开头。将该文件拷贝到usr/lib目录下面,就可以测试了。

现在执行如下命令进行测试:

[root@localhost jni]# java HelloWorld

输出如下:

Hello,the World!!!

这只是一个非常简单的例子,主要是了解JNI在Linux下该如何用。在实际应用中,可能会非常复杂,并且要记住,一旦使用了JNI技术,系统的可移植性被破坏了。有些应用中,正是基于这种特性实现,比如限制软件的传播使用,保护开发商权益,等等。

=======================================================================

什么是JNI?Java Native Interface(Java本地接口)的简写。使用这个接口,可以轻松实现java对动态链接库Dynamic Link Library(dll)文件的调用,以实现一些C/C++的功能
本文章参考自网络一篇文章,做了一些小改动和补充说明。
一、建立java文件
注意点:1.要对dll里面的方法做本地声明
                2.加载dll
示例代码:
public class testdll {
 static{
  System.loadLibrary("goodluck"); //注意:不写扩展名,名字要与dll的文件名一致
 }
 public native int getValue();
 public native void setValue(int i);

 /**
  * @param args
  */
 public static void main(String[] args) {
  testdll test=new testdll();
  test.setValue(10);
  System.out.println(test.getValue());  
 }
}

二、编译
javac testdll.java                   ------如果没有错,进行下一步
javah testdll                           ------生成testdll.h这个头文件

三、实现步骤一中声明的方法
打开VC++6.0,建立一个dll空工程,把testdll.h导入工程中,新建一个test.cpp源文件
打开testdll.h,发现里面有两个方法:
/*
 * Class:     testdll
 * Method:    getValue
 * Signature: ()I
 */
JNIEXPORT jint JNICALL Java_testdll_getValue
  (JNIEnv *, jobject);

/*
 * Class:     testdll
 * Method:    setValue
 * Signature: (I)V
 */
JNIEXPORT void JNICALL Java_testdll_setValue
  (JNIEnv *, jobject, jint);

JNIEXPORT不用管,第二个单词jint/void是返回值,jint相当于int类型,JNICALL不用管,每一个方法名前都有前缀Java_testll_,前两个参数JNIEnv *和jobject不用管,第二个方法的第三个参数jint就是java文件里的int i。
上面有声明了,现在就要实现这两个方法,把代码补充进来吧:)
示例代码:
#include "testdll.h"
int i=0;
JNIEXPORT jint JNICALL Java_testdll_getValue
(JNIEnv *, jobject){
 return i;
}
JNIEXPORT void JNICALL Java_testdll_setValue
(JNIEnv *, jobject, jint j){
 i=j;
}

四、编译
此时编译会出现error,主要是提示找不到jni.h(这个包含在testdll.h)里,需要我们把jni.h、jni_md.h复制到C:\Program Files\Microsoft Visual Studio\VC98\Include这个目录里(默认安装目录,如果安装VC不是这个目录,请自行找到Include目录)。
jni.h------------------jdk安装目录\include
jni_md.h------------jdk安装目录\include\win32
然后再编译,构建,就有了test.dll这个动态链接库文件了,把它重命名成goodluck.dll,然后复制到testdll.java的目录下,运行吧:)
java testdll
结果:
10

 

 

=======================================================================

 

 

 

分享到:
评论

相关推荐

Global site tag (gtag.js) - Google Analytics