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)個別に保持する場合は、元のヘッダーに含める別のヘッダーに移動します。
til.h
namespace Util
{
template<class T>
QString convert2QString(T type , int digits=0);
}
#include "util_impl.h"
til_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つの方法があります。
実装convert2QString
util.hで.
手動でインスタンス化convert2QString
をutil.cppのint
で使用し、この特殊化をutil.hのextern関数として定義します
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);
}