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

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

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

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

Q&A

解決済

1回答

1277閲覧

decltypeとSFINAEによる関数のオーバーロード条件指定のときの、戻り値の型指定について

wistaile

総合スコア24

C++

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

0グッド

0クリップ

投稿2020/05/30 13:21

以下のようなMyClass

C++

1class MyClass { 2 // メンバ変数 3 ... 4 5 public: 6 // デフォルトコンストラクタ無効 7 MyClass() = delete; 8 // 何らかの面倒くさいコンストラクタ 9 MyClass(int, ...); 10};

を戻り値とする、SFINAEによって条件指定した以下のような関数を作りたいです。

C++

1template <class T> 2auto my_function(T t, ClassA a, ClassB b) -> decltype(/*tに関する条件1*/, /*tに関する条件2*/, MyClass{});

decltypeの最後の引数にはMyClass型の値を指定する必要があると理解していますが、MyClassはデフォルトコンストラクタが存在しないため、上のコードはコンパイルできません。(MyClass{}が不可)

コンストラクタを指定せずに、MyClass型の値を指定することはできますか?

-> decltype(std::declval<MyClass>())としても、うまくいきませんでした。

-> std::decay_t<decltype(std::declval<MyClass>())>
とすれば通りましたが、これは適切なのでしょうか。

ご教示ください。よろしくおねがい致します。

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

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

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

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

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

guest

回答1

0

ベストアンサー

こんにちは。

-> decltype(std::declval<MyClass>())としても、うまくいきませんでした。

本質的にはこれで良いはずです。
注意点としては、declvalは(右辺値)参照を返すので、参照を外しておかないと戻り値の型が参照になってしまいます。

C++

1#include<iostream> 2#include<type_traits> 3 4class MyClass { 5 // メンバ変数 6 7 public: 8 // デフォルトコンストラクタ無効 9 MyClass() = delete; 10 // 何らかの面倒くさいコンストラクタ 11 MyClass(int) {} 12}; 13 14struct A { void foo() {} }; 15struct B { void bar() {} }; 16 17template <class T> 18auto my_function(T t) -> std::remove_reference_t<decltype(t.foo(), std::declval<MyClass>())> 19{ 20 std::cout << "my_function(arg has foo())\n"; 21 return MyClass{1}; 22} 23 24template <class T> 25auto my_function(T t) -> std::remove_reference_t<decltype(t.bar(), std::declval<MyClass>())> 26{ 27 std::cout << "my_function(arg has bar())\n"; 28 return MyClass{1}; 29} 30 31int main() 32{ 33 my_function(A{}); // my_function(arg has foo()) 34 my_function(B{}); // my_function(arg has bar()) 35}

wandbox

投稿2020/05/30 14:20

Chironian

総合スコア23272

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

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

yumetodo

2020/05/30 14:24

まあどっちかというと、一度SFINAEで条件チェックだけ行うtraisを作ってenable_if_tで再度やるほうが条件の使い回し的に楽だと思いますけどね
wistaile

2020/05/30 14:49

ありがとうございます! なるほど、declvalはadd_rvalue_reference_t<T>を返却していたのですね。勉強になりました。
guest

あなたの回答

tips

太字

斜体

打ち消し線

見出し

引用テキストの挿入

コードの挿入

リンクの挿入

リストの挿入

番号リストの挿入

表の挿入

水平線の挿入

プレビュー

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

ただいまの回答率
85.35%

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

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

質問する

関連した質問