Swig:如何将 "FILE" 类型从 C++ 传递到 Java

Swig: How to pass "FILE" type from C++ to Java

我有一个 C++ 函数,我需要在 Java 中为其创建一个包装器。该函数需要一个 FILE 类型的输入参数(来自 stdio)。我不相信 Swig 可以在 Java 中自动使用接口,而无需在 swig 接口文件中进行一些手动工程 - 我可能是错的。

这是我所拥有的 --

C++ (theHeader.h):

#include <cstdio>

class SampleClass 
{
 public:
  SampleClass(FILE* file) : file(file)
  {
  }

  private:
  FILE* file;
};

我试过编译,但它导致创建了 Swig 指针类型:SWIGTYPE_p_FILE,这无助于我将 FILE 类型参数从 Java 传递给 C++。

为什么要为此使用 SWIG?这对于直接使用 JNI 来说非常简单(假设指针适合系统架构的 jlong 值,这通常是一个非常安全的假设):

Java:

// get an instance of the native C++ class
static native long getClassInstance();

static native long getClassInstance( long file );

// get the native FILE * from the class instance
static native long getFileFromClass( long cls );

// close the FILE *
static native void closeFile( long file );

你会像这样使用它:

long file = 0;
long cls = getClassInstance();
if ( cls != 0 )
{
    file = getFileFromClass( cls );
}

或者,如果您已经从本机代码获得 FILE *

long instance = getClassInstance( file );

本机 C++ 代码看起来像这样(我懒得从上面的 Java 代码中完成完整的 javac 工作):

JNIEXPORT jlong JNICALL some_class_getClassInstance(
    JNIenv *env, jclass cl )
{
    myClass *cls = new MyClass();
    return( ( jlong )( intptr_t ) cls );
}

JNIEXPORT jlong JNICALL some_class_getClassInstance(
    JNIenv *env, jclass cl, jlong file )
{
    FILE *fp = ( FILE * )( intptr_t ) file;
    myClass *cls = new MyClass( fp );
    return( ( jlong )( intptr_t ) cls );
}

JNIEXPORT jlong JNICALL some_class_getFileFromClass(
    JNIenv *env, jclass cl, jlong cls )
{
    myClass *instance = ( myClass * )( intptr_t ) cls;

    FILE *fp = myClass->getFile();

    return( ( jlong )( intptr_t ) fp );
}

JNIEXPORT void JNICALL some_class_name_closeFile(
    JNIenv *env, jclass cls, jlong file )
{
    FILE *fp = ( FILE * ) ( intptr_t ) file;
    fclose( fp );
}

请注意,代码假定 NULL 的本机定义始终为零值。从技术上讲,JNI 进行 C 调用,而不是 C++ 调用,因此可能无法保证 NULL 始终实际定义为零,具体取决于您的体系结构以及您如何编写 Java 和本机代码。因此,为了完全符合标准,NULL 指针结果将导致返回到 Java 的 jlong 值被显式设置为零。反之亦然,因为 C 标准将零值定义为 a NULL 指针值,它比较等于定义的 NULL 指针常量。 (请注意,可以有多个“NULL 指针值”,即使 NULL 本身只有一个定义。)