I wrote such code:
template <class T>
class A {
template <class U, class =
class std::enable_if_t<std::is_convertible_v<std::decay_t<U>, std::decay_t<T>>>>
void f(U&& val) {}
};
I want that user of my class can call f
only with types that convertible to T
.
Is std::decay
redundant there? If I remove it maybe I miss some special cases?
I think your question is more philosophical, as in: In the universe of types in C++, are there any cases of T and U where there is an observable difference between calling f() and g() on the following class:
What does decay_t actually do?
It may be worth noting: decay_t is modeled on what happens to function argument types when passed into a function.
(UPDATED)
If
U
is taken by value, thendecay_t<U>
will be equivalent to U.But since
U
is a universal reference (in your example), it is possible that U deduces to reference type, so decaying it will have a visible effect. WhenT
can only be constructed from a reference, then decaying it will change the answer of this type trait.Merry's excellent example: