#include <cassert>
#include <vector>
#include <cstddef>


namespace preordering {

template<typename T>
class ArcProperties
{
public:
    typedef T VALUE_TYPE;

    ArcProperties(size_t n, T default_value = T()) : 
        properties_(n, std::vector<T>(n, default_value))
    {}

    ArcProperties(const std::vector<std::vector<T>>& properties) :
        properties_(properties)
    {
        for (auto row : properties)
            assert (row.size() == properties.size() );
    }

    const T& operator()(size_t i, size_t j) const
    {
        return properties_[i][j];
    }

    T& operator()(size_t i, size_t j) 
    {
        return properties_[i][j];
    }

    size_t size() const
    {
        return properties_.size();
    }

    template<class OTHER>
    bool operator==(const OTHER& other) const
    {
        if (this->size() != other.size())
            return false;
        for (size_t i = 0; i < this->size(); ++i)
        for (size_t j = 0; j < this->size(); ++j)
        {
            if (properties_[i][j] != other(i, j))
                return false;
        }
        return true;
    }

    template<class OTHER>
    bool operator!=(const OTHER& other) const
    {
        return !this->operator==(other);
    }

    template<class OSTREAM>
    void write(OSTREAM& stream)
    {
        for (size_t i = 0; i < this->size(); ++i)
        for (size_t j = 0; j < this->size(); ++j)
            stream << this->operator()(i, j) << (j < this->size() - 1 ? " " : "\n");
    }
    
    std::vector<std::vector<T>> data() const
    {
        return properties_;
    }

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


template<typename T>
class ArcPropertiesCons
{
public:
    typedef T VALUE_TYPE;

    ArcPropertiesCons(size_t n, T default_value = T()) : 
        n_(n),
        properties_(n*n, default_value)
    {}

    ArcPropertiesCons(const std::vector<std::vector<T>>& properties) :
        ArcPropertiesCons(properties.size())
    {
        for (size_t i = 0; i < this->size(); ++i)
        {
            assert (properties[i].size() == n_);
            for (size_t j = 0; j < n_; ++j)
            {
                this->operator()(i, j) = properties[i][j];
            }
        }
    }

    const T& operator()(size_t i, size_t j) const
    {
        assert (i < n_ && j < n_);
        return properties_[i*n_+j];
    }

    T& operator()(size_t i, size_t j) 
    {
        assert (i < n_ && j < n_);
        return properties_[i*n_+j];
    }

    size_t size() const
    {
        return n_;
    }

    template<class OTHER>
    bool operator==(const OTHER& other) const
    {
        if (this->size() != other.size())
            return false;
        for (size_t i = 0; i < this->size(); ++i)
        for (size_t j = 0; j < this->size(); ++j)
        {
            if (this->operator()(i, j) != other(i, j))
                return false;
        }
        return true;
    }

    template<class OTHER>
    bool operator!=(const OTHER& other) const
    {
        return !this->operator==(other);
    }

    template<class OSTREAM>
    void write(OSTREAM& stream)
    {
        for (size_t i = 0; i < this->size(); ++i)
        for (size_t j = 0; j < this->size(); ++j)
            stream << this->operator()(i, j) << (j < this->size() - 1 ? " " : "\n");
    }

    std::vector<std::vector<T>> data() const
    {
        std::vector<std::vector<T>> output(n_, std::vector<T>(n_));
        for (size_t i = 0; i < n_; ++i)
        for (size_t j = 0; j < n_; ++j)
            output[i][j] = this->operator()(i, j);
        return output;
    }
    

private:
    size_t n_;
    std::vector<T> properties_;
};

}  // namespace preordering