C++のtemplateを使いたいのですが
解決済
回答 2
投稿
- 評価
- クリップ 1
- VIEW 1,788
前提・実現したいこと
map_doubleという関数を使って全ての要素を2倍にしたいのですが、うまくいきません。
#include <iostream>
using namespace std;
template<int head = 0, int... tail>
class Turing{
public:
template <int new_head>
Turing<new_head, head, tail...> cons(){
return Turing<new_head, head, tail...>();
}
void print(){
cout << head << " " << endl;
if(sizeof...(tail) != 0) Turing<tail...>().print();
}
auto map_double(){
cout << head << " " << endl;
if(sizeof...(tail) != 0){
return Turing<tail...>().map_double().cons<head * 2>();
}
return Turing<head * 2>();
}
};
int main(){
auto a = Turing<1, 2, 3>().map_double().print();
return 0;
}
発生している問題・エラーメッセージ
template_test.cpp:22:66: error: expected expression
return Turing<tail...>().map_double().cons<head * 2>();
^
1 error generated.
補足情報(FW/ツールのバージョンなど)
Apple LLVM version 9.0.0 (clang-900.0.39.2)
Target: x86_64-apple-darwin16.7.0
Thread model: posix
-
気になる質問をクリップする
クリップした質問は、後からいつでもマイページで確認できます。
またクリップした質問に回答があった際、通知やメールを受け取ることができます。
クリップを取り消します
-
良い質問の評価を上げる
以下のような質問は評価を上げましょう
- 質問内容が明確
- 自分も答えを知りたい
- 質問者以外のユーザにも役立つ
評価が高い質問は、TOPページの「注目」タブのフィードに表示されやすくなります。
質問の評価を上げたことを取り消します
-
評価を下げられる数の上限に達しました
評価を下げることができません
- 1日5回まで評価を下げられます
- 1日に1ユーザに対して2回まで評価を下げられます
質問の評価を下げる
teratailでは下記のような質問を「具体的に困っていることがない質問」、「サイトポリシーに違反する質問」と定義し、推奨していません。
- プログラミングに関係のない質問
- やってほしいことだけを記載した丸投げの質問
- 問題・課題が含まれていない質問
- 意図的に内容が抹消された質問
- 過去に投稿した質問と同じ内容の質問
- 広告と受け取られるような投稿
評価が下がると、TOPページの「アクティブ」「注目」タブのフィードに表示されにくくなります。
質問の評価を下げたことを取り消します
この機能は開放されていません
評価を下げる条件を満たしてません
質問の評価を下げる機能の利用条件
この機能を利用するためには、以下の事項を行う必要があります。
- 質問回答など一定の行動
-
メールアドレスの認証
メールアドレスの認証
-
質問評価に関するヘルプページの閲覧
質問評価に関するヘルプページの閲覧
checkベストアンサー
+2
expected expression
の直接的な原因は、メンバ関数呼び出しの部分のテンプレート引数が正しくテンプレート引数として扱われていないことにあります。こういう場合、メンバ関数テンプレート名の前に template
をつける必要があります。
map_double().template cons<head*2>();
どうパースされているのかまでは私には分からないのですが、不等号か何かかと思っているのでしょうね?
この記法を使って修正したコードが次です。
なお、 if constexpr
文は tail
の数が 0 でないときはそもそも Turing<tail...>
を展開しないようにするために利用しています。これにより template <int head = 0, ...>
のような無意味なデフォルト引数が必要なくなります。 (無意味でなかったならごめんなさい) 。
#include <iostream>
using namespace std;
template<int head, int... tail>
struct Turing {
template <int new_head>
Turing<new_head, head, tail...> cons() const {
return Turing<new_head, head, tail...>();
}
void print() {
cout << head << " " << endl;
if constexpr (sizeof...(tail) != 0)
Turing<tail...>().print();
}
auto map_double() {
if constexpr (sizeof...(tail) != 0){
return Turing<tail...>().map_double().template cons<head*2>();
} else {
return Turing<head * 2>();
}
}
};
int main(){
auto a = Turing<1, 2, 3, 4>().map_double();
a.print();
return 0;
}
投稿
-
回答の評価を上げる
以下のような回答は評価を上げましょう
- 正しい回答
- わかりやすい回答
- ためになる回答
評価が高い回答ほどページの上位に表示されます。
-
回答の評価を下げる
下記のような回答は推奨されていません。
- 間違っている回答
- 質問の回答になっていない投稿
- スパムや攻撃的な表現を用いた投稿
評価を下げる際はその理由を明確に伝え、適切な回答に修正してもらいましょう。
+1
こんにちは。
外しているかもしれませんが、テンプレート・パラメータを倍にして表示してみました。
#include <iostream>
template<int... kArgs>
struct Foo
{
void print()
{
using swallow = int[];
(void)swallow
{
(std::cout << kArgs << " ", 0)...
};
std::cout << "\n";
}
Foo<(kArgs*2)...> map_double() const
{
return Foo<(kArgs*2)...>();
}
};
int main()
{
Foo<1, 2, 3>().print();
Foo<1, 2, 3>().map_double().print();
Foo<1, 2, 3>().map_double().map_double().print();
}
実行結果:
1 2 3
2 4 6
4 8 12
【コメントに対する回答】
print()を再帰定義してみました。
#include <iostream>
template<int... kArgs>
class Foo
{
template<typename...>
void printImpl()
{
}
template<int kFirst, int... kRest>
void printImpl()
{
std::cout << kFirst << " ";
printImpl<kRest...>();
}
public:
void print()
{
printImpl<kArgs...>();
std::cout << "\n";
}
Foo<(kArgs*2)...> map_double() const
{
return Foo<(kArgs*2)...>();
}
};
int main()
{
Foo<>().print();
Foo<1>().print();
Foo<1, 2>().print();
Foo<1, 2, 3>().print();
Foo<1, 2, 3>().map_double().print();
Foo<1, 2, 3>().map_double().map_double().print();
}
template<typename...>
void printImpl()
{
}
にて再帰の最後でテンプレート引数が無くなった時、および、テンプレート実引数が0個の時(Foo<>)に対応してみました。「曖昧」にならないかちょっと心配ですが、gcc, clang, msvcで通りましたし、MSDNのこの記述を見る限り使って良さそうです。
投稿
-
回答の評価を上げる
以下のような回答は評価を上げましょう
- 正しい回答
- わかりやすい回答
- ためになる回答
評価が高い回答ほどページの上位に表示されます。
-
回答の評価を下げる
下記のような回答は推奨されていません。
- 間違っている回答
- 質問の回答になっていない投稿
- スパムや攻撃的な表現を用いた投稿
評価を下げる際はその理由を明確に伝え、適切な回答に修正してもらいましょう。
15分調べてもわからないことは、teratailで質問しよう!
- ただいまの回答率 88.23%
- 質問をまとめることで、思考を整理して素早く解決
- テンプレート機能で、簡単に質問をまとめられる
2018/04/30 15:27
constexprすごいですね。ありがとうございます