/* -*- mode: C++; c-basic-offset: 2; indent-tabs-mode: nil -*- */ /* * Main authors: * Christian Schulte * * Copyright: * Christian Schulte, 2003 * * This file is part of Gecode, the generic constraint * development environment: * http://www.gecode.org * * Permission is hereby granted, free of charge, to any person obtaining * a copy of this software and associated documentation files (the * "Software"), to deal in the Software without restriction, including * without limitation the rights to use, copy, modify, merge, publish, * distribute, sublicense, and/or sell copies of the Software, and to * permit persons to whom the Software is furnished to do so, subject to * the following conditions: * * The above copyright notice and this permission notice shall be * included in all copies or substantial portions of the Software. * * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND * NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE * LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION * OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION * WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. * */ #include namespace Gecode { /* * Integer sets * */ forceinline IntSet::IntSet(void) {} /** * \brief Integer set initialization * * Helper class to allow partial specialization of * template member function. * */ template class IntSetInit { public: /// Initialize \a s with iterator \a i static void init(IntSet& s, I& i) { Region reg; Support::DynamicArray d(reg); int n=0; unsigned int size = 0; while (i()) { d[n].min = i.min(); d[n].max = i.max(); size += i.width(); ++n; ++i; } if (n > 0) { IntSet::IntSetObject* o = IntSet::IntSetObject::allocate(n); for (int j=0; jr[j]=d[j]; o->size = size; s.object(o); } } }; /// Initialize integer set with integer set template<> class IntSetInit { public: static void init(IntSet& s, const IntSet& i) { s.object(i.object()); } }; /// Initialize integer set with iterator template IntSet::IntSet(I& i) { IntSetInit::init(*this,i); } /// Initialize integer set with iterator template IntSet::IntSet(const I& i) { IntSetInit::init(*this,i); } forceinline IntSet::IntSet(const int r[][2], int n) { if (n > 0) init(r,n); } forceinline IntSet::IntSet(const int r[], int n) { if (n > 0) init(r,n); } /// Initialize with integers from vector \a r template<> inline IntSet::IntSet(const std::vector& r) { int n = static_cast(r.size()); if (n > 0) { Region reg; Range* dr = reg.alloc(n); for (int i=0; i(i)]; normalize(&dr[0],n); } } /** \brief Initialize with ranges from vector \a r * * The minimum is the first element and the maximum is the * second element. */ template<> inline IntSet::IntSet(const std::vector>& r) { int n = static_cast(r.size()); if (n > 0) { Region reg; Range* dr = reg.alloc(n); int j=0; for (int i=0; i(i)].first <= r[static_cast(i)].second) { dr[j].min=r[static_cast(i)].first; dr[j].max=r[static_cast(i)].second; j++; } normalize(&dr[0],j); } } forceinline IntSet::IntSet(int n, int m) { init(n,m); } forceinline int IntSet::min(int i) const { assert(object() != nullptr); return static_cast(object())->r[i].min; } forceinline int IntSet::max(int i) const { assert(object() != nullptr); return static_cast(object())->r[i].max; } forceinline unsigned int IntSet::width(int i) const { assert(object() != nullptr); IntSetObject* o = static_cast(object()); return static_cast(o->r[i].max-o->r[i].min)+1; } forceinline int IntSet::ranges(void) const { IntSetObject* o = static_cast(object()); return (o == nullptr) ? 0 : o->n; } forceinline bool IntSet::in(int n) const { IntSetObject* o = static_cast(object()); if ((o == nullptr) || (n < o->r[0].min) || (n > o->r[o->n-1].max)) return false; else return o->in(n); } forceinline int IntSet::min(void) const { IntSetObject* o = static_cast(object()); return (o == nullptr) ? Int::Limits::max : o->r[0].min; } forceinline int IntSet::max(void) const { IntSetObject* o = static_cast(object()); return (o == nullptr) ? Int::Limits::min : o->r[o->n-1].max; } forceinline unsigned int IntSet::size(void) const { IntSetObject* o = static_cast(object()); return (o == nullptr) ? 0U : o->size; } forceinline unsigned int IntSet::width(void) const { IntSetObject* o = static_cast(object()); return (o == nullptr) ? 0U : static_cast(max()-min()+1); } forceinline bool IntSet::operator ==(const IntSet& s) const { IntSetObject* o1 = static_cast(object()); IntSetObject* o2 = static_cast(s.object()); if (o1 == o2) return true; if ((o1 == nullptr) || (o2 == nullptr)) return false; if ((o1->size != o2->size) || (o1->n != o2->n)) return false; return o1->equal(*o2); } forceinline bool IntSet::operator !=(const IntSet& s) const { return !(*this == s); } /* * Range iterator for integer sets * */ forceinline IntSetRanges::IntSetRanges(void) {} forceinline void IntSetRanges::init(const IntSet& s) { int n = s.ranges(); if (n > 0) { i = &static_cast(s.object())->r[0]; e = i+n; } else { i = e = nullptr; } } forceinline IntSetRanges::IntSetRanges(const IntSet& s) { init(s); } forceinline void IntSetRanges::operator ++(void) { i++; } forceinline bool IntSetRanges::operator ()(void) const { return imin; } forceinline int IntSetRanges::max(void) const { return i->max; } forceinline unsigned int IntSetRanges::width(void) const { return static_cast(i->max - i->min) + 1; } /* * Value iterator for integer sets * */ forceinline IntSetValues::IntSetValues(void) {} forceinline IntSetValues::IntSetValues(const IntSet& s) { IntSetRanges r(s); Iter::Ranges::ToValues::init(r); } forceinline void IntSetValues::init(const IntSet& s) { IntSetRanges r(s); Iter::Ranges::ToValues::init(r); } template std::basic_ostream& operator <<(std::basic_ostream& os, const IntSet& is) { std::basic_ostringstream s; s.copyfmt(os); s.width(0); s << '{'; for (int i = 0; i < is.ranges(); ) { int min = is.min(i); int max = is.max(i); if (min == max) s << min; else s << min << ".." << max; i++; if (i < is.ranges()) s << ','; } s << '}'; return os << s.str(); } } // STATISTICS: int-var