1. ホーム
  2. c++

[解決済み] テンプレート関数への未定義の参照 [重複]。

2022-10-10 17:31:52

質問

私は3つのファイルを持っています。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

テンプレートの宣言や実装に何か問題があるのでしょうか? なぜこのようなリンクエラーが発生するのでしょう :?

どのように解決すればよいのでしょうか?

非特殊化テンプレートの実装は、それを使用する翻訳ユニットから見えるようにする必要があります。

コンパイラは、コード内のすべての特殊化のためのコードを生成するために、実装を見ることができなければなりません。

これは2つの方法で達成することができます。

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