美文网首页
JNI手动编译和自定义加载运行

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

作者: 王朋6 | 来源:发表于2018-03-10 00:21 被阅读0次

    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 <jni.h>

    #include <stdio.h>

    #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手动编译和自定义加载运行

          本文链接:https://www.haomeiwen.com/subject/qzqvfftx.html