File: pure_virtual_public.cpp

package info (click to toggle)
boost1.83 1.83.0-5
  • links: PTS, VCS
  • area: main
  • in suites: forky, sid
  • size: 545,632 kB
  • sloc: cpp: 3,857,086; xml: 125,552; ansic: 34,414; python: 25,887; asm: 5,276; sh: 4,799; ada: 1,681; makefile: 1,629; perl: 1,212; pascal: 1,139; sql: 810; yacc: 478; ruby: 102; lisp: 24; csh: 6
file content (89 lines) | stat: -rw-r--r-- 2,724 bytes parent folder | download | duplicates (12)
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
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89

// Copyright (C) 2008-2018 Lorenzo Caminiti
// Distributed under the Boost Software License, Version 1.0 (see accompanying
// file LICENSE_1_0.txt or a copy at http://www.boost.org/LICENSE_1_0.txt).
// See: http://www.boost.org/doc/libs/release/libs/contract/doc/html/index.html

#include <boost/contract.hpp>
#include <vector>
#include <cassert>

//[pure_virtual_public_base_begin
template<typename Iterator>
class range {
public:
    // Pure virtual function declaration (contract in definition below).
    virtual Iterator begin(boost::contract::virtual_* v = 0) = 0;
//]

    // Could program class invariants and contracts for the following too.
    virtual Iterator end() = 0;
    virtual bool empty() const = 0;

//[pure_virtual_public_base_end
    /* ... */
};
//]

//[pure_virtual_public_base_impl
// Pure virtual function default implementation (out-of-line in C++).
template<typename Iterator>
Iterator range<Iterator>::begin(boost::contract::virtual_* v) {
    Iterator result; // As usual, virtual pass `result` right after `v`...
    boost::contract::check c = boost::contract::public_function(v, result, this)
        .postcondition([&] (Iterator const& result) {
            if(empty()) BOOST_CONTRACT_ASSERT(result == end());
        })
    ;

    // Pure function body (never executed by this library).
    assert(false);
    return result;
}
//]

template<typename T>
class vector
    #define BASES public range<typename std::vector<T>::iterator>
    : BASES
{
public:
    typedef BOOST_CONTRACT_BASE_TYPES(BASES) base_types;
    #undef BASES
    typedef typename std::vector<T>::iterator iterator;

    iterator begin(boost::contract::virtual_* v = 0) /* override */ {
        iterator result;
        // Again, pass result right after `v`...
        boost::contract::check c = boost::contract::public_function<
                override_begin>(v, result, &vector::begin, this)
            // ...plus postconditions take `result` as parameter (not capture).
            .postcondition([&] (iterator const& result) {
                if(!empty()) BOOST_CONTRACT_ASSERT(*result == front());
            })
        ;

        return result = vect_.begin();
    }
    BOOST_CONTRACT_OVERRIDE(begin)

    // Could program class invariants and contracts for the following too.
    iterator end() { return vect_.end(); }
    bool empty() const { return vect_.empty(); }
    T const& front() const { return vect_.front(); }
    void push_back(T const& value) { vect_.push_back(value); }

private:
    std::vector<T> vect_;
};

int main() {
    vector<int> v;
    v.push_back(1);
    v.push_back(2);
    v.push_back(3);
    range<std::vector<int>::iterator>& r = v;
    assert(*(r.begin()) == 1);
    return 0;
}