[1097] | 1 | // -*- c++ -*- |
---|
| 2 | #ifndef LEMON_EXPRESSION_H |
---|
| 3 | #define LEMON_EXPRESSION_H |
---|
| 4 | |
---|
| 5 | #include <iostream> |
---|
| 6 | #include <map> |
---|
| 7 | |
---|
| 8 | namespace lemon { |
---|
| 9 | |
---|
| 10 | /*! \brief Linear expression |
---|
| 11 | |
---|
| 12 | \c Expr<_Col,_Value> implements a class of linear expressions with the |
---|
| 13 | operations of addition and multiplication with scalar. |
---|
| 14 | |
---|
| 15 | \author Marton Makai |
---|
| 16 | */ |
---|
| 17 | template <typename _Col, typename _Value> |
---|
| 18 | class Expr; |
---|
| 19 | |
---|
| 20 | template <typename _Col, typename _Value> |
---|
| 21 | class Expr { |
---|
[1099] | 22 | // protected: |
---|
| 23 | public: |
---|
[1097] | 24 | typedef |
---|
| 25 | typename std::map<_Col, _Value> Data; |
---|
| 26 | Data data; |
---|
| 27 | public: |
---|
[1099] | 28 | void simplify() { |
---|
| 29 | for (typename Data::iterator i=data.begin(); |
---|
| 30 | i!=data.end(); ++i) { |
---|
| 31 | if ((*i).second==0) data.erase(i); |
---|
| 32 | } |
---|
| 33 | } |
---|
[1097] | 34 | Expr() { } |
---|
| 35 | Expr(_Col _col) { |
---|
| 36 | data.insert(std::make_pair(_col, 1)); |
---|
| 37 | } |
---|
| 38 | Expr& operator*=(_Value _value) { |
---|
| 39 | for (typename Data::iterator i=data.begin(); |
---|
| 40 | i!=data.end(); ++i) { |
---|
| 41 | (*i).second *= _value; |
---|
| 42 | } |
---|
[1099] | 43 | simplify(); |
---|
[1097] | 44 | return *this; |
---|
| 45 | } |
---|
| 46 | Expr& operator+=(const Expr<_Col, _Value>& expr) { |
---|
| 47 | for (typename Data::const_iterator j=expr.data.begin(); |
---|
| 48 | j!=expr.data.end(); ++j) { |
---|
| 49 | typename Data::iterator i=data.find((*j).first); |
---|
| 50 | if (i==data.end()) { |
---|
| 51 | data.insert(std::make_pair((*j).first, (*j).second)); |
---|
| 52 | } else { |
---|
| 53 | (*i).second+=(*j).second; |
---|
| 54 | } |
---|
| 55 | } |
---|
[1099] | 56 | simplify(); |
---|
| 57 | return *this; |
---|
| 58 | } |
---|
| 59 | Expr& operator-=(const Expr<_Col, _Value>& expr) { |
---|
| 60 | for (typename Data::const_iterator j=expr.data.begin(); |
---|
| 61 | j!=expr.data.end(); ++j) { |
---|
| 62 | typename Data::iterator i=data.find((*j).first); |
---|
| 63 | if (i==data.end()) { |
---|
| 64 | data.insert(std::make_pair((*j).first, -(*j).second)); |
---|
| 65 | } else { |
---|
| 66 | (*i).second+=-(*j).second; |
---|
| 67 | } |
---|
| 68 | } |
---|
| 69 | simplify(); |
---|
[1097] | 70 | return *this; |
---|
| 71 | } |
---|
| 72 | template <typename _C, typename _V> |
---|
| 73 | friend std::ostream& operator<<(std::ostream& os, |
---|
| 74 | const Expr<_C, _V>& expr); |
---|
| 75 | }; |
---|
| 76 | |
---|
| 77 | template <typename _Col, typename _Value> |
---|
| 78 | Expr<_Col, _Value> operator*(_Value _value, _Col _col) { |
---|
| 79 | Expr<_Col, _Value> tmp(_col); |
---|
| 80 | tmp*=_value; |
---|
[1099] | 81 | tmp.simplify(); |
---|
[1097] | 82 | return tmp; |
---|
| 83 | } |
---|
| 84 | |
---|
| 85 | template <typename _Col, typename _Value> |
---|
| 86 | Expr<_Col, _Value> operator*(_Value _value, |
---|
| 87 | const Expr<_Col, _Value>& expr) { |
---|
| 88 | Expr<_Col, _Value> tmp(expr); |
---|
| 89 | tmp*=_value; |
---|
[1099] | 90 | tmp.simplify(); |
---|
[1097] | 91 | return tmp; |
---|
| 92 | } |
---|
| 93 | |
---|
| 94 | template <typename _Col, typename _Value> |
---|
| 95 | Expr<_Col, _Value> operator+(const Expr<_Col, _Value>& expr1, |
---|
| 96 | const Expr<_Col, _Value>& expr2) { |
---|
| 97 | Expr<_Col, _Value> tmp(expr1); |
---|
| 98 | tmp+=expr2; |
---|
[1099] | 99 | tmp.simplify(); |
---|
| 100 | return tmp; |
---|
| 101 | } |
---|
| 102 | |
---|
| 103 | template <typename _Col, typename _Value> |
---|
| 104 | Expr<_Col, _Value> operator-(const Expr<_Col, _Value>& expr1, |
---|
| 105 | const Expr<_Col, _Value>& expr2) { |
---|
| 106 | Expr<_Col, _Value> tmp(expr1); |
---|
| 107 | tmp-=expr2; |
---|
| 108 | tmp.simplify(); |
---|
[1097] | 109 | return tmp; |
---|
| 110 | } |
---|
| 111 | |
---|
| 112 | template <typename _Col, typename _Value> |
---|
| 113 | std::ostream& operator<<(std::ostream& os, |
---|
| 114 | const Expr<_Col, _Value>& expr) { |
---|
| 115 | for (typename Expr<_Col, _Value>::Data::const_iterator i= |
---|
| 116 | expr.data.begin(); |
---|
| 117 | i!=expr.data.end(); ++i) { |
---|
| 118 | os << (*i).second << "*" << (*i).first << " "; |
---|
| 119 | } |
---|
| 120 | return os; |
---|
| 121 | } |
---|
| 122 | |
---|
| 123 | } //namespace lemon |
---|
| 124 | |
---|
| 125 | #endif //LEMON_EXPRESSION_H |
---|