未定义的引用错误

undefined reference error

本文关键字:错误 引用 未定义      更新时间:2023-10-16

我有以下简单的模板代码:

#ifndef CLUSTER_H
#define CLUSTER_H
#include <iostream>
#include <vector>
template <typename T, size_t K>
class Cluster
{
public:
    void Print() const;     
private:
    std::vector<T> objects;
};
template <typename T, size_t K>
void Cluster<T,K>::Print() const
{
    for (int i=0; i<objects.size(); i++)
    {
        T curr=objects[i];
        std::cout << curr << " ";
    }
    std::cout << std::endl;
}
#endif

由于某种原因,我收到以下错误:"未定义对'Cluster<int, 5u>::Print() const'的引用。可能是什么原因造成的?谢谢!

所以,我要在这里冒昧地说,你已经在 CPP 文件中定义了一个模板函数,这意味着它最终将出现在不同的翻译单元中。下面是一个简单的示例:

标头,示例.h

#ifndef EXAMPLE_H
#define EXAMPLE_H
template<int TValue>
class example
{
public:
    int get_tvalue();
};
#endif

源文件,例如.cpp

#include "example.h"
template<int TValue>
int example<TValue>::get_tvalue()
{
    return TValue;
}

另一个源文件,主文件.cpp

#include "example.h"
int main()
{
    example<5> instance;
    instance.get_tvalue();
    return 0;
}

如果我使用 GCC 将这些编译在一起,我会得到undefined reference to 'example<5>::get_tvalue()'.这是因为模板类的实例化方式。模板类定义就是这样...模板,而不是实际类。当该类的参数化(或具体而言,完全专用)定义发生时,将创建实际的类定义,在本例中为 example<5> 。这种完全专业化的类定义只存在于主要.cpp...示例中没有这样的类.cpp!示例.cpp仅包含模板,不包含专用项。这意味着函数,get_tvalue 没有为 main.cpp 中的example<5>定义,因此错误。

您可以通过以下两种方式之一解决此问题。第一种方法是始终在其头文件中定义整个模板类。例如,这是使用 STL 容器完成的方式。另一种方法是在示例中强制创建参数化类.cpp...您可以通过添加

template class example<5>;

到示例的末尾.cpp。因为现在在 example.cpp 中有一个实际的 example<5> 类定义,所以你还将得到一个example<5>::get_tvalue的实际函数定义,当你的翻译单元 main.o 和 example.o 在编译步骤结束时链接在一起时,一切都会好起来的。

显然,在大多数情况下,这将是一种糟糕的方法,但是在模板参数仅取一小部分值的情况下,它可以工作。不过,将整个类放在头文件中可能是最简单、最安全和最灵活的。