a couru dans une autre bizarrerie dans les guides de déduction de modèle après C++17 template deduction guide not used for empty parameter set? (ce bug https://gcc.gnu.org/bugzilla/show_bug.cgi?id=81486 est toujours pas fixé dans le coffre GCC malheureusement :():Guide de déduction de gabarit C++ 17 non utilisé pour le jeu de paramètres vide (ver 2)?
#include <utility>
template <class T> struct success
{
T value;
constexpr success(T &&v)
: value(std::move(v))
{
}
constexpr success(const T &v)
: value(v)
{
}
};
template <> struct success<void>
{
};
template <class T> success(T /*unused*/)->success<T>;
success()->success<void>;
template<class T> struct foo
{
foo(success<void>) {}
};
int main(void)
{
auto a = success{5}; // works
auto b = success{}; // works
auto c = success{"hello"}; // works
auto d = success(5); // works
//auto e = success(); // FAILS on GCC 7.2!
auto f = success("hello"); // works
foo<void> g(success()); // FAILS
static_assert(std::is_same<decltype(a), success<int>>::value, "");
static_assert(std::is_same<decltype(b), success<void>>::value, "");
static_assert(std::is_same<decltype(c), success<const char *>>::value, "");
static_assert(std::is_same<decltype(d), success<int>>::value, "");
//static_assert(std::is_same<decltype(e), success<void>>::value, "");
static_assert(std::is_same<decltype(f), success<const char *>>::value, "");
return 0;
}
La ligne surprenante, du moins pour moi, est que foo<void> g(success());
ne parvient pas à utiliser les guides de déduction de modèle sur les deux clang 6.0 tronc et le tronc GCC 7 que vous pouvez voir à https://godbolt.org/g/7m1Zhk
Je trouve cela surprenant et non ce que l'on pouvait s'y attendre. Le guide de modèle dit qu'un sans fioritures success()
doit être construit à success<void>
. cela devrait fonctionne très bien avec unambi de foo constructeur Guous acceptant un success<void>
. Au lieu de cela clang 6.0 rapports du tronc:
34 : <source>:34:17: error: use of class template 'success' requires template arguments; argument deduction not allowed in function return type
foo<void> g(success()); // FAILS
^~~~~~~
3 : <source>:3:27: note: template is declared here
template <class T> struct success
^
et GCC 7.3 Rapports du tronc:
<source>: In function 'int main()':
34 : <source>:34:25: error: 'auto' parameter not permitted in this context
foo<void> g(success()); // FAILS
^
Quelqu'un peut-il expliquer ce qui se passe ici? Est-ce un défaut de la norme C++ 17?
Ce message clang est en fait [bug 34091] (http://llvm.org/pr34091) – Rakete1111
Ok, merci pour la confirmation de la cause, il était difficile de choisir entre ça et l'autre réponse , les deux sont bons, mais celui-ci a aussi une solution de contournement donc je le pense mieux. –