This commit is contained in:
Light-City 2020-01-07 10:59:02 +08:00
parent 19f9a3e401
commit 66fee47bed
23 changed files with 892 additions and 2 deletions

View File

@ -145,6 +145,7 @@ for(decl:col) {
- [字面量、静态断言和成员函数说明符](./modern_C++_30/literalAssert)
- [是不是应该返回对象?](./modern_C++_30/returnObj)
- [编译期多态:泛型编程和模板入门](./modern_C++_30/compilerpoly)
- [SFINAE不是错误的替换失败是怎么回事?](./modern_C++_30/SFINAE)
### 4.拓展部分
@ -172,7 +173,7 @@ Ubuntu 18.04
CLion gcc/g++
### 6.关于作者
### 6.关于作者
个人公众号:

View File

@ -1,7 +1,7 @@
cmake_minimum_required(VERSION 3.14)
project(Morden_C++)
set(CMAKE_CXX_STANDARD 11)
set(CMAKE_CXX_STANDARD 17)
add_executable(heap RAII/heap.cpp)
add_executable(stack RAII/stack.cpp)
@ -40,3 +40,25 @@ add_executable(literal literalAssert/literal.cpp)
add_executable(assert literalAssert/assert.cpp)
add_executable(default_delete literalAssert/default_delete.cpp)
add_executable(overridefinal literalAssert/overridefinal.cpp)
add_executable(SFINAE "SFINAE/SFINAE.cpp")
add_executable(serialize "SFINAE/sfinae paper/serialize.cpp")
add_executable(hana "SFINAE/sfinae paper/hana.cpp")
add_executable(overload "SFINAE/sfinae paper/overload1.cpp")
add_executable(pSFINAE "SFINAE/sfinae paper/p1SFINAE.cpp")
add_executable(p1SFINAE "SFINAE/sfinae paper/p2SFINAE.cpp")
add_executable(sizeof1 "SFINAE/sfinae paper/sizeof1.cpp")
add_executable(sizeof2 "SFINAE/sfinae paper/sizeof2.cpp")
add_executable(combining "SFINAE/sfinae paper/combiningAndGenius.cpp")
add_executable(overload2 "SFINAE/sfinae paper/overload2.cpp")
add_executable(combining1 "SFINAE/sfinae paper/timeGenius.cpp")
add_executable(decltype "SFINAE/sfinae paper/decltype.cpp")
add_executable(auto "SFINAE/sfinae paper/auto.cpp")
add_executable(constexpr "SFINAE/sfinae paper/constexpr.cpp")
add_executable(blending "SFINAE/sfinae paper/blending1.cpp")
add_executable(blending2 "SFINAE/sfinae paper/blending2.cpp")
add_executable(lambda "SFINAE/sfinae paper/lambda.cpp")
add_executable(is_valid "SFINAE/sfinae paper/is_valid.cpp")
add_executable(fis_valid "SFINAE/sfinae paper/fis_valid.cpp")
add_executable(packis_valid "SFINAE/sfinae paper/packis_valid.cpp")

View File

@ -0,0 +1,41 @@
//
// Created by light on 20-1-5.
//
#include <iostream>
using namespace std;
template<typename T>
class IsClassT {
private:
typedef char One;
typedef struct {
char a[2];
} Two;
template<typename C>
static One test(int C::*);
// Will be chosen if T is anything except a class.
template<typename C>
static Two test(...);
public:
enum {
Yes = sizeof(IsClassT<T>::test<T>(0)) == 1
};
enum {
No = !Yes
};
};
struct A {
};
int main() {
// 0不能转换为int int::*因为int不是类所以它不能有成员指针。
cout << IsClassT<int>::Yes << endl;
cout << IsClassT<A>::Yes << endl;
return 0;
}

View File

@ -0,0 +1,26 @@
//
// Created by light on 20-1-6.
//
#include <iostream>
bool f();
auto test = f(); // Famous usage, auto deduced that test is a boolean, hurray!
// vvv t wasn't declare at that point, it will be after as a parameter!
template<typename T>
decltype(t.serialize()) g(const T &t) {} // Compilation error
// Less famous usage:
// vvv auto delayed the return type specification!
// vvv vvv the return type is specified here and use t!
template<typename T>
auto g(const T &t) -> decltype(t.serialize()) {} // No compilation error.
auto myFunction() // Automagically figures out that myFunction returns ints.
{
return int();
}

View File

@ -0,0 +1,48 @@
//
// Created by light on 20-1-6.
//
#include <iostream>
#include "structData.h"
template<class T>
struct hasSerialize {
// We test if the type has serialize using decltype and declval.
// decltype将根据逗号表达式选择最后一个类型作为返回值
// 如果C有serialize将调用当前版本,没有serialize,则失败,此时编译时不报错,继续寻找重载函数,也就是SFINAE.
template<typename C>
static constexpr decltype(std::declval<C>().serialize(), bool()) test(int /* unused */) {
// We can return values, thanks to constexpr instead of playing with sizeof.
return true;
}
template<typename C>
static constexpr bool test(...) {
return false;
}
// int is used to give the precedence!
static constexpr bool value = test<T>(int());
};
template<class T>
std::string serialize(const T &obj) {
// 不加constexpr 报错error: no member named 'serialize' in 'A'.
if constexpr (hasSerialize<T>::value)
return obj.serialize();
else
return to_string(obj);
}
int main() {
A a;
B b;
C c;
// The following lines work like a charm!
std::cout << serialize(a) << std::endl;
std::cout << serialize(b) << std::endl;
std::cout << serialize(c) << std::endl;
}

View File

@ -0,0 +1,37 @@
//
// Created by light on 20-1-6.
//
#include <iostream>
#include "structData.h"
template<typename T, typename=std::string>
struct hasSerialize : std::false_type {
};
template<typename T>
struct hasSerialize<T, decltype(std::declval<T>().serialize())> : std::true_type {
};
template<class T>
std::string serialize(const T &obj) {
// 不加constexpr 报错error: no member named 'serialize' in 'A'.
if constexpr (hasSerialize<T>::value) // c++17特性
return obj.serialize();
else
return to_string(obj);
}
int main() {
A a;
B b;
C c;
// The following lines work like a charm!
std::cout << serialize(a) << std::endl;
std::cout << serialize(b) << std::endl;
std::cout << serialize(c) << std::endl;
}

View File

@ -0,0 +1,96 @@
//
// Created by light on 20-1-6.
//
#include <iostream>
#include "structData.h"
template<class T>
struct hasSerialize {
// 编译时比较
typedef char yes[1];
typedef char no[2];
// 允许我们检查序列化确实是一种方法
// 第二个参数必须是第一个参数的类型
// 例如: reallyHas<int,10> 替换为 reallyHas<int,int 10> 并起作用
// 注意:它仅适用于整数常量和指针(因此函数指针可以使用)
// 例如reallyHas<std::string (C::*)(), &C::serialize> 替换为
// reallyHas<std::string (C::*)(), std::string (C::*)() &C::serialize> 并起作用
template<typename U, U u>
struct reallyHas;
// std::string (C::*)() 是函数指针声明
template<typename C>
static yes &test(reallyHas<std::string (C::*)(), &C::serialize> * /*unused*/) {}
// ()()const 函数指针 -> std::string serialize() const
template<typename C>
static yes &test(reallyHas<std::string (C::*)() const, &C::serialize> * /*unused*/) {}
// The famous C++ sink-hole.
// Note that sink-hole must be templated too as we are testing test<T>(0).
// If the method serialize isn't available, we will end up in this method.
template<typename>
static no &test(...) { /* dark matter */ }
//用作测试的返回值的常数。
//由于编译时评估的大小,因此实际上在这里完成了测试。
static const bool value = sizeof(test<T>(0)) == sizeof(yes);
};
// Using the previous A struct and hasSerialize helper.
struct D : A {
std::string serialize() const {
return "I am a D!";
}
};
template<class T>
bool testHasSerialize(const T & /*t*/) { return hasSerialize<T>::value; }
// 不能够判断仿函数里面的serialize
struct E {
struct Functor {
std::string operator()() {
return "I am a E!";
}
};
Functor serialize;
};
template<class T>
std::string serialize(const T &obj) {
// 不加constexpr 报错error: no member named 'serialize' in 'A'.
if constexpr (hasSerialize<T>::value)
return obj.serialize();
else
return to_string(obj);
}
int main() {
// 检测结构体是否有serialize方法
// Using the struct A, B, C defined in the previous hasSerialize example.
std::cout << hasSerialize<A>::value << std::endl;
std::cout << hasSerialize<B>::value << std::endl;
std::cout << hasSerialize<C>::value << std::endl;
D d;
A &a = d; // Here we lost the type of d at compile time.
std::cout << testHasSerialize(d) << std::endl; // Output 1.
std::cout << testHasSerialize(a) << std::endl; // Output 0.
E e;
std::cout << e.serialize() << std::endl; // Succefully call the functor.
std::cout << testHasSerialize(e) << std::endl; // Output 0.
A a_;
B b_;
C c_;
std::cout << serialize(a_) << std::endl;
std::cout << serialize(b_) << std::endl;
std::cout << serialize(c_) << std::endl;
}

View File

@ -0,0 +1,25 @@
//
// Created by light on 20-1-6.
//
#include <iostream>
constexpr int factorial(int n) {
return n <= 1 ? 1 : (n * factorial(n - 1));
}
struct testStruct : std::true_type {
}; // Inherit from the true type.
int main() {
int i = factorial(5); // Call to a constexpr function.
// Will be replace by a good compiler by:
// int i = 120;
std::cout << i << std::endl;
constexpr bool testVar = testStruct(); // Generate a compile-time testStruct.
bool test = testStruct::value; // Equivalent to: test = true;
std::cout << test << std::endl;
test = testVar; // true_type has a constexpr converter operator, equivalent to: test = true;
std::cout << test << std::endl;
}

View File

@ -0,0 +1,20 @@
//
// Created by light on 20-1-6.
//
#include <iostream>
struct Default {
int foo() const {return 1;}
};
struct NonDefault {
NonDefault(const NonDefault&) {}
int foo() const {return 1;}
};
int main()
{
decltype(Default().foo()) n1 = 1; // int n1
// decltype(NonDefault().foo()) n2 = n1; // error: no default constructor
decltype(std::declval<NonDefault>().foo()) n2 = n1; // int n2
std::cout << "n2 = " << n2 << '\n';
}

View File

@ -0,0 +1,78 @@
//
// Created by light on 20-1-6.
//
#include <iostream>
#include "boost/hana.hpp"
#include "structData.h"
template<typename UnnamedType>
struct container {
// Let's put the test in private.
private:
// We use std::declval to 'recreate' an object of 'UnnamedType'.
// We use std::declval to also 'recreate' an object of type 'Param'.
// We can use both of these recreated objects to test the validity!
template<typename Param>
constexpr auto testValidity(int /* unused */)
-> decltype(std::declval<UnnamedType>()(std::declval<Param>()),boost::hana::true_c) {
// If substitution didn't fail, we can return a true_type.
return boost::hana::true_c;
}
template<typename Param>
constexpr std::false_type testValidity(...) {
// Our sink-hole returns a false_type.
return boost::hana::false_c;
}
public:
// A public operator() that accept the argument we wish to test onto the UnnamedType.
// Notice that the return type is automatic!
template<typename Param>
constexpr auto operator()(const Param &p) {
// The argument is forwarded to one of the two overloads.
// The SFINAE on the 'true_type' will come into play to dispatch.
// Once again, we use the int for the precedence.
return testValidity<Param>(int());
}
};
template<typename UnnamedType>
constexpr auto is_valid(const UnnamedType &t) {
// We used auto for the return type: it will be deduced here.
return container<UnnamedType>();
}
// Check if a type has a serialize method.
auto hasSerialize = is_valid([](auto &&x) -> decltype(x.serialize()) {});
// Notice how I simply swapped the return type on the right?
template<class T>
auto serialize(T &obj)
-> typename std::enable_if<decltype(hasSerialize(obj))::value, std::string>::type {
return obj.serialize();
}
template<class T>
auto serialize(T &obj)
-> typename std::enable_if<!decltype(hasSerialize(obj))::value, std::string>::type {
return to_string(obj);
}
int main() {
A a;
B b;
C c;
auto test = is_valid([](const auto &t) -> decltype(t.serialize()) {});
std::cout << test(a) << std::endl;
std::cout << test(b) << std::endl;
std::cout << test(c) << std::endl;
// The following lines work like a charm!
std::cout << serialize(a) << std::endl;
std::cout << serialize(b) << std::endl;
std::cout << serialize(c) << std::endl;
}

View File

@ -0,0 +1,28 @@
//
// Created by light on 20-1-6.
//
// https://www.boost.org/doc/libs/1_62_0/libs/hana/doc/html/structboost_1_1hana_1_1type.html#a2d2e7e08e284f7e0bd1bd9c3ad0e0a2b
#include <boost/hana.hpp>
#include <vector>
#include <string>
namespace hana = boost::hana;
int main() {
// 检查成员
struct Person {
std::string name;
};
auto has_name = hana::is_valid([](auto &&p) -> decltype(p.name) {});
Person joe{"Joe"};
static_assert(has_name(joe), "");
static_assert(!has_name(1), "");
// 检查嵌套类型
auto has_value_type = hana::is_valid([](auto t) -> hana::type<typename decltype(t)::type::value_type> {});
static_assert(has_value_type(hana::type_c<std::vector<int>>), "");
static_assert(!has_value_type(hana::type_c<Person>), "");
return 0;
}

View File

@ -0,0 +1,78 @@
//
// Created by light on 20-1-6.
//
#include <iostream>
#include "structData.h"
template<typename UnnamedType>
struct container {
// Let's put the test in private.
private:
// We use std::declval to 'recreate' an object of 'UnnamedType'.
// We use std::declval to also 'recreate' an object of type 'Param'.
// We can use both of these recreated objects to test the validity!
template<typename Param>
constexpr auto testValidity(int /* unused */)
-> decltype(std::declval<UnnamedType>()(std::declval<Param>()), std::true_type()) {
// If substitution didn't fail, we can return a true_type.
return std::true_type();
}
template<typename Param>
constexpr std::false_type testValidity(...) {
// Our sink-hole returns a false_type.
return std::false_type();
}
public:
// A public operator() that accept the argument we wish to test onto the UnnamedType.
// Notice that the return type is automatic!
template<typename Param>
constexpr auto operator()(const Param &p) {
// The argument is forwarded to one of the two overloads.
// The SFINAE on the 'true_type' will come into play to dispatch.
// Once again, we use the int for the precedence.
return testValidity<Param>(int());
}
};
template<typename UnnamedType>
constexpr auto is_valid(const UnnamedType &t) {
// We used auto for the return type: it will be deduced here.
return container<UnnamedType>();
}
// Check if a type has a serialize method.
auto hasSerialize = is_valid([](auto &&x) -> decltype(x.serialize()) {});
// Notice how I simply swapped the return type on the right?
template<class T>
auto serialize(T &obj)
-> typename std::enable_if<decltype(hasSerialize(obj))::value, std::string>::type {
return obj.serialize();
}
template<class T>
auto serialize(T &obj)
-> typename std::enable_if<!decltype(hasSerialize(obj))::value, std::string>::type {
return to_string(obj);
}
int main() {
A a;
B b;
C c;
auto test = is_valid([](const auto &t) -> decltype(t.serialize()) {});
std::cout << test(a) << std::endl;
std::cout << test(b) << std::endl;
std::cout << test(c) << std::endl;
// The following lines work like a charm!
std::cout << serialize(a) << std::endl;
std::cout << serialize(b) << std::endl;
std::cout << serialize(c) << std::endl;
}

View File

@ -0,0 +1,62 @@
//
// Created by light on 20-1-6.
//
#include <iostream>
#include "structData.h"
// Equivalent to:
struct l5UnamedType {
template<typename T>
auto operator()(T &t) const -> decltype(t.serialize()) // /!\ This signature is nice for a SFINAE!
{
return t.serialize();
}
};
void fun(A a,B b,C c) {
// ***** Simple lambda unamed type *****
auto l4 = [](int a, int b) { return a + b; };
std::cout << l4(4, 5) << std::endl; // Output 9.
// Equivalent to:
struct l4UnamedType {
int operator()(int a, int b) const {
return a + b;
}
};
l4UnamedType l4Equivalent = l4UnamedType();
std::cout << l4Equivalent(4, 5) << std::endl; // Output 9 too.
// ***** auto parameters lambda unnamed type *****
// b's type is automagically deduced!
auto l5 = [](auto &t) -> decltype(t.serialize()) { return t.serialize(); };
std::cout << l5(b) << std::endl; // Output: I am a B!
// std::cout << l5(a) << std::endl; // Error: no member named 'serialize' in 'A'.
l5UnamedType l5Equivalent = l5UnamedType();
std::cout << l5Equivalent(b) << std::endl; // Output: I am a B!
// std::cout << l5Equivalent(a) << std::endl; // Error: no member named 'serialize' in 'A'.
}
int main() {
A a;
B b;
C c;
auto l1 = [](B &b) { return b.serialize(); }; // Return type figured-out by the return statement.
auto l3 = [](B &b) -> std::string { return b.serialize(); }; // Fixed return type.
auto l2 = [](B &b) -> decltype(b.serialize()) { return b.serialize(); }; // Return type dependant to the B type.
std::cout << l1(b) << std::endl; // Output: I am a B!
std::cout << l2(b) << std::endl; // Output: I am a B!
std::cout << l3(b) << std::endl; // Output: I am a B!
fun(a,b,c);
}

View File

@ -0,0 +1,14 @@
//
// Created by light on 20-1-6.
//
#include <iostream>
void f(std::string s); // int can't be convert into a string.
void f(double d); // int can be implicitly convert into a double, so this version could be selected, but...
void f(int i); // ... this version using the type int directly is even more close!
int main() {
f(1); // Call f(int i);
}

View File

@ -0,0 +1,20 @@
//
// Created by light on 20-1-6.
//
#include <iostream>
std::string f(...) // Variadic functions are so "untyped" that...
{
return "...";
}
template<typename T>
std::string f(const T &t)// ...this templated function got the precedence!
{
return "T";
}
int main() {
std::cout << f<int>(1) << std::endl;
}

View File

@ -0,0 +1,23 @@
//
// Created by light on 20-1-6.
//
/*
The compiler will try this overload since it's less generic than the variadic.
T will be replace by int which gives us void f(const int& t, int::iterator* b = nullptr);
int doesn't have an iterator sub-type, but the compiler doesn't throw a bunch of errors.
It simply tries the next overload.
,()T将被int取代使
void f(const int& t, int::iterator* b = nullptr); int
*/
template<typename T>
void f(const T &t, typename T::iterator *it = nullptr) {}
// The sink-hole.
void f(...) {}
int main() {
f(1); // Calls void f(...) { }
}

View File

@ -0,0 +1,13 @@
//
// Created by light on 20-1-6.
//
// The compiler will be really unhappy when it will later discover the call to hahahaICrash.
// 当以后发现对hahahaICrash的调用时编译器将非常不满意。
template <typename T> void f(T t) { t.hahahaICrash(); }
void f(...) { } // The sink-hole wasn't even considered.
int main() {
f(1);
}

View File

@ -0,0 +1,77 @@
//
// Created by light on 20-1-6.
//
#include <iostream>
#include "structData.h"
template <typename UnnamedType> struct container
{
// Let's put the test in private.
private:
// We use std::declval to 'recreate' an object of 'UnnamedType'.
// We use std::declval to also 'recreate' an object of type 'Param'.
// We can use both of these recreated objects to test the validity!
template <typename... Params> constexpr auto test_validity(int /* unused */)
-> decltype(std::declval<UnnamedType>()(std::declval<Params>()...), std::true_type())
{
// If substitution didn't fail, we can return a true_type.
return std::true_type();
}
template <typename... Params> constexpr std::false_type test_validity(...)
{
// Our sink-hole returns a false_type.
return std::false_type();
}
public:
// A public operator() that accept the argument we wish to test onto the UnnamedType.
// Notice that the return type is automatic!
template <typename... Params> constexpr auto operator()(Params&& ...)
{
// The argument is forwarded to one of the two overloads.
// The SFINAE on the 'true_type' will come into play to dispatch.
return test_validity<Params...>(int());
}
};
template <typename UnnamedType> constexpr auto is_valid(UnnamedType&& t)
{
// We used auto for the return type: it will be deduced here.
return container<UnnamedType>();
}
// Check if a type has a serialize method.
auto hasSerialize = is_valid([](auto &&x) -> decltype(x.serialize()) {});
// Notice how I simply swapped the return type on the right?
template<class T>
auto serialize(T &obj)
-> typename std::enable_if<decltype(hasSerialize(obj))::value, std::string>::type {
return obj.serialize();
}
template<class T>
auto serialize(T &obj)
-> typename std::enable_if<!decltype(hasSerialize(obj))::value, std::string>::type {
return to_string(obj);
}
int main() {
A a;
B b;
C c;
auto test = is_valid([](const auto &t) -> decltype(t.serialize()) {});
std::cout << test(a,b) << std::endl;
std::cout << test(b) << std::endl;
std::cout << test(c) << std::endl;
// The following lines work like a charm!
std::cout << serialize(a) << std::endl;
std::cout << serialize(b) << std::endl;
std::cout << serialize(c) << std::endl;
}

View File

@ -0,0 +1,35 @@
//
// Created by light on 20-1-6.
//
#include <boost/hana.hpp>
#include <iostream>
#include <string>
#include "structData.h"
using namespace std;
namespace hana = boost::hana;
// 检查类型是否有一个serialize方法
auto hasSerialize = hana::is_valid([](auto &&x) -> decltype(x.serialize()) {});
// 序列化任意对象
template<typename T>
std::string serialize(T const &obj) {
return hana::if_(hasSerialize(obj),
[](auto &x) { return x.serialize(); },
[](auto &x) { return to_string(x); }
)(obj);
}
int main() {
A a;
B b;
C c;
std::cout << serialize(a) << std::endl;
std::cout << serialize(b) << std::endl;
std::cout << serialize(c) << std::endl;
}

View File

@ -0,0 +1,17 @@
//
// Created by light on 20-1-6.
//
#include <iostream>
typedef char type_test[42];
type_test &f() {}
int main() {
// In the following lines f won't even be truly called but we can still access to the size of its return type.
// Thanks to the "fake evaluation" of the sizeof operator.
char arrayTest[sizeof(f())];
std::cout << sizeof(f()) << std::endl; // Output 42.
}

View File

@ -0,0 +1,18 @@
#include <iostream>
//
// Created by light on 20-1-6.
//
typedef char yes; // Size: 1 byte.
typedef yes no[2]; // Size: 2 bytes.
// Two functions using our type with different size.
yes &f1() {}
no &f2() {}
int main() {
std::cout << sizeof(f1()) << std::endl;
std::cout << sizeof(f2()) << std::endl;
std::cout << (sizeof(f1()) == sizeof(f2())) << std::endl; // Output 0.
}

View File

@ -0,0 +1,30 @@
//
// Created by light on 20-1-6.
//
#ifndef MORDEN_C_STRUCTDATA_H
#define MORDEN_C_STRUCTDATA_H
// 类型A只有to_string 方法
struct A {
};
std::string to_string(const A &) {
return "I am A";
}
// 类型B有serialize方法
struct B {
std::string serialize() const {
return "I am B";
}
};
// 类型C有个serialize数据成员与to_string方法
struct C {
std::string serialize;
};
std::string to_string(const C &) {
return "I am C";
}
#endif //MORDEN_C_STRUCTDATA_H

View File

@ -0,0 +1,81 @@
//
// Created by light on 20-1-6.
//
#include <iostream>
#include "structData.h"
template<class T>
struct hasSerialize {
// 编译时比较
typedef char yes[1];
typedef char no[2];
// 允许我们检查序列化确实是一种方法
// 第二个参数必须是第一个参数的类型
// 例如: reallyHas<int,10> 替换为 reallyHas<int,int 10> 并起作用
// 注意:它仅适用于整数常量和指针(因此函数指针可以使用)
// 例如reallyHas<std::string (C::*)(), &C::serialize> 替换为
// reallyHas<std::string (C::*)(), std::string (C::*)() &C::serialize> 并起作用
template<typename U, U u>
struct reallyHas;
// std::string (C::*)() 是函数指针声明
template<typename C>
static yes &test(reallyHas<std::string (C::*)(), &C::serialize> * /*unused*/) {}
// ()()const 函数指针 -> std::string serialize() const
template<typename C>
static yes &test(reallyHas<std::string (C::*)() const, &C::serialize> * /*unused*/) {}
// The famous C++ sink-hole.
// Note that sink-hole must be templated too as we are testing test<T>(0).
// If the method serialize isn't available, we will end up in this method.
template<typename>
static no &test(...) { /* dark matter */ }
//用作测试的返回值的常数。
//由于编译时评估的大小,因此实际上在这里完成了测试。
static const bool value = sizeof(test<T>(0)) == sizeof(yes);
};
template<bool B, class T = void> // Default template version.
struct enable_if {
}; // This struct doesn't define "type" and the substitution will fail if you try to access it.
template<class T> // A specialisation used if the expression is true.
struct enable_if<true, T> {
typedef T type;
}; // This struct do have a "type" and won't fail on access.
template<class T>
typename enable_if<hasSerialize<T>::value, std::string>::type serialize(const T &obj) {
return obj.serialize();
}
template<class T>
typename enable_if<!hasSerialize<T>::value, std::string>::type serialize(const T &obj) {
return to_string(obj);
}
int main() {
// Usage:
// enable_if<true, int>::type t1; // Compiler happy. t's type is int.
// enable_if<hasSerialize<B>::value, int>::type t2; // Compiler happy. t's type is int.
//
// enable_if<false, int>::type t3; // Compiler unhappy. no type named 'type' in 'enable_if<false, int>';
// enable_if<hasSerialize<A>::value, int>::type t4; // no type named 'type' in 'enable_if<false, int>';
A a;
B b;
C c;
// The following lines work like a charm!
std::cout << serialize(a) << std::endl;
std::cout << serialize(b) << std::endl;
std::cout << serialize(c) << std::endl;
}