获取架构 arm64 错误的未定义符号

Getting undefined symbols for architecture arm64 error

提问人:Paul 提问时间:12/29/2021 最后编辑:Paul 更新时间:12/29/2021 访问量:9932

问:

我在 M1 Pro Mac (CLion) 上构建,编译时出现以下错误:

FAILED: tree 
: && /Library/Developer/CommandLineTools/usr/bin/c++ -g -arch arm64 -isysroot /Library/Developer/CommandLineTools/SDKs/MacOSX12.0.sdk -Wl,-search_paths_first -Wl,-headerpad_max_install_names  CMakeFiles/tree.dir/main.cpp.o CMakeFiles/tree.dir/Node.cpp.o CMakeFiles/tree.dir/Tree.cpp.o CMakeFiles/tree.dir/Unit.cpp.o -o tree   && :
Undefined symbols for architecture arm64:
  "Node<Unit>::addRoot(Unit)", referenced from:
      _main in main.cpp.o
  "Node<Unit>::Node()", referenced from:
      _main in main.cpp.o
ld: symbol(s) not found for architecture arm64
clang: error: linker command failed with exit code 1 (use -v to see invocation)
ninja: build stopped: subcommand failed.

我有三个类,和 .我遇到的大多数帖子都表明其中一个文件不包含在项目中(例如,使用 g++ 编译器的架构 arm64 的未定义符号)。我怀疑这是问题所在,因为它们似乎都包含在编译器行中。NodeTreeUnit

我的样子是这样的:CMakeLists.txt

cmake_minimum_required(VERSION 3.21)
project(tree)

set(CMAKE_CXX_STANDARD 14)

add_executable(tree main.cpp Node.cpp Node.h Tree.cpp Tree.h Unit.cpp Unit.h)

我正在为该类添加头文件,因为这似乎是冒犯编译器的头文件,但是我怀疑这是问题所在。猜猜是一些制作文件设置?Unit


#include <iostream>
#include <vector>

using namespace std;

template <class T>
class Node {
    weak_ptr<Node> parent;
    vector<shared_ptr<shared_ptr<Node>>> children;
    T data;

public:
    Node();
    Node(weak_ptr<Node> parent, T &data);

    void addRoot(T node_data);
};

在这篇文章中可能会过于冗长,以下是输出:CMakeError.log

Compiling the C compiler identification source file "CMakeCCompilerId.c" failed.
Compiler: /Library/Developer/CommandLineTools/usr/bin/cc 
Build flags: 
Id flags:  

The output was:
1
ld: library not found for -lSystem
clang: error: linker command failed with exit code 1 (use -v to see invocation)


Compiling the CXX compiler identification source file "CMakeCXXCompilerId.cpp" failed.
Compiler: /Library/Developer/CommandLineTools/usr/bin/c++ 
Build flags: 
Id flags:  

The output was:
1
ld: library not found for -lc++
clang: error: linker command failed with exit code 1 (use -v to see invocation)

如果我运行,则输出:c++ main.cpp Node.cpp Tree.cpp Unit.cpp

In file included from main.cpp:3:
./Node.h:16:38: error: a space is required between consecutive right angle brackets (use '> >')
    vector<shared_ptr<shared_ptr<Node>>> children;
                                     ^~
                                     > > 
main.cpp:7:5: warning: 'auto' type specifier is a C++11 extension [-Wc++11-extensions]
    auto unit = std::make_unique<Unit>(1, "Test Unit");
    ^
main.cpp:7:22: error: no member named 'make_unique' in namespace 'std'
    auto unit = std::make_unique<Unit>(1, "Test Unit");
                ~~~~~^
main.cpp:7:34: error: 'Unit' does not refer to a value
    auto unit = std::make_unique<Unit>(1, "Test Unit");
                                 ^
./Unit.h:11:7: note: declared here
class Unit {
      ^
main.cpp:7:40: warning: expression result unused [-Wunused-value]
    auto unit = std::make_unique<Unit>(1, "Test Unit");
                                       ^
main.cpp:8:20: error: expected ';' at end of declaration
    Node<Unit> node {};
                   ^
                   ;
2 warnings and 4 errors generated.
In file included from Node.cpp:5:
./Node.h:16:38: error: a space is required between consecutive right angle brackets (use '> >')
    vector<shared_ptr<shared_ptr<Node>>> children;
                                     ^~
                                     > > 
1 error generated.

我是 C++ 的新手,所以我发现这个问题有点麻烦。

C++ 模板 链接器错误

评论

0赞 12/29/2021
尝试手动编译。只是告诉我发生了什么。c++ main.cpp Node.cpp Tree.cpp Unit.cpp
0赞 w08r 12/29/2021
我甚至会说 xtools 问题。你能编译一个简单的hello world示例吗?
0赞 Paul 12/29/2021
@wobr,是的,如果我在所有编译中都很好地删除了对构造函数的调用。Unitmain.cpp
1赞 12/29/2021
哦,当然,您必须添加到命令行中。如果你回到 cmake,你可以在调试模式下运行,查看命令行。-std=c++14make VERBOSE=1
2赞 12/29/2021
你在哪里定义 Node::Node() ?如果类是模板化的,则必须 (1) 在头文件中包含模板化代码的实现,或者 (2) 在正文 (cpp) 文件中对具体模板类进行显式专用化。

答:

1赞 user8143588 12/29/2021 #1

看起来您的模板有问题。在模板化类中,您必须将方法的实现全部放在头文件中,或者在定义它的 cpp 正文中对具体类进行显式模板实例化。

举个例子更容易看出。在下面的例子中,我在标头中添加了构造函数的主体,因此我不必执行任何其他操作


/////////////////////////////////////////
// In Node.h
/////////////////////////////////////////
template< typename T >
struct Node {
    Node();
};

template< typename T >
Node<T>::Node() {
}

/////////////////////////////////////////
// In Node.cpp
/////////////////////////////////////////
// (nothing)

/////////////////////////////////////////
// Int main.cpp
/////////////////////////////////////////
#include "Node.h"
int main() {
   Node<int> n;
}

在第二种情况下,我将对 Node.cpp 主体中的具体类进行显式模板实例化,这样我就不需要公开模板化类的实现。Node<int>

/////////////////////////////////////////
// In Node.h
/////////////////////////////////////////
template< typename T >
struct Node {
    Node();
};

/////////////////////////////////////////
// In Node.cpp
/////////////////////////////////////////

template< typename T >
Node<T>::Node() {}

// Explicit initialization 
template struct Node<int>;

/////////////////////////////////////////
// Int main.cpp
/////////////////////////////////////////
#include "Node.h"
int main() {
   Node<int> n;
}

我更喜欢使用第二种方法,因为它大大缩短了构建时间。

评论

1赞 Paul 12/29/2021
是的,就是这样。棒。