/* -*- mode: C++; c-basic-offset: 2; indent-tabs-mode: nil -*- */ /* * Main authors: * Christian Schulte * * Copyright: * Christian Schulte, 2005 * * 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 #include #include using namespace Gecode; /// Number of warehouses const int n_warehouses = 5; /// Number of stores const int n_stores = 10; /// Fixed cost for one warehouse const int c_fixed = 30; /// Capacity of a single warehouse const int capacity[n_warehouses] = { 1, 4, 2, 1, 3 }; /// Cost for supply a store by a warehouse const int c_supply[n_stores][n_warehouses] = { {20, 24, 11, 25, 30}, {28, 27, 82, 83, 74}, {74, 97, 71, 96, 70}, { 2, 55, 73, 69, 61}, {46, 96, 59, 83, 4}, {42, 22, 29, 67, 59}, { 1, 5, 73, 59, 56}, {10, 73, 13, 43, 96}, {93, 35, 63, 85, 46}, {47, 65, 55, 71, 95} }; /// Model variants enum { MODEL_SUMCOST, ///< Use sum as total cost MODEL_LEXCOST ///< Use lexicographic cost }; /** * \brief %Example: Locating warehouses * * A company needs to construct warehouses to supply stores with * goods. Each warehouse possibly to be constructed has a certain * capacity defining how many stores it can supply. Constructing a * warehouse incurs a fixed cost. Costs for transportation from * warehouses to stores depend on the locations of warehouses and * stores. * * Determine which warehouses should be constructed and which * warehouse should supply which store such that overall cost * (transportation cost plus construction cost) is smallest. * * Taken from: * Pascal Van Hentenryck, The OPL Optmization Programming Language, * The MIT Press, 1999. * * See also problem 34 at http://www.csplib.org/. * * Note that "Modeling and Programming with Gecode" uses this example * as a case study. * * \ingroup Example * */ template class Warehouses : public Script { protected: /// Which warehouse supplies a store IntVarArray supplier; /// Is a warehouse open (warehouse needed) BoolVarArray open; /// Cost of a store IntVarArray c_store; public: /// Actual model Warehouses(const Options& opt) : Script(opt), supplier(*this, n_stores, 0, n_warehouses-1), open(*this, n_warehouses, 0, 1), c_store(*this, n_stores) { // A warehouse is open, if it supplies to a store for (int s=0; s { protected: /// Total cost IntVar c_total; public: /// Actual model SumCostWarehouses(const Options& opt) : Warehouses(opt) { // Compute total cost c_total = expr(*this, c_fixed*sum(open) + sum(c_store)); } /// Return solution cost virtual IntVar cost(void) const { return c_total; } /// Constructor for cloning \a s SumCostWarehouses(SumCostWarehouses& s) : Warehouses(s) { c_total.update(*this, s.c_total); } /// Copy during cloning virtual Space* copy(void) { return new SumCostWarehouses(*this); } /// Print solution virtual void print(std::ostream& os) const { os << "\tSupplier: " << supplier << std::endl << "\tOpen warehouses: " << open << std::endl << "\tStore cost: " << c_store << std::endl << "\tTotal cost: " << c_total << std::endl << std::endl; } }; /// Model with cost defined lexicographically class LexCostWarehouses : public Warehouses { protected: /// Cost for open warehouses IntVar c_open; /// Cost for stores IntVar c_stores; public: /// Actual model LexCostWarehouses(const Options& opt) : Warehouses(opt) { // Compute costs c_open = expr(*this, sum(open)); c_stores = expr(*this, sum(c_store)); } /// Return solution cost virtual IntVarArgs cost(void) const { return {c_open, c_stores}; } /// Constructor for cloning \a s LexCostWarehouses(LexCostWarehouses& s) : Warehouses(s) { c_open.update(*this, s.c_open); c_stores.update(*this, s.c_stores); } /// Copy during cloning virtual Space* copy(void) { return new LexCostWarehouses(*this); } /// Print solution virtual void print(std::ostream& os) const { os << "\tSupplier: " << supplier << std::endl << "\tOpen warehouses: " << open << std::endl << "\tOpen cost: " << c_open << std::endl << "\tStores cost: " << c_stores << std::endl << std::endl; } }; /** \brief Main-function * \relates Warehouses */ int main(int argc, char* argv[]) { Options opt("Warehouses"); opt.model(MODEL_SUMCOST); opt.model(MODEL_SUMCOST, "sum", "use sum of costs"); opt.model(MODEL_LEXCOST, "lex", "use lexicographic cost"); opt.solutions(0); opt.iterations(10); opt.parse(argc,argv); switch (opt.model()) { case MODEL_SUMCOST: IntMinimizeScript::run(opt); break; case MODEL_LEXCOST: IntLexMinimizeScript::run(opt); break; } return 0; } // STATISTICS: example-any