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