プリプロセッサを使用してコードを生成したい場合は、Boost.Preprocessorに頼ることができます。たとえば、次の方法で何かを行うことができます。
#define GET_TYPE( pair ) BOOST_PP_TUPLE_ELEM( 2, 0, pair )
#define GET_NAME( pair ) BOOST_PP_TUPLE_ELEM( 2, 1, pair )
#define DECLARE_MANDATORY( r, data, elem ) GET_TYPE( elem ) GET_NAME( elem );
#define DECLARE_MANDATORY_ATTRIBUTES( attributes ) \
BOOST_PP_SEQ_FOR_EACH( DECLARE_MANDATORY, ~, attributes )
#define DECLARE_OPTIONAL( r, data, elem ) boost::optional< GET_TYPE( elem ) > GET_NAME( elem );
#define DECLARE_OPTIONAL_ATTRIBUTES( attributes ) \
BOOST_PP_SEQ_FOR_EACH( DECLARE_OPTIONAL, ~, attributes )
#define MANDATORY_ACCESSORS( r, data, elem ) \
BOOST_PP_CAT( void set_, GET_NAME( elem ) ) (GET_TYPE( elem ) const & value) { GET_NAME( elem ) = value; } \
GET_TYPE( elem ) BOOST_PP_CAT( get_, GET_NAME( elem ) ) () const { return GET_NAME( elem ); }
#define DEFINE_MANDATORY_ACCESSORS( attributes ) \
BOOST_PP_SEQ_FOR_EACH( MANDATORY_ACCESSORS, ~, attributes )
#define OPTIONAL_ACCESSORS( r, data, elem ) \
BOOST_PP_CAT( void set_, GET_NAME( elem ) ) (GET_TYPE( elem ) const & value) { GET_NAME( elem ).reset(value); } \
GET_TYPE( elem ) BOOST_PP_CAT( get_, GET_NAME( elem ) ) () const { return *GET_NAME( elem ); }
#define DEFINE_OPTIONAL_ACCESSORS( attributes ) \
BOOST_PP_SEQ_FOR_EACH( OPTIONAL_ACCESSORS, ~, attributes )
class Book
{
#define BOOK_MANDATORY_ATTRIBUTES ((std::string, title))
#define BOOK_OPTIONAL_ATTRIBUTES ((std::string, author)) ((std::string, secondaryAuthor))
public:
DECLARE_MANDATORY_ATTRIBUTES( BOOK_MANDATORY_ATTRIBUTES )
DECLARE_OPTIONAL_ATTRIBUTES( BOOK_OPTIONAL_ATTRIBUTES )
DEFINE_MANDATORY_ACCESSORS( BOOK_MANDATORY_ATTRIBUTES )
DEFINE_OPTIONAL_ACCESSORS( BOOK_OPTIONAL_ATTRIBUTES )
};
int main()
{
Book b;
b.set_title("H2G2");
std::cout << b.get_title();
b.set_author("Douglas Adams");
std::cout << b.get_author();
}
別のマクロを作成して、[ boost::optional
](http://www.boost.org/libs/optional] の使用を利用してシリアル化関数のコードを生成し、オプションの属性が設定されているかどうかをテストできます。
コード生成を行いたくない場合、属性のアクセサーを自動的に取得することはできません。ただし、シリアル化の側面は、たとえばすべての属性に共通の基本クラスを使用するなど、より古典的に処理できます。このようにして、それらをコンテナーに格納し、それらを反復処理して、それら自体をシリアル化するように依頼できます。
struct Attribute
{
virtual QString serializePlainText() = 0;
};
// Generic attribute for the most common cases
template <typename T>
class MandatoryAttribute : public Attribute
{
T value;
public:
std::string serializePlainText()
{
return boost::lexical_cast<std::string>(value);
}
};
// Generic attribute for the most common cases
template <typename T>
class OptionalAttribute : public Attribute
{
boost::optional<T> value;
public:
bool isSet() const { return value; }
std::string serializePlainText()
{
return isSet() ?
boost::lexical_cast<std::string>(value) :
"";
}
};