#ifndef POSINT_H
#define POSINT_H
#include <iostream>
#include <vector>
/* This class represents an arbitrarily large integer
* that is at least 0. It is represented by a vector of
* digits, starting from the least-significant digit, and
* with each digit between 0 and B-1.
*/
class PosInt {
private:
// It must ALWAYS be the case that B = Bbase ^ Bpow.
// B is really the one to be concerned about for arithmetic;
// Bbase just determines how the number looks for I/O operations.
static int B;
static int Bbase;
static int Bpow;
std::vector<int> digits;
// Removes leading 0 digits
void normalize();
// Result is -1, 0, or 1 if a is <, =, or > than b,
// up to the specified length.
static int compareDigits (const int* a, int alen, const int* b, int blen);
// Computes dest += x, digit-wise
static void addArray (int* dest, const int* x, int len);
// Computes dest -= x, digit-wise
static void subArray (int* dest, const int* x, int len);
// Computes dest = dest * x, digit-wise
static void mulArray (int* dest, int x, int len);
// Computes dest = dest / x, digit-wise, and returns dest % x
static int divArray (int* dest, int x, int len);
// Computes division with remainder, digit-wise.
static void divremArray
(int* q, int* r, const int* x, int xlen, const int* y, int ylen);
static void fastMulArray (int* dest, const int* x, const int* y, int len);
public:
// Computes division with remainder. After the call, we have
// x = q*y + r, and 0 <= r < y.
static void divrem (PosInt& q, PosInt& r, const PosInt& x, const PosInt& y);
// Sets the base to base^pow.
static void setBase(int base, int pow=1);
PosInt() { }
PosInt (int x) { set(x); }
PosInt (const char* s) { read(s); }
// I/O routines
void print_array(std::ostream& out) const;
void print(std::ostream& out) const;
void read(std::istream& in);
void read(const char* s);
// Sets this PosInt to the given value
void set (int x);
void set (const PosInt& rhs);
// Returns this PosInt as a regular int
int convert () const;
// Sets this PosInt to a random number between 0 and x-1
void rand (const PosInt& x);
// Common comparison tests
bool isZero() const { return digits.empty(); }
bool isOne() const
{ return digits.size() == 1 && digits[0] == 1; }
bool isEven() const;
// Result is -1, 0, or 1 if this is <, =, or > than rhs.
int compare (const PosInt& x) const;
// this = x + y
PosInt& add (const PosInt& x, const PosInt& y);
// this = this + x
PosInt& add (const PosInt& x);
// this = x - y
PosInt& sub (const PosInt& x, const PosInt& y);
// this = this - x
PosInt& sub (const PosInt& x);
// this = x * y
PosInt& mul (const PosInt& x, const PosInt& y);
// this = this * x
PosInt& mul (const PosInt& x)
{ PosInt temp(*this); return mul(temp, x); }
// this = x / y
PosInt& div (const PosInt& x, const PosInt& y)
{ PosInt temp; divrem(*this, temp, x, y); return *this; }
// this = this / y
PosInt& div (const PosInt& x)
{ PosInt temp; divrem(*this, temp, *this, x); return *this; }
// this = x % y
PosInt& mod (const PosInt& x, const PosInt& y)
{ PosInt temp; divrem(temp, *this, x, y); return *this; }
// this = this % y
PosInt& mod (const PosInt& x)
{ PosInt temp; divrem(temp, *this, *this, x); return *this; }
// this = x ^ y
PosInt& pow (const PosInt& x, const PosInt& y);
// this = this ^ x
PosInt& pow (const PosInt& x) { return pow(*this, x); }
// this = gcd(x,y)
PosInt& gcd (const PosInt& x, const PosInt& y);
// this = gcd(x,y) = s*x - t*y
// NOTE THE MINUS SIGN! This is required so that both s and t are
// always non-negative.
PosInt& xgcd (PosInt& s, PosInt& t, const PosInt& x, const PosInt& y);
// return true/false if this is PROBABLY prime
bool MillerRabin () const;
// Like "mul", only faster!
PosInt& fasterMul (const PosInt& x, const PosInt& y);
};
std::ostream& operator<< (std::ostream& out, const PosInt& x);
std::istream& operator>> (std::istream& out, PosInt& x);
#endif // POSINT_H