C++編で扱っている C++ は 2003年に登場した C++03
という、とても古いバージョンのものです。C++ はその後、C++11 -> C++14
-> C++17 -> C++20 -> C++23 と更新されています。
なかでも C++11 での更新は非常に大きなものであり、これから C++
の学習を始めるのなら、C++11
よりも古いバージョンを対象にするべきではありません。特に事情がないなら、新しい
C++ を学んでください。 当サイトでは、C++14 をベースにした新C++編を作成中です。
この章の概要です。
関数がテンプレート化できるのと同様に、メンバ関数をテンプレート化できます。この場合、メンバ関数テンプレート(メンバテンプレート)と呼ばれます。
なお、メンバ関数テンプレートは仮想関数(第27章)にできません。
メンバ関数テンプレートは、所属するクラス自体はテンプレートでない場合と、クラス自体がクラステンプレートになっている場合とがあります。
まず、所属するクラス自体はテンプレートでない場合の形を見ていきましょう。
#include <iostream>
class DataStore {
public:
explicit DataStore(double value) :
(value)
mValue{}
template <typename U>
inline const U GetValue() const
{
return static_cast<U>(mValue);
}
private:
double mValue;
};
int main()
{
(10.5);
DataStore ds
std::cout << ds.GetValue<int>() << "\n"
<< ds.GetValue<double>() << std::endl;
}
実行結果:
10
10.5
DataStoreクラスの GetValue関数テンプレートは、テンプレート実引数に指定した型へ static_cast で変換を行い、値を返しています。この例のような型変換の実現のために、メンバ関数テンプレートを使うことはよくあります。
なお、メンバ関数テンプレートの実装をクラス外に書く場合は、次のようになります。
template <typename U>
const U DataStore::GetValue() const
{
return static_cast<U>(mValue);
}
「template <typename U>」の部分が必要なのは、クラステンプレートのメンバ関数のときと同様ですが、DataStore は通常のクラスなので、「DataStore<T>::」ではなく「DataStore::」になります。
ところで、先ほどの例の場合、「ds.GetValue()」のように型指定がない使い方ができません。型変換を目的としておらず、デフォルトの型(この場合は double型)のまま返してほしい場合には、明示的な型指定無しで使える方が便利でしょう。そこで、テンプレートでない通常のメンバ関数としての GetValue() を追加します。
#include <iostream>
class DataStore {
public:
explicit DataStore(double value) :
(value)
mValue{}
inline double GetValue() const
{
return mValue;
}
template <typename U>
inline const U GetValue() const
{
return static_cast<U>(GetValue());
}
private:
double mValue;
};
int main()
{
(10.5);
DataStore ds
std::cout << ds.GetValue<int>() << "\n"
<< ds.GetValue() << std::endl;
}
実行結果:
10
10.5
テンプレート版の GetValue関数は、非テンプレート版の GetValue関数を呼び出す形に変更しています。これは必須ではありませんが、目的から言って、呼出し元へ返す型を変換すること以外は同じにしたいはずなので、保守の面から、処理を移譲させるようにしました。無限再帰してしまいそうに見えますが、非テンプレート版はテンプレート版よりも優先度が高いので問題ありません。
次に、クラステンプレート内でメンバ関数テンプレートを使う形を見てみましょう。先ほどの DataStoreクラスをクラステンプレートに変えたものを例に挙げます。
#include <iostream>
template <typename T>
class DataStore {
public:
explicit DataStore(const T& value) :
(value)
mValue{}
inline const T GetValue() const
{
return mValue;
}
template <typename U>
inline const U GetValue() const
{
return static_cast<U>(GetValue());
}
private:
;
T mValue};
int main()
{
<double> ds(10.5);
DataStore
std::cout << ds.GetValue<int>() << "\n"
<< ds.GetValue() << std::endl;
}
実行結果:
10
10.5
クラステンプレートになっても、メンバ関数テンプレートの宣言に変化はありません。ただし、実装をクラステンプレートの定義の外側に書く場合は、クラステンプレートのテンプレート仮引数と、メンバ関数テンプレートのテンプレート仮引数の両方を記述しなければならないため、次のように少々面倒な形になります。
template <typename T>
template <typename U>
const U DataStore<T>::GetValue() const
{
return static_cast<U>(GetValue());
}
C++11 では、メンバ関数テンプレートのテンプレート仮引数に、デフォルトの引数を与えることが可能になりました。
#include <iostream>
class DataStore {
public:
explicit DataStore(double value) :
(value)
mValue{}
template <typename U = double>
inline const U GetValue() const
{
return static_cast<U>(mValue);
}
private:
double mValue;
};
int main()
{
(10.5);
DataStore ds
std::cout << ds.GetValue<int>() << "\n"
<< ds.GetValue() << std::endl;
}
実行結果:
10
10.5
コンストラクタもテンプレート化できます。これは、コンストラクタテンプレートと呼ばれます。
一般に、コンストラクタテンプレートの意義は、クラステンプレートに異なるテンプレート実引数を与えてインスタンス化されたオブジェクトがあるとき、互いの暗黙的な変換を受け付けることにあります。
たとえば、DataStoreクラステンプレートが、次のようにコピーコンストラクタを定義していたとします。
template <typename T>
class DataStore {
public:
(const DataStore<T>& rhs);
DataStore};
コピーコンストラクタがあれば、DataStore<int>型のオブジェクトを使って、DataStore<int>型の新たなオブジェクトを作り出せます。
<int> iStore(10);
DataStore<int> iStore2(iStore); // OK DataStore
ところが、次のコードはコンパイルエラーになります。
<int> iStore(10);
DataStore<double> dStore(iStore); // エラー DataStore
これは、DataStore<int>型のオブジェクトを使って、DataStore<double>型の新たなオブジェクトを作ろうとしている訳ですが、DataStoreクラステンプレートのテンプレート仮引数 T に当てはめられる型が異なるため、定義したコピーコンストラクタが使えずに、コンパイルエラーになります。
このような場面で、テンプレート仮引数の違いがあっても、同一の型であるように扱いたい場合、コンストラクタテンプレートを利用して解決を図れます。
#include <iostream>
template <typename T>
class DataStore {
template <typename>
friend class DataStore;
public:
explicit DataStore(const T& value) :
(value)
mValue{}
// DataStore<U> からコピー
template <typename U>
(const DataStore<U>& rhs) :
DataStore(static_cast<T>(rhs.mValue))
mValue{}
inline const T GetValue() const
{
return mValue;
}
private:
;
T mValue};
int main()
{
<int> iStore(10);
DataStore<double> dStore(iStore); // OK
DataStore
std::cout << iStore.GetValue() << "\n"
<< dStore.GetValue() << std::endl;
}
実行結果:
10
10.0
コピーコンストラクタ自体がテンプレート化された訳ですが、形としては、メンバ関数テンプレートそのものであることが分かると思います。結果、DataStore<T> のオブジェクトを作るために DataStore<U> を使えるようになりました。
メンバ変数 mValue をコピーする際には、U型から T型への型変換が必要になります。ここで static_cast をしていますが、U型と T型の関係性が static_cast でキャストできないようなものであれば、この部分でコンパイルエラーが起きます。
また、このとき、DataStore<int> と DataStore<double> は別の型なので、「rhs.mValue」という相手方の「非公開」なメンバをアクセスしなくてはなりません。そのため、相手をフレンドクラス(第25章)として指定しておく必要があります。
第19章で説明した変換演算子についても、テンプレートにできます。テンプレート変換演算子により、任意の型への型変換が定義できます。
#include <iostream>
class DataStore {
public:
explicit DataStore(double value) :
(value)
mValue{}
template <typename T>
inline operator T() const
{
return static_cast<T>(mValue);
}
private:
double mValue;
};
int main()
{
(123.456);
DataStore fStore
int i = fStore;
double f = fStore;
std::cout << i << "\n"
<< f << std::endl;
}
実行結果:
123
123.456
テンプレート変換演算子は、代入や初期化の際の左辺側の型によって、テンプレート仮引数を推測していると考えられます。
テンプレート変換演算子と、テンプレートでない変換演算子を共存させることは可能です。その場合、いつものように、より一致度が高い変換演算子が優先的に使用されるので、たとえば上記の例で言えば、int型に変換する場合にだけ、テンプレートでない変換演算子を使わせるといったこともできます。
問題① 次のようなクラスがあります。
template <typename T>
class DataStoreArray {
public:
void Push(const T& data)
{
.push_back(data);
mVec}
private:
std::vector<T> mVec;
};
このクラスに、保持している要素を、STLコンテナにコピーして返すようなテンプレート変換演算子を追加してください(map はキーと値のペアが必要になるので省いて良いです)。つまり、以下の変換を可能にしてください。
<int> store;
DataStoreArray
std::vector<int> v = store;
std::list<int> l = store;
std::set<int> s = store;
コンパイラの対応状況について、対応している場合は明記しない方針にした。
clang 3.7 (Xcode 7.3) を、Xcode 8.3.3 に置き換え。
VisualC++ 2017 に対応。
新規作成。
Programming Place Plus のトップページへ
はてなブックマーク に保存 | Pocket に保存 | Facebook でシェア |
X で ポスト/フォロー | LINE で送る | noteで書く |
RSS | 管理者情報 | プライバシーポリシー |