#include <iostream>
#include <limits>
#include <type_traits>
using namespace std;
struct vec
{
double x;
double y;
double z;
};
namespace details
{
template <typename T>
constexpr double X(const T& param) { return param.x; }
template <typename T>
constexpr double Y(const T& param) { return param.y; }
template <typename T>
constexpr double Z(const T& param) { return param.z; }
}
template <typename T, typename = void>
constexpr double (*my_temp[])(const vec&) = { &details::X<T>, &details::Y<T> };
template <typename T>
constexpr double (*my_temp<T, enable_if_t<is_floating_point<decltype(details::X(T()))>::value>>[])(const vec&) = { &details::X<T>, &details::Y<T>, &details::Z<T> };
int main() {
vec foo = { 1.0, 2.0, 3.0 };
for(const auto i : my_temp<decltype(foo)>) {
cout << (*i)(foo) << endl;
}
}
I2luY2x1ZGUgPGlvc3RyZWFtPgojaW5jbHVkZSA8bGltaXRzPgojaW5jbHVkZSA8dHlwZV90cmFpdHM+Cgp1c2luZyBuYW1lc3BhY2Ugc3RkOwoKc3RydWN0IHZlYwp7CiAgZG91YmxlIHg7CiAgZG91YmxlIHk7CiAgZG91YmxlIHo7Cn07CgpuYW1lc3BhY2UgZGV0YWlscwp7CnRlbXBsYXRlIDx0eXBlbmFtZSBUPgpjb25zdGV4cHIgZG91YmxlIFgoY29uc3QgVCYgcGFyYW0pIHsgcmV0dXJuIHBhcmFtLng7IH0KCnRlbXBsYXRlIDx0eXBlbmFtZSBUPgpjb25zdGV4cHIgZG91YmxlIFkoY29uc3QgVCYgcGFyYW0pIHsgcmV0dXJuIHBhcmFtLnk7IH0KCnRlbXBsYXRlIDx0eXBlbmFtZSBUPgpjb25zdGV4cHIgZG91YmxlIFooY29uc3QgVCYgcGFyYW0pIHsgcmV0dXJuIHBhcmFtLno7IH0KfQoKdGVtcGxhdGUgPHR5cGVuYW1lIFQsIHR5cGVuYW1lID0gdm9pZD4KY29uc3RleHByIGRvdWJsZSAoKm15X3RlbXBbXSkoY29uc3QgdmVjJikgPSB7ICZkZXRhaWxzOjpYPFQ+LCAmZGV0YWlsczo6WTxUPiB9OwoKdGVtcGxhdGUgPHR5cGVuYW1lIFQ+CmNvbnN0ZXhwciBkb3VibGUgKCpteV90ZW1wPFQsIGVuYWJsZV9pZl90PGlzX2Zsb2F0aW5nX3BvaW50PGRlY2x0eXBlKGRldGFpbHM6OlgoVCgpKSk+Ojp2YWx1ZT4+W10pKGNvbnN0IHZlYyYpID0geyAmZGV0YWlsczo6WDxUPiwgJmRldGFpbHM6Olk8VD4sICZkZXRhaWxzOjpaPFQ+IH07CgoKaW50IG1haW4oKSB7CiAgdmVjIGZvbyA9IHsgMS4wLCAyLjAsIDMuMCB9OwoKICBmb3IoY29uc3QgYXV0byBpIDogbXlfdGVtcDxkZWNsdHlwZShmb28pPikgewogICAgY291dCA8PCAoKmkpKGZvbykgPDwgZW5kbDsKICB9Cn0=