java如何去调用C++的方法?
5个回答
展开全部
Java是通过JNI调用其他语言(包括C++)编译的本地方法的,而本地方法是以库文件的形式存放的(在WINDOWS平台上是DLL文件形式,在UNIX机器上是SO文件形式)。
如下是详细讲解:
1、JAVA中所需要做的工作
在JAVA程序中,首先需要在类中声明所调用的库名称,如下:
static {
System.loadLibrary(“goodluck”);
}
在这里,库的扩展名字可以不用写出来,究竟是DLL还是SO,由系统自己判断。
还需要对将要调用的方法做本地声明,关键字为native。并且只需要声明,而不需要具体实现。如下:
public native static void set(int i);
public native static int get();
然后编译该JAVA程序文件,生成CLASS,再用JAVAH命令,JNI就会生成C/C++的头文件。
例如程序TestDll.java,内容为:
public class TestDll
{
static
{
System.loadLibrary("goodluck");
}
public native static int get();
public native static void set(int i);
public static void main(String[] args)
{
TestDll test = new TestDll();
test.set(10);
System.out.println(test.get());
}
}
用javac TestDll.java编译它,会生成TestDll.class。
再用javah TestDll,则会在当前目录下生成TestDll.h文件,这个文件需要被C/C++程序调用来生成所需的库文件。
2、C/C++中所需要做的工作
对于已生成的.h头文件,C/C++所需要做的,就是把它的各个方法具体的实现。然后编译连接成库文件即可。再把库文件拷贝到JAVA程序的路径下面,就可以用JAVA调用C/C++所实现的功能了。
接上例子。我们先看一下TestDll.h文件的内容:
/* DO NOT EDIT THIS FILE - it is machine generated */
#include <jni.h>
/* Header for class TestDll */
#ifndef _Included_TestDll
#define _Included_TestDll
#ifdef __cplusplus
extern "C" {
#endif
JNIEXPORT jint JNICALL Java_TestDll_get (JNIEnv *, jclass);
JNIEXPORT void JNICALL Java_TestDll_set (JNIEnv *, jclass, jint);
#ifdef __cplusplus
}
#endif
#endif
在具体实现的时候,我们只关心两个函数原型
JNIEXPORT jint JNICALL Java_TestDll_get (JNIEnv *, jclass);
和
JNIEXPORT void JNICALL Java_TestDll_set (JNIEnv *, jclass, jint);
这里JNIEXPORT和JNICALL都是JNI的关键字,表示此函数是要被JNI调用的。而jint是以JNI为中介使JAVA的int类型与本地的int沟通的一种类型,我们可以视而不见,就当做int使用。函数的名称是JAVA_再加上java程序的package路径再加函数名组成的。参数中,我们也只需要关心在JAVA程序中存在的参数,至于JNIEnv*和jclass我们一般没有必要去碰它。
下面我们用TestDll.cpp文件具体实现这两个函数:
#include "TestDll.h"
int i = 0;
JNIEXPORT jint JNICALL Java_TestDll_get (JNIEnv *, jclass)
{
return i;
}
JNIEXPORT void JNICALL Java_TestDll_set (JNIEnv *, jclass, jint j)
{
i = j;
}
编译连接成库文件,本例是在WINDOWS下做的,生成的是DLL文件。并且名称要与JAVA中需要调用的一致,这里就是goodluck.dll
把goodluck.dll拷贝到TestDll.class的目录下,java TestDll运行它,就可以观察到结果了。
如下是详细讲解:
1、JAVA中所需要做的工作
在JAVA程序中,首先需要在类中声明所调用的库名称,如下:
static {
System.loadLibrary(“goodluck”);
}
在这里,库的扩展名字可以不用写出来,究竟是DLL还是SO,由系统自己判断。
还需要对将要调用的方法做本地声明,关键字为native。并且只需要声明,而不需要具体实现。如下:
public native static void set(int i);
public native static int get();
然后编译该JAVA程序文件,生成CLASS,再用JAVAH命令,JNI就会生成C/C++的头文件。
例如程序TestDll.java,内容为:
public class TestDll
{
static
{
System.loadLibrary("goodluck");
}
public native static int get();
public native static void set(int i);
public static void main(String[] args)
{
TestDll test = new TestDll();
test.set(10);
System.out.println(test.get());
}
}
用javac TestDll.java编译它,会生成TestDll.class。
再用javah TestDll,则会在当前目录下生成TestDll.h文件,这个文件需要被C/C++程序调用来生成所需的库文件。
2、C/C++中所需要做的工作
对于已生成的.h头文件,C/C++所需要做的,就是把它的各个方法具体的实现。然后编译连接成库文件即可。再把库文件拷贝到JAVA程序的路径下面,就可以用JAVA调用C/C++所实现的功能了。
接上例子。我们先看一下TestDll.h文件的内容:
/* DO NOT EDIT THIS FILE - it is machine generated */
#include <jni.h>
/* Header for class TestDll */
#ifndef _Included_TestDll
#define _Included_TestDll
#ifdef __cplusplus
extern "C" {
#endif
JNIEXPORT jint JNICALL Java_TestDll_get (JNIEnv *, jclass);
JNIEXPORT void JNICALL Java_TestDll_set (JNIEnv *, jclass, jint);
#ifdef __cplusplus
}
#endif
#endif
在具体实现的时候,我们只关心两个函数原型
JNIEXPORT jint JNICALL Java_TestDll_get (JNIEnv *, jclass);
和
JNIEXPORT void JNICALL Java_TestDll_set (JNIEnv *, jclass, jint);
这里JNIEXPORT和JNICALL都是JNI的关键字,表示此函数是要被JNI调用的。而jint是以JNI为中介使JAVA的int类型与本地的int沟通的一种类型,我们可以视而不见,就当做int使用。函数的名称是JAVA_再加上java程序的package路径再加函数名组成的。参数中,我们也只需要关心在JAVA程序中存在的参数,至于JNIEnv*和jclass我们一般没有必要去碰它。
下面我们用TestDll.cpp文件具体实现这两个函数:
#include "TestDll.h"
int i = 0;
JNIEXPORT jint JNICALL Java_TestDll_get (JNIEnv *, jclass)
{
return i;
}
JNIEXPORT void JNICALL Java_TestDll_set (JNIEnv *, jclass, jint j)
{
i = j;
}
编译连接成库文件,本例是在WINDOWS下做的,生成的是DLL文件。并且名称要与JAVA中需要调用的一致,这里就是goodluck.dll
把goodluck.dll拷贝到TestDll.class的目录下,java TestDll运行它,就可以观察到结果了。
展开全部
这是一个调用c++ jni 的列子
首先写一个GoodLuck 类,里面包含native本地方法,这是用作C/C++实现的。也就是用C/c++实现java的native方法。
public class GoodLuck {
private String string="";
private int num;
public GoodLuck() {
}
static {
System.out.println(System.getProperty("java.library.path"));
System.loadLibrary("goodluck");
}
public native static String getString();
public native static String getInt();
public native static void printOutString();
public native static void printOutInt();
public static void main(String args[]){
GoodLuck gl = new GoodLuck();
gl.getString();
gl.getInt();
gl.printOutString();
gl.printOutInt();
}
}
然后用 javac命令运行GoodLuck.java文件,生成GoodLuck.class文件:javac GoodLuck.java,然后再用javah运行GoodLuck.class文件:javah GoodLuck。生成GoodLuck.h。GoodLuck.h的内容为:
/* DO NOT EDIT THIS FILE - it is machine generated */
#include "jni.h"
/* Header for class GoodLuck */
#ifndef _Included_GoodLuck
#define _Included_GoodLuck
#ifdef __cplusplus
extern "C" {
#endif
/*
* Class: GoodLuck
* Method: getString
* Signature: ()Ljava/lang/String;
*/
JNIEXPORT jstring JNICALL Java_GoodLuck_getString
(JNIEnv *, jclass);
/*
* Class: GoodLuck
* Method: getInt
* Signature: ()Ljava/lang/String;
*/
JNIEXPORT jstring JNICALL Java_GoodLuck_getInt
(JNIEnv *, jclass);
/*
* Class: GoodLuck
* Method: printOutString
* Signature: ()V
*/
JNIEXPORT void JNICALL Java_GoodLuck_printOutString
(JNIEnv *, jclass);
/*
* Class: GoodLuck
* Method: printOutInt
* Signature: ()V
*/
JNIEXPORT void JNICALL Java_GoodLuck_printOutInt
(JNIEnv *, jclass);
#ifdef __cplusplus
}
#endif
#endif
然后用VC6.0新建工程testdll(注意:工程为MFC AppWizard(dll)工程)
然后记得将GoodLuck.h引入,还有jni.h(在jdk目录下),最后用C++实现这几个方法,编译后找到生成的dll文件,重命名为goodluck.dll,复制该dll文件与GoodLuck.class同目录下,运行java GoodLuck,就可以了。
首先写一个GoodLuck 类,里面包含native本地方法,这是用作C/C++实现的。也就是用C/c++实现java的native方法。
public class GoodLuck {
private String string="";
private int num;
public GoodLuck() {
}
static {
System.out.println(System.getProperty("java.library.path"));
System.loadLibrary("goodluck");
}
public native static String getString();
public native static String getInt();
public native static void printOutString();
public native static void printOutInt();
public static void main(String args[]){
GoodLuck gl = new GoodLuck();
gl.getString();
gl.getInt();
gl.printOutString();
gl.printOutInt();
}
}
然后用 javac命令运行GoodLuck.java文件,生成GoodLuck.class文件:javac GoodLuck.java,然后再用javah运行GoodLuck.class文件:javah GoodLuck。生成GoodLuck.h。GoodLuck.h的内容为:
/* DO NOT EDIT THIS FILE - it is machine generated */
#include "jni.h"
/* Header for class GoodLuck */
#ifndef _Included_GoodLuck
#define _Included_GoodLuck
#ifdef __cplusplus
extern "C" {
#endif
/*
* Class: GoodLuck
* Method: getString
* Signature: ()Ljava/lang/String;
*/
JNIEXPORT jstring JNICALL Java_GoodLuck_getString
(JNIEnv *, jclass);
/*
* Class: GoodLuck
* Method: getInt
* Signature: ()Ljava/lang/String;
*/
JNIEXPORT jstring JNICALL Java_GoodLuck_getInt
(JNIEnv *, jclass);
/*
* Class: GoodLuck
* Method: printOutString
* Signature: ()V
*/
JNIEXPORT void JNICALL Java_GoodLuck_printOutString
(JNIEnv *, jclass);
/*
* Class: GoodLuck
* Method: printOutInt
* Signature: ()V
*/
JNIEXPORT void JNICALL Java_GoodLuck_printOutInt
(JNIEnv *, jclass);
#ifdef __cplusplus
}
#endif
#endif
然后用VC6.0新建工程testdll(注意:工程为MFC AppWizard(dll)工程)
然后记得将GoodLuck.h引入,还有jni.h(在jdk目录下),最后用C++实现这几个方法,编译后找到生成的dll文件,重命名为goodluck.dll,复制该dll文件与GoodLuck.class同目录下,运行java GoodLuck,就可以了。
本回答被提问者和网友采纳
已赞过
已踩过<
评论
收起
你对这个回答的评价是?
展开全部
如果是c应该可以使用jni 和 jna调用,不知道是否可以使用jna调用c++
已赞过
已踩过<
评论
收起
你对这个回答的评价是?
2012-09-12
展开全部
好像不行,我当时项目遇到这种情况要用java重写一次方法。
已赞过
已踩过<
评论
收起
你对这个回答的评价是?
已赞过
已踩过<
评论
收起
你对这个回答的评价是?
推荐律师服务:
若未解决您的问题,请您详细描述您的问题,通过百度律临进行免费专业咨询