在OSx:ld上使用CMake构建C++项目:找不到体系结构x86_64的符号

Building a C++ project with CMake on OSx: ld: symbol(s) not found for architecture x86_64

本文关键字:x86 体系结构 找不到 符号 项目 构建 ld OSx CMake C++      更新时间:2023-10-16

我使用Xerces-C++获得了一个旧C++项目的源代码,我正试图用CMake在CLion上构建该项目。OSx版本:Catalina。

我开始准备CMakeLists.txt,这在旧项目中是不可用的。

我得到了这个构建异常:

/Applications/CLion.app/Contents/bin/cmake/mac/bin/cmake --build /Users/miloscuculovic/CLionProjects/Test2/cmake-build-debug --target all -- -j 4 VERBOSE=1
/Applications/CLion.app/Contents/bin/cmake/mac/bin/cmake -S/Users/miloscuculovic/CLionProjects/Test2 -B/Users/miloscuculovic/CLionProjects/Test2/cmake-build-debug --check-build-system CMakeFiles/Makefile.cmake 0
/Applications/CLion.app/Contents/bin/cmake/mac/bin/cmake -E cmake_progress_start /Users/miloscuculovic/CLionProjects/Test2/cmake-build-debug/CMakeFiles /Users/miloscuculovic/CLionProjects/Test2/cmake-build-debug/CMakeFiles/progress.marks
/Library/Developer/CommandLineTools/usr/bin/make -f CMakeFiles/Makefile2 all
/Library/Developer/CommandLineTools/usr/bin/make -f src/CMakeFiles/Test2.dir/build.make src/CMakeFiles/Test2.dir/depend
cd /Users/miloscuculovic/CLionProjects/Test2/cmake-build-debug && /Applications/CLion.app/Contents/bin/cmake/mac/bin/cmake -E cmake_depends "Unix Makefiles" /Users/miloscuculovic/CLionProjects/Test2 /Users/miloscuculovic/CLionProjects/Test2/src /Users/miloscuculovic/CLionProjects/Test2/cmake-build-debug /Users/miloscuculovic/CLionProjects/Test2/cmake-build-debug/src /Users/miloscuculovic/CLionProjects/Test2/cmake-build-debug/src/CMakeFiles/Test2.dir/DependInfo.cmake --color=
/Library/Developer/CommandLineTools/usr/bin/make -f src/CMakeFiles/Test2.dir/build.make src/CMakeFiles/Test2.dir/build
[ 50%] Linking CXX executable Test2
cd /Users/miloscuculovic/CLionProjects/Test2/cmake-build-debug/src && /Applications/CLion.app/Contents/bin/cmake/mac/bin/cmake -E cmake_link_script CMakeFiles/Test2.dir/link.txt --verbose=1
/Library/Developer/CommandLineTools/usr/bin/c++  -g -isysroot /Library/Developer/CommandLineTools/SDKs/MacOSX10.15.sdk -Wl,-search_paths_first -Wl,-headerpad_max_install_names  CMakeFiles/Test2.dir/ComputeDelta.cpp.o  -o Test2 -framework CoreFoundation -framework CoreFoundation -framework IOKit /usr/lib/libobjc.dylib /usr/lib/libcurl.dylib 
Undefined symbols for architecture x86_64:
"xercesc_3_2::XMLAttDefList::serialize(xercesc_3_2::XSerializeEngine&)", referenced from:
vtable for xercesc_3_2::XMLAttDefList in ComputeDelta.cpp.o
"NodesManager::PrintStats()", referenced from:
XidXyDiff(XID_DOMDocument*, char const*, XID_DOMDocument*, char const*, bool, bool) in ComputeDelta.cpp.o
"NodesManager::FullBottomUp(int)", referenced from:
XidXyDiff(XID_DOMDocument*, char const*, XID_DOMDocument*, char const*, bool, bool) in ComputeDelta.cpp.o
"NodesManager::topdownMatch(int, int)", referenced from:
XidXyDiff(XID_DOMDocument*, char const*, XID_DOMDocument*, char const*, bool, bool) in ComputeDelta.cpp.o
"NodesManager::setUniqueIdHandler(UniqueIdHandler*)", referenced from:
XidXyDiff(XID_DOMDocument*, char const*, XID_DOMDocument*, char const*, bool, bool) in ComputeDelta.cpp.o
"NodesManager::registerResultDocument(XID_DOMDocument*)", referenced from:
XidXyDiff(XID_DOMDocument*, char const*, XID_DOMDocument*, char const*, bool, bool) in ComputeDelta.cpp.o
"NodesManager::registerSourceDocument(XID_DOMDocument*)", referenced from:
XidXyDiff(XID_DOMDocument*, char const*, XID_DOMDocument*, char const*, bool, bool) in ComputeDelta.cpp.o
"NodesManager::Optimize(int)", referenced from:
XidXyDiff(XID_DOMDocument*, char const*, XID_DOMDocument*, char const*, bool, bool) in ComputeDelta.cpp.o
"NodesManager::MatchById(int)", referenced from:
XidXyDiff(XID_DOMDocument*, char const*, XID_DOMDocument*, char const*, bool, bool) in ComputeDelta.cpp.o
"NodesManager::NodesManager()", referenced from:
XidXyDiff(XID_DOMDocument*, char const*, XID_DOMDocument*, char const*, bool, bool) in ComputeDelta.cpp.o
"NodesManager::~NodesManager()", referenced from:
XidXyDiff(XID_DOMDocument*, char const*, XID_DOMDocument*, char const*, bool, bool) in ComputeDelta.cpp.o
"DeltaConstructor::getDeltaDocument()", referenced from:
XidXyDiff(XID_DOMDocument*, char const*, XID_DOMDocument*, char const*, bool, bool) in ComputeDelta.cpp.o
"DeltaConstructor::constructDeltaDocument()", referenced from:
XidXyDiff(XID_DOMDocument*, char const*, XID_DOMDocument*, char const*, bool, bool) in ComputeDelta.cpp.o
"DeltaConstructor::DeltaConstructor(NodesManager*, char const*, XID_DOMDocument*, char const*, XID_DOMDocument*, bool)", referenced from:
XidXyDiff(XID_DOMDocument*, char const*, XID_DOMDocument*, char const*, bool, bool) in ComputeDelta.cpp.o
"xercesc_3_2::XMLAttDefList::getProtoType() const", referenced from:
vtable for xercesc_3_2::XMLAttDefList in ComputeDelta.cpp.o
"xercesc_3_2::XMLAttDefList::isSerializable() const", referenced from:
vtable for xercesc_3_2::XMLAttDefList in ComputeDelta.cpp.o
"_main", referenced from:
implicit entry/start for main executable
ld: symbol(s) not found for architecture x86_64
clang: error: linker command failed with exit code 1 (use -v to see invocation)
make[2]: *** [src/Test2] Error 1
make[1]: *** [src/CMakeFiles/Test2.dir/all] Error 2
make: *** [all] Error 2

