如何使错误日志更详细地说明对的未定义引用


我需要编译一些c/c++代码。使用NDK和Cygwin的工具,我不止一次出现错误"undefined-reference to"。大多数人说,这是由于调用了一个函数,而该函数的lib文件没有包含在源文件中。然而,我对c/c++知之甚少,而且错误日志提供了模糊的信息,所以我不知道是哪个文件中的哪一行导致了问题,甚至不知道我真正需要的是哪个lib文件。

我希望我能找到一些工具或方法来处理这样的问题,这样我就可以定位错误线或获得有关错误的更多细节。

非常感谢。

Android.mk

LOCAL_PATH:= $(call my-dir)
include $(CLEAR_VARS)
include ../includeOpenCV.mk
ifeq ("$(wildcard $(OPENCV_MK_PATH))","")
#try to load OpenCV.mk from default install location
include $(TOOLCHAIN_PREBUILT_ROOT)/user/share/OpenCV/OpenCV.mk
else
include $(OPENCV_MK_PATH)
endif
LOCAL_MODULE:=AvatarSpeak
LOCAL_SRC_FILES:=AvatarSpeak.cpp
LOCAL_LDLIBS    := -llog
include $(BUILD_SHARED_LIBRARY)

$NDK/NDK构建V=1

$ $NDK/ndk-build V=1
rm -f ./libs/armeabi/lib*.so ./libs/armeabi-v7a/lib*.so ./libs/mips/lib*.so        ./libs/x86/lib*.so
rm -f ./libs/armeabi/gdbserver ./libs/armeabi-v7a/gdbserver ./libs/mips/gdbserver     ./libs/x86/gdbserver
rm -f ./libs/armeabi/gdb.setup ./libs/armeabi-v7a/gdb.setup ./libs/mips/gdb.setup      ./libs/x86/gdb.setup
Gdbserver      : [arm-linux-androideabi-4.6] libs/armeabi/gdbserver
mkdir -p ./libs/armeabi
install -p /cygdrive/d/android-ndk-r8b/prebuilt/android-arm/gdbserver/gdbserver   ./libs/armeabi/gdbserver
Gdbsetup       : libs/armeabi/gdb.setup
mkdir -p ./libs/armeabi
echo "set solib-search-path ./obj/local/armeabi" > ./libs/armeabi/gdb.setup
echo "directory D:/android-ndk-r8b/platforms/android-14/arch-arm/usr/include   "../../OpenCV-2.3.1/share/OpenCV/../../include/opencv" "../../OpenCV-    2.3.1/share/OpenCV/../../include" D:/android-ndk-r8b/sources/cxx-stl/gnu-     libstdc++/4.6/include D:/android-ndk-r8b/sources/cxx-stl/gnu-    libstdc++/4.6/libs/armeabi/include jni" >> ./libs/armeabi/gdb.setup
SharedLibrary  : libAvatarSpeak.so
/cygdrive/d/android-ndk-r8b/toolchains/arm-linux-androideabi-     4.6/prebuilt/windows/bin/arm-linux-androideabi-g++  -Wl,-soname,libAvatarSpeak.so  -shared       --sysroot=D:/android-ndk-r8b/platforms/android-14/arch-arm  ./obj/local/armeabi/objs-      debug/AvatarSpeak/com_huawei_avatar_AvatarSpeak.o ./obj/local/armeabi/libopencv_contrib.a       ./obj/local/armeabi/libopencv_calib3d.a ./obj/local/armeabi/libopencv_objdetect.a       ./obj/local/armeabi/libopencv_features2d.a ./obj/local/armeabi/libopencv_video.a      ./obj/local/armeabi/libopencv_imgproc.a ./obj/local/armeabi/libopencv_highgui.a          ./obj/local/armeabi/libopencv_ml.a ./obj/local/armeabi/libopencv_legacy.a     ./obj/local/armeabi/libopencv_flann.a ./obj/local/armeabi/libopencv_core.a        ./obj/local/armeabi/libopencv_androidcamera.a ./obj/local/armeabi/liblibjpeg.a       ./obj/local/armeabi/liblibpng.a ./obj/local/armeabi/liblibtiff.a      ./obj/local/armeabi/liblibjasper.a ./obj/local/armeabi/libzlib.a       ./obj/local/armeabi/libgnustl_static.a    -Wl,--no-undefined -Wl,-z,noexecstack -Wl,-      z,relro -Wl,-z,now  -LD:/android-ndk-r8b/platforms/android-14/arch-arm/usr/lib -llog -lc -           lm  -o obj/local/armeabi/libAvatarSpeak.so
./obj/local/armeabi/objs-debug/AvatarSpeak/AvatarSpeak.o: In function      `Java_com_hachi_avatar_AvatarSpeak_speak':
D:eclipse_workspaceTestSo/jni/com_hachi_avatar_AvatarSpeak.cpp:52: undefined     reference to `Fap2Avi2D(char const*, char const*, char const*, char const*, char const*,    int)'
collect2: ld returned 1 exit status
/cygdrive/d/android-ndk-r8b/build/core/build-binary.mk:378: recipe for target     `obj/local/armeabi/libAvatarSpeak.so' failed
make: *** [obj/local/armeabi/libAvatarSpeak.so] Error 1

AvatarSpeak.cpp

if (fImage != NULL && fPot != NULL) {
bool sucavi = Fap2Avi2D(str3, str5, str4, str1, str2, -1);
LOGI("sucess and exit!");
} else {
LOGI("unsucess and exit!");
}

Animation2D_new.h(定义方法的头文件Fap2Avi2D)

#ifndef ANIMATION_2D_NEW_H_
#define ANIMATION_2D_NEW_H_
#include <vector>
#include <string>
#include "public.h"
#include "FaceExpressionEditor.h"
//#define  __declspec(dllexport)
bool Fap2Avi2D(const char *fap_file,const char *avi_file,const char *wav_file, const   char* image_file,const char *pot_file=NULL,int codec_type=-1);
...

通常情况下,"未定义引用"意味着您的构建配置错误。例如,Android.mk文件中的意外换行可能会导致make工具为您的项目选择错误的链接命令。

通常,错误消息包括缺少的函数或全局变量的名称,以及查找该引用的模块(文件)。如果名称不是不言自明的,您通常可以在源代码中找到调用的名称,然后将其追溯到您使用的某个外部库,因为它很可能是源代码中包含的某些头文件(.h.hpp)中声明的。

您不太可能无意中使用了必须下载的某些外部库,因为当编译器找不到必要的include(header)文件时,这种错误通常会更早地出现。

Android原生项目显示"未定义引用"错误的一个特殊情况是,当项目使用非官方的Android API时,例如OpenMax/stagefreight。在这种情况下,头文件可以在Android源代码树中找到,但链接阶段所需的编译库不在那里。通常的解决方法是从开发设备中提取二进制文件,例如

adb pull /system/lib/libstagefreight.so .

无论如何,运行V=1的ndk-build通常有助于查看执行了哪些构建命令。

没有通用的方法可以找到"丢失的库"。这就是你作为程序员(和谷歌用户)的技能必须被激活的地方。

第一步总是确定函数属于哪个库。使用谷歌总是一个不错的选择。

当您知道这一点时,您需要了解该库是否安装在您的系统上。如果没有,就这么做。

然后,您需要将此依赖项添加到构建系统中。如何做到这一点取决于所使用的环境和工具。谷歌再次成为你的最佳选择。

最新更新