您的位置:首页 > 编程语言 > C语言/C++

0072 JAVA JNI 如何调用C或者C++生成的dll【基础】

2014-12-08 09:49 731 查看
JAVA如何调用C或者C++生成的dll

本文为在 32 位 Windows 平台上实现 Java 本地方法调用dll的示例。由于对C语言不是很熟悉,有部分内容摘自网络。

第一步、编写java文件

package com.ztsoft.jni;

public class MyDll {

static {

System.loadLibrary("MyDll");

}

public native static String HelloWord();

public native static String HelloWord(int num);

}

第二步、编译生成h文件

1、编译

D:\MyEclipse 6.5M1 Blue\workspace\javaWorkSpace\src>javac com/ztsoft/jni/MyDll.java

(生成MyDll.class位于jni目录下)

2、生成h文件

D:\MyEclipse 6.5M1 Blue\workspace\javaWorkSpace\src>javah com.ztsoft.jni.MyDll

(生成com_ztsoft_jni_MyDll.h这个文件位于src目录下)

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

#include <jni.h>

/* Header for class com_ztsoft_jni_MyDll */

#ifndef _Included_com_ztsoft_jni_MyDll

#define _Included_com_ztsoft_jni_MyDll

#ifdef __cplusplus

extern "C" {

#endif

/*

 * Class:     com_ztsoft_jni_MyDll

 * Method:    HelloWord

 * Signature: ()Ljava/lang/String;

 */

JNIEXPORT jstring JNICALL Java_com_ztsoft_jni_MyDll_HelloWord__

  (JNIEnv *, jclass);

/*

 * Class:     com_ztsoft_jni_MyDll

 * Method:    HelloWord

 * Signature: (I)Ljava/lang/String;

 */

JNIEXPORT jstring JNICALL Java_com_ztsoft_jni_MyDll_HelloWord__I

  (JNIEnv *, jclass, jint);

 

#ifdef __cplusplus

}

#endif

#endif

第三步、编写C++文件

在具体实现的时候,我们只关心两个函数原型。

JNIEXPORT jstring JNICALL Java_com_ztsoft_jni_MyDll_HelloWord__

  (JNIEnv *, jclass);

JNIEXPORT jstring JNICALL Java_com_ztsoft_jni_MyDll_HelloWord__I

  (JNIEnv *, jclass, jint);

以下我用的是vs2005编写c程序。

1、选择Visual C++ -> Win32 -> Win32项目(填写项目名称等)点击确定。

 

这一步选择DLL,勾选空项目,点击完成。

然后右击右侧的源文件,添加新相

 

然后再将com_ztsoft_jni_MyDll.h文件添加到C++的头文件中(先拷贝到工程下,然后在引入现有项)

 

然后拷贝原函数到MyDll.cpp文件中

#include "com_ztsoft_jni_MyDll.h"

JNIEXPORT jstring JNICALL Java_com_ztsoft_jni_MyDll_HelloWord__

(JNIEnv *, jclass){

}

JNIEXPORT jstring JNICALL Java_com_ztsoft_jni_MyDll_HelloWord__I

(JNIEnv *, jclass, jint){

}

注意:一定要把JDK中的include文件夹中(和它下面的win32)的几个头文件拷贝到C++工程文件夹中。然后引入。

 

否则编译的时候出现类似的错误

 

 

 

根据头文件的路径,修改com_ztsoft_jni_MyDll.h

 

 

修改jni_md.h文件的引入

 

根据实际情况,修改MyDll.cpp文件。

 

 

第四步、编译生成DLL

选择build->rebuild all对所写的程序进行编译。点击build->build MyDll.DLL生成DLL文件。位于debug目录下,将它放置你的jdk的bin目录,或者system32下。

然后编写main方法测试即可。

/**

 * 测试dll方法

 * @param args

 */

public static void main(String[] args){

System.out.println(MyDll.HelloWord());

}

对程序的一点解释:

1)加了static和不加只是一个参数的区别吗。就是jclass的不同,不加static这里就是jobject。也就是 JNIEXPORT void JNICALL Java_MyNative_HelloWord(JNIEnv *env,  jobject obj)。

2)这里JNIEXPORT和JNICALL都是JNI的关键字,表示此函数是要被JNI调用的。而 jstring是以JNI为中介使JAVA的String类型与本地的string沟通的一种类型,我们可以视而不见,就当做String使用(具体对应 见表一)。函数的名称是JAVA_再加上java程序的package路径再加函数名组成的(参见有包的情况)。参数中,我们也只需要关心在JAVA程序 中存在的参数,至于JNIEnv*和jclass我们一般没有必要去碰它。

3)NewStringUTF()是JNI函数,从一个包含UTF格式编码字符的char类型数组中创建一个新的jstring对象。

4) 以上程序片断jstr=env->NewStringUTF(str);是C++中的写法,不必使用env指针。因为JNIEnv函数的C ++版本包含有直接插入成员函数,他们负责查找函数指针。而对于C的写法,应改为:jstr=(*env)->NewStringUTF(env, str);因为所有JNI函数的调用都使用env指针,它是任意一个本地方法的第一个参数。env指针是指向一个函数指针表的指针。因此在每个JNI函数 访问前加前缀(*env)->,以确保间接引用函数指针。

在C和Java编程语言之间传送值时,需要理解这些值类型在这两种语言间的对应关系。这些都在头文件jni.h中,用typedef语句声明了这些类在目标平台上的代价类。头文件也定义了常量如:JNI_FALSE=0 和JNI_TRUE=1;

表一说明了Java类型和C类型之间的对应关系。

表一 Java类型和C类型

Java编程语言 C编程语言 字节 

boolean jboolean 1 

byte jbyte 1 

char jchar 2 

short jshort 2 

int jint 4 

long jlong 8 

float jfloat 4 

double jdouble 8 

   
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
相关文章推荐