Undefined ссылка на функцию шаблона

У меня есть три файла. Содержимое 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);
        }
}

Когда я пытаюсь скомпилировать эти файлы со следующей командой, я получаю ссылку undefined reference error

[email protected]:~/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 я получаю эти ошибки связывания:?

Ответ 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);
        }
}

Ответ 2

У вас есть 2 способа:

  • Внедрить convert2QString в util.h.

  • Вручную создайте convert2QString с помощью int в util.cpp и определите эту специализацию как функцию extern в util.h

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); 
}