2011-01-20 4 views
3

Я планирую создать собственный архив, например boost :: archive :: xml_oachive, и я нашел хорошие примеры в папке boost/libs/serialization/example.Как создать архив, являющийся указателем разбора?

Смотреть следующий код (есть в каталоге выше):

// simple_log_archive.hpp 
... 
class simple_log_archive 
{ 
    ... 
    template <class Archive> 
    struct save_primitive 
    { 
     template <class T> 
     static void invoke(Archive& ar, const T& t) 
     { 
      // streaming 
     } 
    }; 

    template <class Archive> 
    struct save_only 
    { 
     template <class T> 
     static void invoke(Archive& ar, const T& t) 
     { 
      boost::serialization::serialize_adl(ar, const_cast<T&>(t), 
       ::boost::serialization::version<T>::value); 
     } 
    }; 

    template <class T> 
    void save(const T& t) 
    { 
     typedef BOOST_DEDUCED_TYPENAME boost::mpl::eval_if<boost::is_enum<T>, 
      boost::mpl::identity<save_enum_type<simple_log_archive> >, 
     //else 
     BOOST_DEDUCED_TYPENAME boost::mpl::eval_if< 
      // if its primitive 
       boost::mpl::equal_to< 
        boost::serialization::implementation_level<T>, 
        boost::mpl::int_<boost::serialization::primitive_type> 
       >, 
       boost::mpl::identity<save_primitive<simple_log_archive> >, 
     // else 
      boost::mpl::identity<save_only<simple_log_archive> > 
     > >::type typex; 
     typex::invoke(*this, t); 
    } 
public: 
    // the << operators 
    template<class T> 
    simple_log_archive & operator<<(T const & t){ 
     m_os << ' '; 
     save(t); 
     return * this; 
    } 
    template<class T> 
    simple_log_archive & operator<<(T * const t){ 
     m_os << " ->"; 
     if(NULL == t) 
      m_os << " null"; 
     else 
      *this << * t; 
     return * this; 
    } 
    ... 
}; 

Кроме того, я был сделать свой собственный архив. Но мой и выше код не является автоматическим литьем базового указателя на производный указатель. Например,

Base* base = new Derived; 
{ 
    boost::archive::text_oarchive ar(std::cout); 
    ar << base;// Base pointer is auto casted to derived pointer! It's fine. 
} 

{ 
    simple_log_archive ar; 
    ar << base;// Base pointer is not auto casting. This is my problem. 
} 

Не могли бы вы мне помочь? Как получить базовый указатель на производный указатель?

+0

У меня было найдено решение. simple_log_archive.hpp - всего лишь образец для основного потока mpl. Если вы хотите создать автоматический литой пользовательский архив, во-первых, наследование detail :: common_oarchive или detail :: common_iarchive и реализовать. Ссылка basic_binary/text/xml_archive. detail :: common_archive уже реализовал процесс литья указателей. Во-вторых, определите BOOST_CLASS_EXPORT (Производный). Этот макрос make guid, этот guid используется в деталях :: common_archive для поиска «истинного типа». –

+0

Наконец-то я до сих пор не знаю, почему text_oarchive возможен без register_type(), но вы должны зарегистрировать тип для архива. (Ar.register_type ()) –

ответ

0

Если вам нужно всего лишь преобразовать указатель базового класса в производный класс, то это должно сделать это: dynamic_cast<Derived*>(base).

+0

Усиление сериализации имеет карту типов одноэлементного типа. Таким образом, это не необходимость ручного литья. –

Смежные вопросы