有两个CMakeLists.txt文件,一个在项目的根目录中,第二个在src:中

根目录下的CMakeLists.txt:

cmake_minimum_required(VERSION 3.15)
project(Test2)
set(CMAKE_CXX_STANDARD 14)
add_subdirectory(src)

位于src:的CMakeLists.txt

include_directories(${CMAKE_CURRENT_SOURCE_DIR}/include)
include_directories(${CMAKE_CURRENT_SOURCE_DIR})
include_directories(/usr/local/Cellar/xerces-c/3.2.2/include)
include_directories(/usr/local/Cellar/opencascade/7.3.0p3/include)
include_directories(/usr/local/Cellar/libuv/1.31.0/include)
add_executable(Test2 ComputeDelta.cpp)
set(STLINK_LIB_SHARED ${PROJECT_NAME})
find_library(ObjC objc)
find_library(Curl curl)
find_library(CoreServices CoreServices)
find_library(CoreFoundation CoreFoundation)
find_library(IOKit IOKit)
target_link_libraries(${STLINK_LIB_SHARED} ${CoreServices} ${CoreFoundation} ${IOKit} ${ObjC} ${Curl})

知道怎么解决这个问题吗?我已经安装了opencascade、libuv、libev。

libuv和xerces-c都有一个pkg配置(.pc(文件,因此您可以将CMakeLists.txt简化为以下文件。这利用了FindPkgConfig模块,该模块将pkg配置文件转换为IMPORTED目标(请参阅"是时候做正确的CMake了"。此IMPORT目标将自动知道要包括哪些标头和要链接哪些库,您只需要使用target_link_libraries即可与之链接。

include(FindPkgConfig)
pkg_check_modules(Xerces REQUIRED IMPORTED_TARGET xerces-c)
pkg_check_modules(LibUv REQUIRED IMPORTED_TARGET libuv)
include_directories(${CMAKE_CURRENT_SOURCE_DIR}/include)
include_directories(${CMAKE_CURRENT_SOURCE_DIR})
include_directories(/usr/local/Cellar/opencascade/7.3.0p3/include)
add_library(libxyDelta STATIC
convertUTF.cpp
StringPusher.cpp 
ComputeDelta.cpp
Diff_DeltaConstructor.pp 
Diff_NodesManager.cpp 
Diff_UniqueIdHandler.cpp 
DeltaApply.cpp 
DeltaException.cpp 
DeltaManager.cpp 
DeltaReverse.cpp 
DeltaSortOperations.cpp 
easy_css.cpp 
lcss.cpp 
lookup2.cpp 
Tools.cpp 
XID_map.cpp 
XID_DOMDocument.cpp 
XyDeltaFileImpl.cpp 
XyDeltaDomImpl.cpp 
XyInt.cpp 
XyLatinStr.cpp 
XyStr.cpp
XyStrDiff.cpp
XyStrDelta.cpp
XyUTF8Str.cpp)
set(STLINK_LIB_SHARED ${PROJECT_NAME})
find_library(ObjC objc)
find_library(Curl curl)
find_library(CoreServices CoreServices)
find_library(CoreFoundation CoreFoundation)
find_library(IOKit IOKit)
target_link_libraries(libxyDelta ${CoreServices} ${CoreFoundation} ${IOKit} ${ObjC} ${Curl})
target_link_libraries(libxyDelta PkgConfig::LibUv PkgConfig::Xerces)
add_executable(xydiff execComputeDelta.cpp)
target_link_libraries(xydiff libxyDelta)
add_executable(xydelta execDeltaApply.cpp)
target_link_libraries(xydelta libxyDelta)

我还将您的add_executable更改为add_library。这将一起消除关于xerces和main的任何错误。这只会给您留下关于NodesManagerDeltaConstructor的错误,但我认为这些错误在一个单独的.cpp文件中,您忘记添加到目标中。

您可能也可以去掉大多数find_libraries调用,但如果不知道源文件的样子,我就无法做到这一点。