首页 > 编程知识 正文

关于java如何用jna将字节的信息

时间:2023-12-29 20:32:03 阅读:331144 作者:YBFK

本文目录一览:

想知道java怎么取com.sun.jna.ptr.ByteByReference里面的数据。

我感觉你这样设计有问题,JNI中java和C/C++的方法应该是保留java的方法名,当然这个不是重点,重点是你里面的参数显然不一致。ByteByReference是个对象类型,我们无法把它弄成一个指针类型,毕竟java没有指针。我的建议是,pBuffer不要作为参数,用具体的类型。pBuffer作为全局变量在别的地方初始化,这样不影响你在invoke方法里面用pBuffer来获取值。我这只是从你的描述简单的看,具体的需求我也不太清楚你要做什么,个人建议。

java中jna的问题

1、在VC下创建一个动态链接库项目testJNA

2、在头文件里声明函数

C代码 收藏代码

extern "C" _declspec(dllexport) int add(int first, int second);

红色字体部分是必须的,包括定义结构体时也需要。应该是说此函数是发布的。

3、在源码里实现函数

C代码 收藏代码

int add(int first, int second) {

printf("(c) test jna : %d + %d = %d", first, second, first + second);

return first + second;

}

4、生成dll文件

5、定义一个表示链接库的接口

接口TestJnaLib继承自com.sun.jna.Library,此接口有一个实例

Java代码 收藏代码

TestJnaLib INSTANCE = (TestJnaLib)Native.loadLibrary("testJNA.dll", TestJnaLib.class);

此实例由jna通过反射自动生成。

6、定义对应dll里的方法

Java代码 收藏代码

int add(int first, int second);

7、调用本地方法

Java代码 收藏代码

TestJnaLib.INSTANCE.add(3, 5);

Jna回调Java方法:

1、在C语言部分定义带回调函数的函数

C代码 收藏代码

extern "C" _declspec(dllexport) void methodWithCallback(int (*fp)(int left, int right), int left, int right);

红色加粗部分是函数指针。

2、Java部分定义一个回调接口

必须继承自com.sun.jna.Callback接口

Java代码 收藏代码

public interface FunCallBack extends Callback {

int invoke(int left, int right);

}

Invoke方法里的参数顺序与C函数的对应

3、定义回调接口的实现

Java代码 收藏代码

public class CallbackFunImpl implements FunCallBack {

@Override

public int invoke(int left, int right) {

System.out.printf("in java :%d + %d = %dn", left, right, left + right);

return left + right;

}

}

4、在表示链接库实现的接口里定义要回调的本地函数

Java代码 收藏代码

void methodWithCallback(Callback callback, int left, int right);

本地函数的函数指针用Callback 接口替代。

5、调用带函数指针的本地函数

Java代码 收藏代码

TestJnaLib.INSTANCE.methodWithCallback(new CallbackFunImpl(), 4, 6)

怎样在linux环境下使用java调用C++动态链接库时将图片字节流传到C++方法的void*参数中?(使用的JNA)

的(在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

/* 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运行它,就可以观察到结果了。

3、你想保存一个"samplemyfile.txt"到变量str中,原本就要写成

版权声明:该文观点仅代表作者本人。处理文章:请发送邮件至 三1五14八八95#扣扣.com 举报,一经查实,本站将立刻删除。