typedef enum colors {
MIN_COLOR,
RED = MIN_COLOR,
GREEN,
MAX_COLOR
} colors;
template< colors C >
struct MapEntry {
std::string s;
MapEntry(std::string s_):s(s_) {}
};
void do_in_order() {}
template<typename F0, typename... Fs>
void do_in_order(F0&& f0, Fs&&... fs) {
std::forward<F0>(f0)();
do_in_order(std::forward<Fs>(fs)...);
}
struct MapInit {
std::map< std::string, color > retval;
operator std::map< std::string, color >() {
return std::move(retval);
}
template<colors C>
void AddToMap(MapEntry<C>&& ent) {
retval.insert(std::make_pair(std::move(end.s), C));
}
template< typename... Entries >
MapInit(Entries&& entries) {
do_in_order([&](){ AddToMap(entries); }...);
}
};
template<typename... Entries>
MapInit mapInit(Entries&&... entries) {
return MapInit(std::forward<Entries>(entries)...);
}
const std::map<std::string, colors> = mapInit(MapEntry<RED>("red"), MapEntry<GREEN>("green"));
जो आपको एक सी ++ 11 रास्ता संकलन समय color
से एक std::map
का निर्माण और रन-टाइम string
डेटा देता है।
की सूची colors
की सूची में "मेटाफंक्शन" की सूची में फेंको।
template<colors... Cs>
struct color_list {};
template<typename... Ts>
struct type_list {};
template<typename MapEnt>
struct extract_color;
template<colors C>
struct extract_color<MapEntry<C>> {
enum {value=C};
};
template<typename Entries>
struct extract_colors;
template<typename... MapEntries>
struct extract_colors<type_list<MapEntries...>> {
typedef color_list< ((colors)extract_colors<MapEntries>::value)... > type;
};
उस सूची को सॉर्ट करें। डुप्लिकेट का पता लगाएं - यदि वहां हैं, तो आप खराब हो गए हैं।
संकलन-समय सॉर्टिंग बाकी के मुकाबले कठिन है, और कोड की 100+ रेखाएं हैं। यदि आप बहुत ज्यादा ध्यान नहीं देते हैं तो मैं इसे छोड़ दूंगा! Here is a compile time merge sort I wrote in the past to answer a stack overflow question जो अपेक्षाकृत सरल अनुकूलन के साथ काम करेगा (यह मूल्यों के साथ प्रकारों को प्रकार देता है, इस मामले में हम सीधे संकलन-समय मानों की एक सूची क्रमबद्ध कर रहे हैं)।
// takes a sorted list of type L<T...>, returns true if there are adjacent equal
// elements:
template<typename clist, typename=void>
struct any_duplicates:std::false_type {};
template<typename T, template<T...>class L, T t0, T t1, T... ts>
struct any_duplicates< L<t0, t1, ts...>, typename std::enable_if<t0==t1>::type>:
std::true_type {};
template<typename T, template<T...>class L, T t0, T t1, T... ts>
struct any_duplicates< L<t0, t1, ts...>, typename std::enable_if<t0!=t1>::type>:
any_duplicates< L<t1, ts...> > {};
colors
(यानी, <MIN_COLOR
या >=MAX_COLOR
) की मान्य श्रेणी से बाहर तत्वों का पता लगाने। यदि हां, तो आप खराब हो गए।
template<typename List>
struct min_max;
template<typename T, template<T...>class L, T t0>
struct min_max {
enum {
min = t0,
max = t1,
};
};
template<typename T, template<T...>class L, T t0, T t1, T... ts>
struct min_max {
typedef min_max<L<t1, ts...>> rest_of_list;
enum {
rest_min = rest_of_list::min,
rest_max = rest_of_list::max,
min = (rest_min < t0):rest_min:t0,
max = (rest_max > t0):rest_max:t0,
};
};
template< typename T, T min, T max, typename List >
struct bounded: std::integral_constant< bool,
(min_max<List>::min >= min) && (min_max<List>::max < max)
> {};
गणना देखते हैं कि कितने तत्वों - वहाँ MAX_COLOR
तत्वों होना चाहिए। यदि नहीं, तो आप खराब हो गए।
template<typename List>
struct element_count;
template<typename T, template<T...>L, T... ts>
struct element_count<L<ts...>>:std::integral_constant< std::size_t, sizeof...(ts) > {};
यदि इनमें से कोई भी नहीं हुआ, तो कबूतर द्वारा आपको उनमें से प्रत्येक को प्रारंभ करना होगा।
एकमात्र चीज गायब है कि आप चले गए थे और दो मानों के लिए उसी string
का उपयोग कर सकते थे।संकलन समय string
एस दर्द के रूप में, बस इसे रन टाइम पर जांचें (map
में प्रविष्टियों की संख्या colors
की संख्या को प्रारंभ करने के बाद बराबर होती है)।
सी ++ 03 में ऐसा करना कठिन होगा। आपको विविध टेम्पलेट्स की कमी है, इसलिए आप उन्हें नकली बनाते हैं। दर्द क्या है। mpl
आपकी मदद करने में सक्षम हो सकता है।
वैरार्डिक टेम्पलेट्स नवंबर 2012 एमएसवीसी सीटीपी कंपाइलर अपडेट में उपलब्ध हैं।
डुप्लिकेट जांच के बिना और बाउंड जांच के बिना एक खिलौना उदाहरण है (यह सिर्फ जांचता है कि मानचित्र प्रविष्टियों के मिलान की संख्या);
#include <cstddef>
#include <utility>
#include <string>
#include <map>
enum TestEnum {
BeginVal = 0,
One = BeginVal,
Two,
Three,
EndVal
};
template<TestEnum e>
struct MapEntry {
enum { val = e };
std::string s;
MapEntry(std::string s_):s(s_) {}
};
void do_in_order() {}
template<typename F0, typename... Fs>
void do_in_order(F0&& f0, Fs&&... fs) {
std::forward<F0>(f0)();
do_in_order(std::forward<Fs>(fs)...);
}
template<typename... MapEntries>
struct count_entries:std::integral_constant< std::size_t, sizeof...(MapEntries) > {};
// should also detect duplicates and check the range of the values:
template<typename... MapEntries>
struct caught_them_all:
std::integral_constant<
bool,
count_entries<MapEntries...>::value == (TestEnum::EndVal-TestEnum::BeginVal)
>
{};
struct BuildMap {
typedef std::map<std::string, TestEnum> result_map;
mutable result_map val;
operator result_map() const {
return std::move(val);
}
template<typename... MapEntries>
BuildMap(MapEntries&&... entries) {
static_assert(caught_them_all<MapEntries...>::value, "Missing enum value");
bool _[] = { ((val[ entries.s ] = TestEnum(MapEntries::val)), false)... };
}
};
std::map< std::string, TestEnum > bob = BuildMap(
MapEntry<One>("One")
,MapEntry<Two>("Two")
#if 0
,MapEntry<Three>("Three")
#endif
);
int main() {}
#if 1
साथ #if 0
बदलें यह संकलन को देखने के लिए। Live link अगर आप खेलना चाहते हैं।
आप अपने कोड शिपिंग से पहले, सही परीक्षण करते हैं? तो यह सुनिश्चित करने के लिए एक रनटाइम चेक पर्याप्त होगा, क्या आपको नहीं लगता? –
एकमात्र भाषा निर्माण जो मुझे पता है, आपको चेतावनी देगा यदि आप एक मूल्य मान चूक गए हैं तो 'स्विच' है यदि आप प्रत्येक मान के लिए 'केस' जोड़ते हैं और आपके पास' डिफ़ॉल्ट 'शाखा नहीं है। मुझे लगता है कि इसका मतलब है कि आपको एक मैक्रो की आवश्यकता होगी लेकिन मैं कुछ विशिष्ट सुझाव देने से बहुत दूर हूं :) –
मैं मानता हूं कि रन-टाइम परीक्षण पर्याप्त होना चाहिए, और मुझे चेतावनी के बारे में भी पता है। मुझे लगता है कि मैं यह समझने की कोशिश कर रहा हूं कि इसे – FKaria