Simplify the SFINAE construct a bit
authorMikko Rasa <tdb@tdb.fi>
Fri, 24 May 2019 16:57:22 +0000 (19:57 +0300)
committerMikko Rasa <tdb@tdb.fi>
Fri, 24 May 2019 16:57:22 +0000 (19:57 +0300)
More of the common stuff is now stored in the helper struct.

source/core/meta.h
source/strings/lexicalcast.h

index a13f5cb3a5714de60aba9a7bdc5682370aac110a..d6ca35935543049d8026b842ecca40785a92ecae 100644 (file)
@@ -43,10 +43,13 @@ struct Sfinae
        struct Yes { char c[2]; };
        struct No { char c; };
 
-       template<size_t s>
+       template<typename T>
+       static No f(...);
+
+       template<typename C, typename T>
        struct Evaluate
        {
-               enum { value = (s==sizeof(Yes)) };
+               enum { value = (sizeof(C::template f<T>(0))==sizeof(Yes)) };
        };
 };
 
index 2ed0621291858c0d0f4d40fb37036fd5decc5700..13eab45f0b81c9311381f43f880caf2abb5fcf5b 100644 (file)
@@ -93,36 +93,25 @@ void operator>>(const LexicalConverter &, std::string &);
 
 // Generic operators using stringstream
 
-template<typename T>
-struct HasFormattedOutput: Sfinae
+struct CheckFormattedOutput: Sfinae
 {
        static std::ostream &s;
-       static T &v;
-
-       /* The expression must depend on the template parameter, or the compiler
-       will give an error. */
-       template<typename U>
-       static Yes f(int (*)[sizeof(s<<HasFormattedOutput<U>::v)]);
-       template<typename U>
-       static No f(...);
-
-       enum { value = Evaluate<sizeof(f<T>(0))>::value };
+       template<typename T>
+       static Yes f(int (*)[sizeof(s<<T())]);
+       using Sfinae::f;
 };
 
-template<typename T>
-struct HasFormattedInput: Sfinae
+struct CheckFormattedInput: Sfinae
 {
        static std::istream &s;
-       static T &v;
-
-       template<typename U>
-       static Yes f(int (*)[sizeof(s>>HasFormattedOutput<U>::v)]);
-       template<typename U>
-       static No f(...);
-
-       enum { value = Evaluate<sizeof(f<T>(0))>::value };
+       template<typename T>
+       static Yes f(int (*)[sizeof(s>>reinterpret_cast<T &>(s))]);
+       using Sfinae::f;
 };
 
+template<typename T> struct HasFormattedOutput: Sfinae::Evaluate<CheckFormattedOutput, T> { };
+template<typename T> struct HasFormattedInput: Sfinae::Evaluate<CheckFormattedInput, T> { };
+
 
 template<typename T>
 typename EnableIf<HasFormattedOutput<T>::value, void>::Yes