是否可以使用宏来处理定义内容?

时间:2011-10-13 23:15:45

标签: c++ c java-native-interface c-preprocessor

我目前正在使用JNI(Java Native Interface)在Java和C ++之间发送数据。在实现了一些代码后,我意识到每个方法的代码总是相似的。一个例子可能是:

JNIEXPORT void JNICALL Java_com_trial_jni_Receiver_setData__II(JNIEnv * env, jobject thiz, jint nativeObject, jint value)  
{  
    reinterpret_cast<Receiver *>(nativeObject)->setData(value);  
}  

JNIEXPORT void JNICALL Java_com_trial_jni_Receiver_setData__ILjava_lang_String_2(JNIEnv *env, jobject thiz, jint nativeObject, jstring value)  
{  
    reinterpret_cast<Receiver *>(nativeObject)->setData(value);  
}

由于所有代码都有类似的结构,我决定生成一组宏来自动生成所有这些代码。感谢Gregory Pakosz在此链接中的回答Is it possible to iterate over arguments in variadic macros?我现在能够使用预处理器验证我在宏中引入了多少参数并处理每个参数。

但是从前面的例子来看,有一件事我无法实现。假设我在一个名为JNI_METHOD的宏中有这个方法。我想做这样的事情:

#define JNI_METHOD(package,clazz,method,...) \  
    JNIEXPORT void JNICALL Java_ ##package## _ ##clazz## _ ##method##__II(JNIEnv * env, jobject thiz, jint nativeObject, SET_DECLARATION_PARAMS(__VA_ARGS__ )) \  
    { \  
        reinterpret_cast<clazz *>(nativeObject)->method(SET_DECLARED_PARAMS(__VA_ARGS__)); \  
    }  

JNI_METHOD(com_trial_jni,Receiver,setData,jint);  
JNI_METHOD(com_trial_jni,Receiver,setData,jstring);  

为了避免这个问题太长,我没有粘贴SET_DECLARATION_PARAMS和SET_DECLARED_PARAMS的声明,但第一个会产生类似'jint arg1'的东西,而第二个会产生'arg1'而没有类型。

问题是:有没有办法生成一个宏,为'jint'返回'I',为'jstring'返回'Ljava_lang_String_2'。请注意,无法使用字符串化,并且需要在第二个生成的方法名称中使用“ILjava_lang_String_2”而不是“II”。

谢谢!

2 个答案:

答案 0 :(得分:2)

嗯,您提供的链接几乎为您提供所需的解决方案。考虑一下:

#define CONCATENATE(arg1, arg2)   CONCATENATE1(arg1, arg2)
#define CONCATENATE1(arg1, arg2)  CONCATENATE2(arg1, arg2)
#define CONCATENATE2(arg1, arg2)  arg1##arg2

#define JNI_TRANSLATE_TYPE_jint I
#define JNI_TRANSLATE_TYPE_jstring Ljava_lang_String_2

#define JNI_TRANSLATE_TYPE(T) CONCATENATE(JNI_TRANSLATE_TYPE_, T)

在VS2010上进行测试:

#define STRINGIZE(arg)  STRINGIZE1(arg)
#define STRINGIZE1(arg) STRINGIZE2(arg)
#define STRINGIZE2(arg) #arg

#pragma message("jint: " STRINGIZE(JNI_TRANSLATE_TYPE(jint)))
#pragma message("jstring: " STRINGIZE(JNI_TRANSLATE_TYPE(jstring)))

输出:

1>  jint: I
1>  jstring: Ljava_lang_String_2

答案 1 :(得分:0)

您也可以考虑使用SWIG为您生成JNI包装器。它可以处理和隐藏使用JNI包装对象的所有讨厌代码,以及许多其他语言映射。