boost/libs/serialization/test/test_non_intrusive.cpp
2021-10-05 21:37:46 +02:00

175 lines
4.6 KiB
C++

/////////1/////////2/////////3/////////4/////////5/////////6/////////7/////////8
// test_non_intrursive.cpp
// (C) Copyright 2002 Robert Ramey - http://www.rrsd.com .
// Use, modification and distribution is subject to the Boost Software
// License, Version 1.0. (See accompanying file LICENSE_1_0.txt or copy at
// http://www.boost.org/LICENSE_1_0.txt)
// should pass compilation and execution
// this tests:
// a) non-intrusive method of implementing serialization
// b) usage of a non-default constructor
#include <fstream>
#include <cstdlib> // for rand()
#include <cstdio> // remove
#include <boost/config.hpp>
#include <boost/detail/workaround.hpp>
#include <boost/limits.hpp>
#include <boost/math/special_functions/next.hpp>
#if defined(BOOST_NO_STDC_NAMESPACE)
namespace std{
using ::rand;
using ::remove;
#if BOOST_WORKAROUND(BOOST_MSVC, >= 1400) && !defined(UNDER_CE)
using ::numeric_limits;
#endif
}
#endif
#include <boost/archive/archive_exception.hpp>
#include "test_tools.hpp"
///////////////////////////////////////////////////////
// simple class test - using non-intrusive syntax
// illustrates the usage of the non-intrusve syntax
class A
{
public:
signed char s;
unsigned char t;
signed int u;
unsigned int v;
float w;
double x;
A();
bool operator==(const A & rhs) const;
bool operator<(const A & rhs) const;
};
A::A() :
s(static_cast<signed char>(0xff & std::rand())),
t(static_cast<signed char>(0xff & std::rand())),
u(std::rand()),
v(std::rand()),
w((float)std::rand() / std::rand()),
x((double)std::rand() / std::rand())
{
}
bool A::operator==(const A &rhs) const
{
return
s == rhs.s
&& t == rhs.t
&& u == rhs.u
&& v == rhs.v
&& std::abs( boost::math::float_distance(w, rhs.w)) < 2
&& std::abs( boost::math::float_distance(x, rhs.x)) < 2
;
}
bool A::operator<(const A &rhs) const
{
if(! (s == rhs.s) )
return s < rhs.s;
if(! (t == rhs.t) )
return t < rhs.t;
if(! (u == rhs.u) )
return t < rhs.u;
if(! (v == rhs.v) )
return t < rhs.v;
if(std::abs( boost::math::float_distance(w, rhs.w)) > 1)
return false;
if(std::abs( boost::math::float_distance(x, rhs.x)) > 1)
return false;
return false;
}
// note the following:
// function specializations must be defined in the appropriate
// namespace - boost::serialization
namespace boost {
namespace serialization {
// This first set of overrides should work with all compilers.
// The last argument is int while the default versions
// defined in serialization.hpp have long as the last argument.
// This is part of the work around for compilers that don't
// support correct function template ordering. These functions
// are always called with 0 (i.e. an int) as the last argument.
// Our specialized versions also have int as the last argument
// while the default versions have a long as the last argument.
// This makes our specialized versions a better match than the
// default ones as no argument conversion is required to make a match
template<class Archive>
void serialize(
Archive & ar,
A & a,
const unsigned int /* file_version */
){
ar & boost::serialization::make_nvp("s", a.s);
ar & boost::serialization::make_nvp("t", a.t);
ar & boost::serialization::make_nvp("u", a.u);
ar & boost::serialization::make_nvp("v", a.v);
ar & boost::serialization::make_nvp("w", a.w);
ar & boost::serialization::make_nvp("x", a.x);
}
} // serialization
} // namespace boost
void save(const char * testfile){
test_ostream os(testfile, TEST_STREAM_FLAGS);
test_oarchive oa(os, TEST_ARCHIVE_FLAGS);
A a;
oa << BOOST_SERIALIZATION_NVP(a);
// save a copy pointer to this item
A *pa1 = &a;
oa << BOOST_SERIALIZATION_NVP(pa1);
// save pointer to a new object
A *pa2 = new A();
oa << BOOST_SERIALIZATION_NVP(pa2);
delete pa2;
}
void load(const char * testfile){
test_istream is(testfile, TEST_STREAM_FLAGS);
test_iarchive ia(is, TEST_ARCHIVE_FLAGS);
A a;
ia >> BOOST_SERIALIZATION_NVP(a);
A *pa1;
ia >> BOOST_SERIALIZATION_NVP(pa1);
BOOST_CHECK_MESSAGE(pa1 == &a, "Copy of pointer not correctly restored");
A *pa2;
ia >> BOOST_SERIALIZATION_NVP(pa2);
BOOST_CHECK_MESSAGE(pa2 != &a, "Pointer not correctly restored");
delete pa2;
}
int
test_main( int /* argc */, char* /* argv */[] )
{
const char * testfile = boost::archive::tmpnam(NULL);
BOOST_REQUIRE(NULL != testfile);
save(testfile);
load(testfile);
std::remove(testfile);
return EXIT_SUCCESS;
}
// EOF