Java采用JNI调用VC++生成的dll

jopen 10年前

     应项目需求,需要android调用java,java再调用C++实现android一个图片匹配的功能,我们作为java组需要和C++和Android进行交互。下面是java和C++采用JNI的方式进行接口传参交互,我做的一个demo并运行成功。

     什么是JNI?

     JNI 全拼是Java Native Interface,中文意思为Java本地调用。JNI标准是Java平台的一部分,它允许java和其他语言的代码进行交互。JNI开始是为了本地已经编译好的语言,尤其是C和C++而设计的,我们也可以使用JNI和其他语言进行沟通,我们只要调用约定受支持了就可以了。

     什么原理?

     我们知道C++中或者C#等.net都能生成.dll类库,而C++(或.net)中生成的.dll文件,相当于我们java程序中java源代码通过编译生成的.class文件,我们在java程序中直接通过code调用C++的dll文件即可。

     在 C语言中如果生成一个动态链接库dll文件的时候,需要两部分,一部分是为.h的头文件,这个文件是负责对一些方法的声明,而.cpp文件负责对方法的实现。而在java中,在JDK安装目录的include包中有个jni.h和jni_md.h,这个可以说是java发布的一个接口,在C++中可以直接使用这两个head类型的文件对java要调用的C++方法编译,让java调用,并可以接收java传递过来的参数。直接上代码啦。


     具体的步骤如下所示(java发布一个接口,让C++实现乘法的功能)

     1、建立java源代码如下所示。

     建立java project,在src下建立com包,在com包下面建立我们的java类Reliability.java,代码如下所示。

    package com;        public class Reliability {            static {                try {                    System.loadLibrary("Reliability"); // call dll                } catch (UnsatisfiedLinkError e) {                    System.out.println("The load problem");                }                    }            public native int shanfei(int i);            public static void main(String[] args) {                Reliability reliability = new Reliability();                System.out.print(reliability.shanfei(6));            }        }  

     2、编译.java源文件。

     我们想实现传递int 整型参数来实现自己和自己相乘的结果,而实现shanfei(int p)方法的是C++。我们可以通过System.loadLibrary("Reliability")方法实现加载C++生成的dll,通过native声明方法接口以及参数。     

     编写好自己的Reliability.java代码后,可以使用MyEclipse编译为Reliability.class文件或者是直接用javac命令编译为.class文件。

     我的使用的MyEclipse进行编译的,会在/Reliability/bin/com文件夹中生成Reliability.class文件。


     3、用javah 命令让Reliability.class文件生成C++需要的.h文件,也就是交给C++的接口。

     在这里需要注意的是,一定要转到你编译成class文件的上一级目录进行javah命令,并且你的JDK已经配置好环境变量,这里的环境变量为 JAVA_HOME:安装jdk bin文件夹的上一级目录,比如我的jdk安装在C:\java\bin....,那我的JAVA_HOME为:C:\java 。path目录:引用JAVA_HOME并且加上\bin ,为 %JAVA_HOME%\bin 。classpath为: .;%JAVA_HOME%\lib\dt.jar;%JAVA_HOME%\lib\tools.jar;这是一定要配置好的。然后在dos窗口如下编写如下命令,生成给C++ .h的文件。

p1.png


     这样在你的class文件上一级目录中就会出现.h 的文件,并且引入到你的MyEclipse中,与com包同级。

p2.png

     4、这样java已经给C++发布好了com_Reliability.h的头文件,C++要去实现这个方法了。

     因为要生成dll文件,所以我们需要建立动态链接库文件,如果用的是C++ 6.0环境,就是建立projects --Dynamic-LInk Library工程。

之后添加我们的com_Reliability.h文件到我们的DLL项目中,目录结构如下所示。

p3.png

     头文件中的代码如下所示,com_Reliability.h。

    /* DO NOT EDIT THIS FILE - it is machine generated */        #include <jni.h>        /* Header for class com_Reliability */                #ifndef _Included_com_Reliability        #define _Included_com_Reliability        #ifdef __cplusplus        extern "C" {        #endif        /*        * Class:     com_Reliability        * Method:    shanfei        * Signature: (I)I        */        JNIEXPORT jint JNICALL Java_com_Reliability_shanfei          (JNIEnv *, jclass, jint);                #ifdef __cplusplus        }        #endif        #endif  

     实现的Reliability.cpp代码如下所示。

    #include<jni.h>        #include "com_Reliability.h"        #include<stdio.h>        #include<jni_md.h>                JNIEXPORT jint JNICALL Java_com_Reliability_shanfei(JNIEnv *, jclass, jint p)        {             int j = p*p;             return j;                        }  

     切忌:.cpp中和.h中的方法名一定要相同!!!并且把从java中找到的jni.h 和jni_md.h粘贴到外部依赖文件夹,或者直接放到VC98\include文件夹中。


     5、生成dll文件

     这样我们ctrl+F7编译,F7在C++项目的debug中生成Reliability.dll文件,这样我们再把这个dll文件放到java编译的.class目录,同时放到MyEclipse与src同级。这样运行java的Reliability.java文件,就看到C++给我们计算的结果了。显示如下所示。

p4.png


     总结:

     上述大致就是这个过程,在这个过程中会出现这样或者那样的问题,我也在不断的去尝试错误的所在,尝试改动,最根本的就是计算机报错了,就一定是自己哪个地方有错误,抱着这样的心态,耐心的找答案、尝试、思考,设置断点调试,看到底是哪步出现的错误,最终会解决问题,让你兴奋的看到运行结果。