JNI包装器的多个实例

时间:2016-07-28 10:02:04

标签: java java-native-interface instance

我有一个类Modbus

public class Modbus {
  // init TCP
  public native void init(String ip, int port, int id);
  public native short[] read_registers(int slaveid, int addr, int len);

如果执行id

,init方法中的read_registers将打印到stdout

如果我这样做

Modbus mb1 = new Modbus("192.168.0.250", 502, 1);
Modbus mb2 = new Modbus("192.168.0.250", 502, 2);

然后

mb1.read_registers(....);
mb2.read_registers(....);

在两种情况下都会将2打印到标准输出。所以似乎第二个实例覆盖了第一个实例。如何使用jni-wrapper的多个实例?

编辑:

这里是我正在使用的C代码,我的目标是使用多个连接到samre或不同的modbus设备

#include "eu_company_lib_comm_modbus_Modbus.h"
#include <modbus/modbus-rtu.h>
#include <modbus/modbus.h>
#include <errno.h>
modbus_t *mb;
int readError;
int writeError;
int errnr;
const char* errStr;
int c_id;
JNIEXPORT void JNICALL 
Java_eu_company_lib_comm_modbus_Modbus_init__Ljava_lang_String_2I(
    JNIEnv * env, jobject obj, jstring ip, jint port, jint id) {

const char* c_ip = (*env)->GetStringUTFChars(env, ip, 0);
mb = modbus_new_tcp(c_ip, port);
c_id=id;
if (mb == NULL) {
    fprintf(stderr, "Unable to allocate libmodbus context\n");
    fflush(stderr);
    return;
}
modbus_set_debug(mb, 1);

if (modbus_connect(mb) == -1) {
    fprintf(stderr, "Connection failed: %s\n",     modbus_strerror(errno));
    fflush(stderr);
    modbus_free(mb);
    return;
}

(*env)->ReleaseStringUTFChars(env, ip, c_ip);
}
JNIEXPORT jshortArray JNICALL Java_eu_company_lib_comm_modbus_Modbus_read_1registers(
    JNIEnv * env, jobject obj, jint slave_id, jint address, jint len) {

jshortArray result = (*env)->NewShortArray(env, len);
uint16_t tab_reg[len];
printf("%d\n",c_id);
fflush(stdout);
readError = modbus_read_registers(mb, address, len, tab_reg);

errnr = errno;
errStr = modbus_strerror(errno);

(*env)->SetShortArrayRegion(env, result, 0, len, tab_reg);

return result;
}

1 个答案:

答案 0 :(得分:1)

您的主要问题是在JNI部分中您正在使用全局变量。全局变量。因此,这些变量与Modbus实例无关。

从我的观点来看,你应该将尽可能多的代码从JNI移动到ModBus的Java实现中,并使用本地字段来存储Java和JNI级别所需的所有持久值:

公共类Modbus {

private final String ip;
private final int port;
private final int id;

public ModBus(String ip, int port, ind id) {
     this.ip = ip;
     this.port = port;
     this.id = id;
}    

// init TCP
public native void init(String ip, int port, int id);
public native short[] read_registers(int slaveid, int addr, int len);

}

在JNI方面,您可以使用JNI方法GetObjectClass(..)GetFieldID(..)GetObjectField(..)访问Java字段以检索值。

搜索SO以获取C和C ++中的大量示例。

或者,您可以将本机方法设为私有,并添加一个包装器方法,该方法在每次调用时都会将所需的值传递给JNI部件:

public void init() {
    init(ip, port, id);
}

private native void init(String ip, int port, int id);