JNI手动编译和自定义加载运行

1、使用JNI

用JNI实现

实例:

创建HelloWorld.java

class HelloWorld

{

private native void print();

public staticvoid main(String[] args)

{

new HelloWorld().print();

}

static

{

System.loadLibrary("HelloWorld");

}

}

编译HelloWorld.java

在命令行中运行如下命令:

javac HelloWorld.java

在当前文件夹编译生成HelloWorld.class。

便宜生成头文件HelloWorld.h

在命令行中运行如下命令:

javah -jni HelloWorld

在当前文件夹中会生成HelloWorld.h

实现HelloWorld.c

创建HelloWorld.c文件输入如下的代码:

#include

#include

#include "HelloWorld.h"

JNIEXPORT void JNICALL

Java_HelloWorld_print(JNIEnv *env, jobject obj)

{

printf("Hello World!\n");

}

注意必须要包含jni.h头文件,该文件中定义了JNI用到的各种类型,宏定义等。

另外需要注意Java_HelloWorld_print的两个参数,本例比较简单,不需要用到这两个参数。但是这两个参数在JNI中非常重要。

env代表java虚拟机环境,Java传过来的参数和c有很大的不同,需要调用JVM提供的接口来转换成C类型的,就是通过调用env方法来完成转换的。

obj代表调用的对象,相当于c++的this。当c函数需要改变调用对象成员变量时,可以通过操作这个对象来完成。

编译生成libHelloWorld.so

在Linux下执行如下命令来完成编译工作:

cc -I/usr/lib/jvm/java-6-sun/include/linux/

-I/usr/lib/jvm/java-6-sun/include/

-fPIC -shared -o libHelloWorld.so HelloWorld.c

或者(我是这样编译的)

gcc -I$JAVA_HOME/include -I$JAVA_HOME/include/linux -fPIC -shared -o libHelloWorld.so HelloWorld.c

在当前目录生成libHelloWorld.so。注意一定需要包含Java的include目录(请根据自己系统环境设定),因为Helloworld.c中包含了jni.h。

另外一个值得注意的是在HelloWorld.java中我们LoadLibrary方法加载的是

“HelloWorld”,可我们生成的Library却是libHelloWorld。这是Linux的链接规定的,一个库的必须要是:lib+库

名+.so。链接的时候只需要提供库名就可以了。

运行Java程序HelloWorld

大功告成最后一步,验证前面的成果的时刻到了:

java HelloWorld

如果你这步发生问题,如果这步你收到java.lang.UnsatisfiedLinkError异常,可以通过如下方式指明加载共享库的路径:

java -Djava.library.path='.'//表示在当前路径加载so库

下面是如何添加自己的so库加载路径

下面以JAVA Web项目中DLL/SO文件的动态加载为例。说是动态其实是指可以让程序在运行期间随意指定去什么路径加载so,而jave虚拟机通常已经指定了加载so的路径(%JAVAHOME%/jre/lib),如果不指定自己的路径,要想使用so库,必须需要将so文件在启动虚拟机前拷贝到虚拟机的路径下,但运行期间程序通常没办法拷贝。

在Java Web项目中,通常的做法是将这些dll文件复制到 %JAVA_HOME%\jre\bin\ 文件夹或者 应用中间件(Tomcat|Weblogic)的bin目录下之后,在程序中才能正常使用。

第一步 建立一个监听类

    建立监听类的作用是在应用中间件启动时自动执行加载程序。

    1)创建一个类实现ServletContextListener 接口

    2)实现contextInitialized方法

    3)在项目的web.xml 文件中配置此监听类

第二步 添加动态库到系统变量

    将dll/so文件所在的路径添加到系统环境java.library.path 中,但上面说到的java -D java.library.path="."只在命令行中设置有用,在程序中不能使用System.setProperty("java.library.path")进行相应设置,设置了也无效。

应该这样设置:

private void addDirToPath(String s){

    try {

        //获取系统path变量对象

        Field field=ClassLoader.class.getDeclaredField("sys_paths");

        //设置此变量对象可访问

        field.setAccessible(true);

        //获取此变量对象的值

        String[] path=(String[])field.get(null);

        //创建字符串数组,在原来的数组长度上增加一个,用于存放增加的目录

        String[] tem=new String[path.length+1];

        //将原来的path变量复制到tem中

        System.arraycopy(path,0,tem,0,path.length);

        //将增加的目录存入新的变量数组中

        tem[path.length]=s;

        //将增加目录后的数组赋给path变量对象

        field.set(null,tem);

    } catch (Exception e) {

        e.printStackTrace();

    }

}

第三步 加载动态库文件

    接下来就可以写上下文初始化的方法了:

public void contextInitialized(ServletContextEvent arg0) {

        //获取存放dll文件的绝对路径(假设将dll文件放在系统根目录下的WEB-INF文件夹中)

        String path=arg0.getServletContext().getRealPath("WEB-INF");

        //将此目录添加到系统环境变量中

        addDirToPath(path);

        //加载相应的dll/so文件,注意要将'\'替换为'/'

        System.load(path.replaceAll("\\\\","/")+"/XXXX.dll");

}

第四步 重启启动应用中间件(Tomcat|Weblogic)

    至此就可以在你的java程序中使用dll/so文件的方法了。

你可能感兴趣的:(JNI手动编译和自定义加载运行)