현재 새로운 C++ 0x 가변 템플릿을 사용하고 있습니다. 회원 인스턴스 생성 과정에 대한 질문이 있지만 상당히 재미 있습니다.가변 템플릿 정보
이 예제에서는 강력하고 강력한 열거 형을 선택할 가능성이있는 강력한 형식의 열거 형을 에뮬레이트하려고합니다 (단위 테스트 용으로 사용됨).
#include<vector>
#include<iostream>
using namespace std;
template<unsigned... values> struct sp_enum;
/*
this is the solution I found, declaring a globar var
vector<unsigned> _data;
and it work just fine
*/
template<> struct sp_enum<>{
static const unsigned _count = 0;
static vector<unsigned> _data;
};
vector<unsigned> sp_enum<>::_data;
template<unsigned T, unsigned... values>
struct sp_enum<T, values...> : private sp_enum<values...>{
static const unsigned _count = sp_enum<values...>::_count+1;
static vector<unsigned> _data;
sp_enum( ) : sp_enum<values...>(values...) {_data.push_back(T);}
sp_enum(unsigned v ) {_data.push_back(v);}
sp_enum(unsigned v, unsigned...) : sp_enum<values...>(values...) {_data.push_back(v);}
};
template<unsigned T, unsigned... values> vector<unsigned> sp_enum<T, values...>::_data;
int main(){
enum class t:unsigned{Default = 5, t1, t2};
sp_enum<t::Default, t::t1, t::t2> test;
cout <<test._count << endl << test._data.size() << endl;
for(auto i= test._data.rbegin();i != test._data.rend();++i){cout<< *i<< ":";}
}
이 코드로지고있어 결과는 다음과 같습니다
3 1 5:
사람이 내가 여기에 쌓여있는 무슨 날 지점 수 있습니다 ???
시 : 내가 더 일반적인 것으로 가능 (@Matthieu M.)와 같은 하드 코어 코딩을 줄이기 위해 코드를 재 작업 한 GCC 4.4.3
를 사용하여. 그러나 나는 왜 내가이 모든 일을하고 있는지 설명하고 싶다.
많은 개발자들이 내 코드에 새로운 C++ 0x 표준을 채택 했으므로 기쁘게 생각합니다. 하지만 테스트 단위를 작성하려고 할 때 강력한 형식의 열거 형에 문제가 있습니다.
문제는 임의의 강력한 형식의 열거 형을 생성 할 수 없다는 것입니다. (내가 할 수는 있지만 더 세련된 방식으로 수행하려고합니다.) 이제이 코드를 사용하여 가변성 템플릿과 가변 매크로 (예전의 더티 매크로)를 사용하여 강력한 형식의 범위가 지정된 enum을 선언하고 무작위로 선택할 수 있습니다. 여기
코드입니다 :의 PP_NARG 매크로의 한계가 지금 귀찮게 무엇
#include<vector>
#include<iostream>
#include <boost/preprocessor/array/elem.hpp>
#include <boost/preprocessor/repetition/repeat.hpp>
#include <boost/preprocessor/repetition/repeat_from_to.hpp>
using namespace std;
template<typename T, T... values> class sp_enum;
template<typename T> class sp_enum<T>{
protected: static const unsigned _count = 0;
};
template<typename T, T head, T... values>
class sp_enum<T, head, values...> : public sp_enum<T, values...>{
protected:
static const unsigned _count = sp_enum<T, values...>::_count+1;
static vector<T> _data;
public:
sp_enum( ) : sp_enum<T, values...>(values...) {_data.push_back(head);for(auto i= sp_enum<T, values...>::_data.begin();i != sp_enum<T, values...>::_data.end();++i){_data.push_back(*i);}}
sp_enum(T v ) {_data.push_back(v );}
sp_enum(T v, T...) : sp_enum<T, values...>(values...) {_data.push_back(v );for(auto i= sp_enum<T, values...>::_data.begin();i != sp_enum<T, values...>::_data.end();++i){_data.push_back(*i);}}
vector<T> data() const { return _data ;}
unsigned count() const { return _count ;}
static T randomEnum() { srand (time(NULL));return _data[rand()%_count];}
};
template<typename T, T head, T... values> vector<T> sp_enum<T, head, values...>::_data;
#define PP_NARG(...) PP_NARG_(__VA_ARGS__,PP_RSEQ_N())
#define PP_NARG_(...) PP_ARG_N(__VA_ARGS__)
#define PP_ARG_N(\
_1, _2, _3, _4, _5, _6, _7, _8, _9,_10, _11,_12,_13,_14,_15,_16,_17,_18,_19,_20, \
_21,_22,_23,_24,_25,_26,_27,_28,_29,_30, _31,_32,_33,_34,_35,_36,_37,_38,_39,_40, \
_41,_42,_43,_44,_45,_46,_47,_48,_49,_50, _51,_52,_53,_54,_55,_56,_57,_58,_59,_60, \
_61,_62,_63,N,...) N
#define PP_RSEQ_N() \
63,62,61,60,59,58,57,56,55,54,53,52,51,50,49,48,47,46,45,44,43,42,41,40, \
39,38,37,36,35,34,33,32,31,30,29,28,27,26,25,24,23,22,21,20,19,18,17,16, \
15,14,13,12,11,10,9,8,7,6,5,4,3,2,1,0
#define FOREACH_ARRAY(...) (PP_NARG(__VA_ARGS__) , (__VA_ARGS__))
#define FOREACH(Name, A, ...) BOOST_PP_REPEAT_FROM_TO(1, PP_NARG(Name, __VA_ARGS__), A, FOREACH_ARRAY(Name, __VA_ARGS__))
#define DECORATION(z,n,data) ,BOOST_PP_ARRAY_ELEM(0, data) :: BOOST_PP_ARRAY_ELEM(n, data)
#define SP_ENUM(enumName, ...) \
enum class _##enumName : unsigned { Default, __VA_ARGS__ }; \
typedef sp_enum<_##enumName FOREACH(_##enumName, DECORATION, Default, __VA_ARGS__) > enumName;
SP_ENUM(xx, test1, test2, test3)
int main(){
xx test;
cout <<test.count() << endl << test.data().size() << endl;
auto dt = test.data();
for(auto i = dt.rbegin(); i != dt.rend();++i){ cout<< (unsigned)*i << ":" ; }
cout << "random strongly typed enum : " << (unsigned) test.randomEnum() << endl;
}
(I는 인수의 수를 계산하는 다른 방법을 발견하지 않았습니다).
나는 이것을 향상시키기 위해 기꺼이 모든 포인터 또는 힌트를 수락 할 것입니다.
난 당신이 ....'