質問をすることでしか得られない、回答やアドバイスがある。

15分調べてもわからないことは、質問しよう!

新規登録して質問してみよう
ただいま回答率
85.35%
関数

関数(ファンクション・メソッド・サブルーチンとも呼ばれる)は、はプログラムのコードの一部であり、ある特定のタスクを処理するように設計されたものです。

C++

C++はC言語をもとにしてつくられた最もよく使われるマルチパラダイムプログラミング言語の1つです。オブジェクト指向、ジェネリック、命令型など広く対応しており、多目的に使用されています。

Q&A

解決済

1回答

1822閲覧

引数なしの可変長引数テンプレート関数のオーバーロードの解決方法

vann_2921

総合スコア190

関数

関数(ファンクション・メソッド・サブルーチンとも呼ばれる)は、はプログラムのコードの一部であり、ある特定のタスクを処理するように設計されたものです。

C++

C++はC言語をもとにしてつくられた最もよく使われるマルチパラダイムプログラミング言語の1つです。オブジェクト指向、ジェネリック、命令型など広く対応しており、多目的に使用されています。

0グッド

0クリップ

投稿2020/09/06 21:03

前提・実現したいこと

以下の可変長引数テンプレートメンバ関数Init()のオーバーロードを解決したい。

c++

1#include <vector> 2#include <type_traits> 3 4struct Base {}; 5struct Derived : Base {}; 6 7template<class... T> 8struct A { 9 A() { Init<T...>(); } 10private: 11 template<class Head, class... Body> 12 void Init() { 13 if (std::is_base_of<Base, Head>) return; 14 bases_.emplace_back(new Head); 15 Init<Body...>(); 16 } 17 template<class Head> 18 void Init() { 19 if (std::is_base_of<Base, Head>) return; 20 }; 21 std::vector<Base*> bases_; 22}; 23 24 25int main() { 26 A<Derived> a; 27}

発生している問題・エラーメッセージ

'void A<Derived>::Init<Derived>(void)' の可能性があります または 'void A<Derived>::Init<Derived,>(void)'

気になる質問をクリップする

クリップした質問は、後からいつでもMYページで確認できます。

またクリップした質問に回答があった際、通知やメールを受け取ることができます。

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

guest

回答1

0

ベストアンサー

C++ におけるオーバーロードは、候補をひとつに絞れないならエラーです。 この場合は (オーバーロードではなく) 部分特殊化であるべきなのですが、関数は部分特殊化はできません。

このようなときによく使われるテクニックとしては、クラス内にヘルパとなるクラスを作って部分特殊化する方法があります。

具体的なコードとしてはこのようなものです。

#include <vector> #include <type_traits> struct Base {}; struct Derived : Base {}; template<class... T> struct A { A(void) { Init<T...>(); } private: template<class Head, class... Body> struct init_helper { static void Init(std::vector<Base*>& bases_) { if (std::is_base_of<Base, Head>::value) return; bases_.emplace_back(new Head); init_helper<Body...>::Init(bases_); } }; template<class Head> struct init_helper<Head> { static void Init(std::vector<Base*>&) { if (std::is_base_of<Base, Head>::value) return; } }; template<class Head, class... Body> void Init(void) { init_helper<Head, Body...>::Init(this->bases_); } std::vector<Base*> bases_; }; int main(void) { A<Derived> a; }

投稿2020/09/06 23:53

SaitoAtsushi

総合スコア5684

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

vann_2921

2020/09/07 11:52

ヘルパークラスはクラスなので「関数とは違い」部分特殊化が可能で、そのクラスにInitを委譲しているという認識で大丈夫でしょうか?
SaitoAtsushi

2020/09/07 12:50

そうです。 関数で出来ないのでクラスでやっているというだけです。
guest

あなたの回答

tips

太字

斜体

打ち消し線

見出し

引用テキストの挿入

コードの挿入

リンクの挿入

リストの挿入

番号リストの挿入

表の挿入

水平線の挿入

プレビュー

15分調べてもわからないことは
teratailで質問しよう!

ただいまの回答率
85.35%

質問をまとめることで
思考を整理して素早く解決

テンプレート機能で
簡単に質問をまとめる

質問する

関連した質問