template struct Tuple2 { A a; B b; bool operator==(const Tuple2& x) const { return a == x.a && b == x.b; } bool operator!=(const Tuple2& x) const { return !operator==(x); } int Compare(const Tuple2& x) const { return CombineCompare(a, x.a)(b, x.b); } bool operator<=(const Tuple2& x) const { return Compare(x) <= 0; } bool operator>=(const Tuple2& x) const { return Compare(x) >= 0; } bool operator<(const Tuple2& x) const { return Compare(x) < 0; } bool operator>(const Tuple2& x) const { return Compare(x) > 0; } unsigned GetHashValue() const { return CombineHash(a, b); } String ToString() const { return String().Cat() << '(' << a << ", " << b << ')'; } }; template inline void AssertMoveable0(Tuple2 *) { AssertMoveable(); AssertMoveable(); } template inline Tuple2 MakeTuple(const A& a, const B& b) { Tuple2 r; r.a = a; r.b = b; return r; } template struct Tuple3 { A a; B b; C c; bool operator==(const Tuple3& x) const { return a == x.a && b == x.b && c == x.c; } bool operator!=(const Tuple3& x) const { return !operator==(x); } int Compare(const Tuple3& x) const { return CombineCompare(a, x.a)(b, x.b)(c, x.c); } bool operator<=(const Tuple3& x) const { return Compare(x) <= 0; } bool operator>=(const Tuple3& x) const { return Compare(x) >= 0; } bool operator<(const Tuple3& x) const { return Compare(x) < 0; } bool operator>(const Tuple3& x) const { return Compare(x) > 0; } unsigned GetHashValue() const { return CombineHash(a, b, c); } String ToString() const { return String().Cat() << '(' << a << ", " << b << ", " << c << ')'; } }; template inline void AssertMoveable0(Tuple3 *) { AssertMoveable(); AssertMoveable(); AssertMoveable(); } template inline Tuple3 MakeTuple(const A& a, const B& b, const C& c) { Tuple3 r; r.a = a; r.b = b; r.c = c; return r; } template struct Tuple4 { A a; B b; C c; D d; bool operator==(const Tuple4& x) const { return a == x.a && b == x.b && c == x.c && d == x.d; } bool operator!=(const Tuple4& x) const { return !operator==(x); } int Compare(const Tuple4& x) const { return CombineCompare(a, x.a)(b, x.b)(c, x.c)(d, x.d); } bool operator<=(const Tuple4& x) const { return Compare(x) <= 0; } bool operator>=(const Tuple4& x) const { return Compare(x) >= 0; } bool operator<(const Tuple4& x) const { return Compare(x) < 0; } bool operator>(const Tuple4& x) const { return Compare(x) > 0; } unsigned GetHashValue() const { return CombineHash(a, b, c, d); } String ToString() const { return String().Cat() << '(' << a << ", " << b << ", " << c << ", " << d << ')'; } }; template inline void AssertMoveable0(Tuple4 *) { AssertMoveable(); AssertMoveable(); AssertMoveable(); AssertMoveable(); } template inline Tuple4 MakeTuple(const A& a, const B& b, const C& c, const D& d) { Tuple4 r; r.a = a; r.b = b; r.c = c; r.d = d; return r; } template inline T *FindTuple(T *x, int n, const U& key) { while(n--) { if(x->a == key) return x; x++; } return NULL; }