Files
contract/example/push_back.cpp

67 lines
2.0 KiB
C++

#include <boost/contract.hpp>
#include <vector>
template<typename T>
struct pusahble {
void push_back(T const& value, boost::contract::virtual_body v = 0) {
auto contract = boost::contract::public_member(v, this)
.precondition([&] {
BOOST_CONTRACT_ASSERT(false); // Force check derived precond.
})
.postcondition[&] {
BOOST_CONTRACT_ASSERT(
boost::contract::call_if<boost::has_equal_to<T> >(
boost::bind(std::equal_to<T>(),
boost::cref(this->back()), boost::cref(value))
).else_(boost::contract::always(true))
);
})
;
}
virtual void push_back(T const& value) = 0; // Subcontract pure-virtual.
protected:
virtual T const& back() const = 0;
};
template<typename T>
class vector
#define BASES public ushable<T>
: BASES
{
public:
typedef BOOST_CONTRACT_BASE_TYPES(BASES) base_types;
#undef BASES
void invariant() const {
BOOST_CONTRACT_ASSERT(empty() == (size() == 0));
}
void push_back(T const& value, boost::contract::virtual_body v = 0) {
auto old_size = BOOST_CONTRACT_OLDOF(v, size());
boost::contract::type contract = boost::contract::public_member<
introspect_push_back>(v, this, &vector::push_back, value)
.precondition([&]() {
BOOST_CONTRACT_ASSERT(this->size() < this->max_size());
})
.postcondition([&]() {
BOOST_CONTRACT_ASSERT(this->size() == old_size + 1);
})
;
push_back_body(value);
}
virtual void push_back_body(T const& value) { vector_.push_back(value); }
bool empty() const { return vector_.empty(); }
unsigned size() const { return vector_.size(); }
unsigned max_size() const { return vector_.max_size(); }
T const& back() const { return vector_.back(); }
private:
std::vector<T> vector_;
BOOST_CONTRACT_INTROSPECT(push_back)
};