对模板函数的未定义引用 [duplicate]

undefined reference to template function [duplicate]

提问人:Vihaan Verma 提问时间:5/17/2012 最后编辑:Vihaan Verma 更新时间:11/14/2014 访问量:107843

问:

我有三个文件。main.cpp 的内容是

#include<iostream>
#include<QString>

#include "util.h"

int main()
{
    using Util::convert2QString;

    using namespace std;
    int n =22;
    QString tmp = convert2QString<int>(n);

    return 0;
}

util.h

namespace Util
{
    template<class T>
    QString convert2QString(T type , int digits=0);
}

util.cpp

namespace Util
{
    template<class T>
        QString convert2QString(T type, int digits=0)
        {
            using std::string;

            string temp = (boost::format("%1%") % type).str();

            return QString::fromStdString(temp);
        }
}

当我尝试使用以下命令编译这些文件时,出现未定义的引用错误

vickey@tb:~/work/trash/template$ g++ main.cpp  util.cpp -lQtGui -lQtCore  -I. -I/usr/local/Trolltech/Qt-4.8.0/include/QtCore -I/usr/local/Trolltech/Qt-4.8.0/include/QtGui -I/usr/local/Trolltech/Qt-4.8.0/include
/tmp/cca9oU6Q.o: In function `main':
main.cpp:(.text+0x22): undefined reference to `QString Util::convert2QString<int>(int, int)'
collect2: ld returned 1 exit status

模板声明或实现有问题吗?为什么 M I 会出现这些链接错误:?

C++ Qt Boost 未定义引用

评论


答:

211赞 Luchian Grigore 5/17/2012 #1

非专用模板的实现必须对使用它的翻译单元可见。

编译器必须能够看到实现,才能为代码中的所有专用化生成代码。

这可以通过两种方式实现:

1) 将实现移动到标头内。

2)如果要将其分开,请将其移动到原始标题中包含的其他标题中:

util.h

namespace Util
{
    template<class T>
    QString convert2QString(T type , int digits=0);
}
#include "util_impl.h"

util_impl.h

namespace Util
{
    template<class T>
        QString convert2QString(T type, int digits=0)
        {
            using std::string;

            string temp = (boost::format("%1") % type).str();

            return QString::fromStdString(temp);
        }
}

评论

27赞 Matteo Italia 5/17/2012
许多人将扩展名用于模板实现文件。.tcc
30赞 lashgar 6/4/2020
我第一次听到.tcc
45赞 inkooboo 5/17/2012 #2

您有 2 种方法:

  1. 在 util.h 中实现。convert2QString

  2. 使用 in util.cpp 手动实例化,并在 util.h 中将此专用化定义为 extern 函数convert2QStringint

util.h

namespace Util
{
    template<class T>
    QString convert2QString(T type , int digits=0);

    extern template <> QString convert2QString<int>(int type , int digits);
}

util.cpp

 namespace Util {
     template<class T>
     QString convert2QString(T type, int digits)
     {
         using std::string;

         string temp = (boost::format("%1") % type).str();

         return QString::fromStdString(temp);
     }

     template <> QString convert2QString<int>(int type , int digits); 
}

评论

0赞 Saam 8/6/2023
extern template在.h文件中有错误
0赞 Brandlingo 8/23/2023
是的,没尖括号就是正确的。extern template QString convert2QString<int>(int type , int digits);