我正在尝试将有效的C JNI绑定转换为C ++,因为我想从某些Java代码链接到C ++源代码(图像处理算法的开源实现)。我使用C语言,包括Java类,JNI绑定的C版本和构建系统,使用C ++源代码排除。
我现在正在尝试将JNI绑定转换为C ++,因此我可以将C ++源代码添加到构建系统并从绑定中调用它,但我遇到了一个未定义的符号。
我的Makefile很简单:
GCC = gcc -Wall -g -std=gnu99
CXX = g++ -Wall -g
LIBOBJS = parallelsurf_jni.o
LIB=libparallelsurf.so
DESTLIB=../../lib/$(LIB)
all: $(DESTLIB)
$(DESTLIB): $(LIB)
cp $(LIB) $(DESTLIB)
libparallelsurf.so: parallelsurf_ParallelSURF.h $(LIBOBJS)
ld --shared $(LIBOBJS) -o libparallelsurf.so
parallelsurf_ParallelSURF.h:
echo "Rebuilding JNI headers. Ensure java file has been recently built."
javah -classpath ../../parallelsurf.jar -jni parallelsurf.ParallelSURF
clean:
rm -f $(LIBOBJS) *~ parallelsurf_ParallelSURF.h $(LIB) $(DESTLIB) $(LIBOBJS)
JNI_INCLUDES = -I/usr/lib/jvm/java-6-sun/include/ -I/usr/lib/jvm/java-6-sun/include/linux -I/usr/lib/jvm/java-6-openjdk/include/
%.o: %.cpp
$(CXX) -shared -O2 -c -fPIC -fno-omit-frame-pointer -fno-stack-protector -D_REENTRANT $< $(JNI_INCLUDES)
我的头文件因此会自动使用javah
生成,从.h
复制到.cpp
时,我似乎没有对其进行修改。根据{{3}}的指示,.cpp文件中没有extern "C"
个阻止。
我还在这篇文章的指导下包含了一个pragma:this post。该编译指示为#pragma GCC java_exceptions
现在,这个未定义的符号(gxx_personality)似乎在网上被很好地引用/记录。许多帖子都指向Undefined Symbol _gxx_personality_v0 on link上的评论,建议使用该编译指示。但是,在包含该pragma之后,我得到了另一个未被很好覆盖的未定义符号(我得到了1个Google点击):_gcj_personality_v0
Exception in thread "main" java.lang.UnsatisfiedLinkError: /home/chardson/chardson/parallelsurf/lib/libparallelsurf.so: /home/chardson/chardson/parallelsurf/lib/libparallelsurf.so: undefined symbol: __gcj_personality_v0
at java.lang.ClassLoader$NativeLibrary.load(Native Method)
at java.lang.ClassLoader.loadLibrary0(ClassLoader.java:1750)
at java.lang.ClassLoader.loadLibrary(ClassLoader.java:1675)
at java.lang.Runtime.loadLibrary0(Runtime.java:840)
at java.lang.System.loadLibrary(System.java:1047)
at parallelsurf.ParallelSURF.<clinit>(ParallelSURF.java:17)
at testpsurf.<init>(testpsurf.java:11)
at testpsurf.main(testpsurf.java:23)
我不知道该怎么做这个未定义的符号,我不知道还能在哪里转。也许我的JNI代码中只有一个错误可以修复以解决这个问题?我的JNI C ++代码包含在下面,以保证完整性。
#include <jni.h>
#include <stdio.h>
#include <assert.h>
#pragma GCC java_exceptions
/*
* Class: parallelsurf_ParallelSURF
* Method: parallelsurf_get_keypoints
* Signature: (II[FZ)Ljava/util/ArrayList;
*/
JNIEXPORT jobject JNICALL Java_parallelsurf_ParallelSURF_parallelsurf_1get_1keypoints
(JNIEnv *jenv, jclass jcls, jint width, jint height, jfloatArray gray, jboolean rotInvariant)
{
// init arraylist class
jclass arrayListClass = jenv->FindClass("java/util/ArrayList");
assert(arrayListClass != NULL);
jmethodID alInitMethodId = jenv->GetMethodID(arrayListClass, "<init>", "()V");
assert(alInitMethodId != NULL);
jobject arrayList = jenv->NewObject(arrayListClass, alInitMethodId);
jmethodID alAddMethodId = jenv->GetMethodID(arrayListClass, "add", "(Ljava/lang/Object;)Z");
assert(alAddMethodId != NULL);
// init keypoint class
jclass keypointClass = jenv->FindClass("parallelsurf/Keypoint");
assert(keypointClass != NULL);
jmethodID kpInitMethodId = jenv->GetMethodID(keypointClass, "<init>", "(DDDDID[D)V");
assert(kpInitMethodId != NULL);
jdoubleArray descriptor = jenv->NewDoubleArray(10);
assert(descriptor != NULL);
jdouble vals[10];
for (int i=0; i < 10; i++) {
vals[i] = (jdouble) i;
}
jenv->SetDoubleArrayRegion(descriptor, 0, 10, vals);
// create keypoint
jobject kp = jenv->NewObject(keypointClass, kpInitMethodId,
(jdouble) 1, (jdouble) 2, (jdouble) 3,
(jdouble) 4, (jint) 5, (jdouble) 6,
descriptor);
// add to arraylist
jenv->CallBooleanMethod(arrayList, alAddMethodId, kp);
jenv->DeleteLocalRef(kp);
return arrayList;
}
谢谢你的期待。非常感谢您的帮助。
答案 0 :(得分:0)
gcc-help邮件列表上的一位优秀绅士想出了这个:
#pragma GCC java_exceptions
来处理__gxx_personality_v0,在ld
-l gcc_s
链接
ld
-l gcj
链接
ld
个链接-l stdc++
如果您使用g ++进行链接(我不是),其中一些可能会自动发生(stdc ++?)
另外,请确保您没有忘记(或丢失)JNI标头的#import
行 - 您需要extern C {}
位,否则您将获得UnsatisfiedLinkError