2011-12-09 21:50:25 +01:00
|
|
|
class Z2
|
|
|
|
{
|
|
|
|
public:
|
2011-12-14 19:32:28 +01:00
|
|
|
typedef ::linear_combination<Z2> linear_combination;
|
2011-12-09 21:50:25 +01:00
|
|
|
// typedef linear_combination_iter<Z2> linear_combination_iter;
|
2011-12-14 19:32:28 +01:00
|
|
|
typedef ::linear_combination_const_iter<Z2> linear_combination_const_iter;
|
2011-12-09 21:50:25 +01:00
|
|
|
|
|
|
|
private:
|
|
|
|
bool v;
|
|
|
|
|
|
|
|
public:
|
|
|
|
Z2 () : v(0) { }
|
|
|
|
Z2 (int x) : v((bool)(x & 1)) { }
|
2012-04-11 07:06:04 +02:00
|
|
|
Z2 (unsigned x) : v((bool)(x & 1)) { }
|
2011-12-09 21:50:25 +01:00
|
|
|
Z2 (bool v_) : v(v_) { }
|
|
|
|
Z2 (const Z2 &x) : v(x.v) { }
|
2011-12-30 01:38:02 +01:00
|
|
|
Z2 (copy, const Z2 &x) : v(x.v) { }
|
2012-05-03 21:27:05 +02:00
|
|
|
Z2 (reader &r) { v = r.read_bool (); }
|
2011-12-09 21:50:25 +01:00
|
|
|
~Z2 () { }
|
|
|
|
|
|
|
|
Z2 &operator = (const Z2 &x) { v = x.v; return *this; }
|
|
|
|
Z2 &operator = (int x) { v = (bool)(x & 1); return *this; }
|
|
|
|
|
|
|
|
bool operator == (const Z2 &x) const { return v == x.v; }
|
2013-01-17 17:05:44 +01:00
|
|
|
bool operator != (const Z2 &x) const { return v != x.v; }
|
2011-12-09 21:50:25 +01:00
|
|
|
|
|
|
|
bool operator == (int x) const { return v == (bool)(x & 1); }
|
|
|
|
bool operator != (int x) const { return !operator == (x); }
|
|
|
|
|
|
|
|
bool is_unit () const { return v; }
|
|
|
|
|
|
|
|
Z2 operator + (const Z2 &x) const { return Z2 (v ^ x.v); }
|
|
|
|
Z2 operator - (const Z2 &x) const { return Z2 (v ^ x.v); }
|
2011-12-14 19:07:53 +01:00
|
|
|
Z2 operator - () const { return *this; }
|
2011-12-09 21:50:25 +01:00
|
|
|
Z2 operator * (const Z2 &x) const { return Z2 (v & x.v); }
|
|
|
|
Z2 operator / (const Z2 &x) const { assert (x.v); return *this; }
|
|
|
|
|
|
|
|
Z2 recip () const { assert (v); return *this; }
|
|
|
|
|
|
|
|
Z2 &operator += (const Z2 &x) { v ^= x.v; return *this; }
|
|
|
|
Z2 &operator -= (const Z2 &x) { v ^= x.v; return *this; }
|
|
|
|
Z2 &operator *= (const Z2 &x) { v &= x.v; return *this; }
|
|
|
|
Z2 &operator /= (const Z2 &x) { assert (x.v); return *this; }
|
|
|
|
|
2012-10-22 19:43:45 +02:00
|
|
|
// *this += z1*z2
|
|
|
|
Z2 &muladdeq (const Z2 &z1, const Z2 &z2)
|
|
|
|
{
|
|
|
|
return operator += (z1 * z2);
|
|
|
|
}
|
|
|
|
|
2011-12-09 21:50:25 +01:00
|
|
|
bool divides (Z2 x) const { return v || !x.v; }
|
|
|
|
bool operator | (const Z2 x) const { return divides (x); }
|
|
|
|
|
|
|
|
Z2 div (Z2 x) const
|
|
|
|
{
|
|
|
|
assert (x.divides (*this));
|
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
|
|
|
|
Z2 gcd (Z2 x) const
|
|
|
|
{
|
|
|
|
assert (v && x.v);
|
|
|
|
return Z2 (1);
|
|
|
|
}
|
|
|
|
|
2012-07-27 21:37:47 +02:00
|
|
|
tuple<Z2, Z2, Z2> extended_gcd (Z2 x) const
|
2011-12-09 21:50:25 +01:00
|
|
|
{
|
|
|
|
if (v)
|
2012-07-27 21:37:47 +02:00
|
|
|
return make_tuple (Z2 (1), Z2 (1), Z2 (0));
|
2011-12-09 21:50:25 +01:00
|
|
|
else
|
2012-07-27 21:37:47 +02:00
|
|
|
return make_tuple (Z2 (1), Z2 (0), Z2 (1));
|
2011-12-09 21:50:25 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
static void show_ring () { printf ("Z2"); }
|
2012-05-03 21:27:05 +02:00
|
|
|
void write_self (writer &w) const { w.write_bool (v); }
|
2011-12-09 21:50:25 +01:00
|
|
|
void show_self () const { printf ("%d", (int)v); }
|
|
|
|
void display_self () const { printf ("%d\n", (int)v); }
|
|
|
|
};
|