First round of using boost/operators is done.

This commit is contained in:
John Wiegley 2007-05-02 03:05:10 +00:00
parent 9e80a6fbcc
commit 103881ff80
7 changed files with 73 additions and 25 deletions

View file

@ -84,7 +84,7 @@ class amount_t::bigint_t
~bigint_t(); ~bigint_t();
}; };
unsigned int sizeof_bigint_t() { std::size_t sizeof_bigint_t() {
return sizeof(amount_t::bigint_t); return sizeof(amount_t::bigint_t);
} }
@ -216,7 +216,7 @@ void amount_t::_copy(const amount_t& amt)
commodity_ = amt.commodity_; commodity_ = amt.commodity_;
} }
void amount_t::_resize(unsigned int prec) void amount_t::_resize(precision_t prec)
{ {
assert(prec < 256); assert(prec < 256);
@ -560,7 +560,7 @@ amount_t& amount_t::operator*=(const amount_t& amt)
commodity_ = amt.commodity_; commodity_ = amt.commodity_;
if (has_commodity() && ! (quantity->flags & BIGINT_KEEP_PREC)) { if (has_commodity() && ! (quantity->flags & BIGINT_KEEP_PREC)) {
unsigned int comm_prec = commodity().precision(); precision_t comm_prec = commodity().precision();
if (quantity->prec > comm_prec + 6U) { if (quantity->prec > comm_prec + 6U) {
mpz_round(MPZ(quantity), MPZ(quantity), quantity->prec, comm_prec + 6U); mpz_round(MPZ(quantity), MPZ(quantity), quantity->prec, comm_prec + 6U);
quantity->prec = comm_prec + 6U; quantity->prec = comm_prec + 6U;
@ -612,7 +612,7 @@ amount_t& amount_t::operator/=(const amount_t& amt)
// plus six places. // plus six places.
if (has_commodity() && ! (quantity->flags & BIGINT_KEEP_PREC)) { if (has_commodity() && ! (quantity->flags & BIGINT_KEEP_PREC)) {
unsigned int comm_prec = commodity().precision(); precision_t comm_prec = commodity().precision();
if (quantity->prec > comm_prec + 6U) { if (quantity->prec > comm_prec + 6U) {
mpz_round(MPZ(quantity), MPZ(quantity), quantity->prec, comm_prec + 6U); mpz_round(MPZ(quantity), MPZ(quantity), quantity->prec, comm_prec + 6U);
quantity->prec = comm_prec + 6U; quantity->prec = comm_prec + 6U;
@ -650,6 +650,44 @@ bool amount_t::zero() const
return realzero(); return realzero();
} }
long amount_t::to_long() const
{
if (! quantity)
return 0;
mpz_set(temp, MPZ(quantity));
mpz_ui_pow_ui(divisor, 10, quantity->prec);
mpz_tdiv_q(temp, temp, divisor);
return mpz_get_si(temp);
}
double amount_t::to_double() const
{
if (! quantity)
return 0.0;
mpz_t remainder;
mpz_init(remainder);
mpz_set(temp, MPZ(quantity));
mpz_ui_pow_ui(divisor, 10, quantity->prec);
mpz_tdiv_qr(temp, remainder, temp, divisor);
char * quotient_s = mpz_get_str(NULL, 10, temp);
char * remainder_s = mpz_get_str(NULL, 10, remainder);
std::ostringstream num;
num << quotient_s << '.' << remainder_s;
std::free(quotient_s);
std::free(remainder_s);
mpz_clear(remainder);
return lexical_cast<double>(num.str());
}
amount_t amount_t::value(const moment_t& moment) const amount_t amount_t::value(const moment_t& moment) const
{ {
if (quantity) { if (quantity) {
@ -660,7 +698,7 @@ amount_t amount_t::value(const moment_t& moment) const
return *this; return *this;
} }
amount_t amount_t::round(unsigned int prec) const amount_t amount_t::round(precision_t prec) const
{ {
amount_t t = *this; amount_t t = *this;

View file

@ -66,8 +66,9 @@ DECLARE_EXCEPTION(amount_error);
* math, and also for uncommoditized math. In the commoditized case, * math, and also for uncommoditized math. In the commoditized case,
* commodities keep track of how they are used, and will always * commodities keep track of how they are used, and will always
* display back to the user after the same fashion. For * display back to the user after the same fashion. For
* uncommoditized numbers, no display truncation is ever done. * uncommoditized numbers, no display truncation is ever done. In
* Internally, precision is always kept to an excessive degree. * both cases, internal precision is always kept to an excessive
* degree.
*/ */
class amount_t class amount_t
: public ordered_field_operators<amount_t, : public ordered_field_operators<amount_t,
@ -78,6 +79,10 @@ DECLARE_EXCEPTION(amount_error);
public: public:
class bigint_t; class bigint_t;
// jww (2007-05-01): Change my uses of unsigned int to use this type
// for precision values. Or perhaps just std::size_t?
typedef uint_least16_t precision_t;
static void initialize(); static void initialize();
static void shutdown(); static void shutdown();
@ -92,7 +97,7 @@ protected:
void _copy(const amount_t& amt); void _copy(const amount_t& amt);
void _release(); void _release();
void _dup(); void _dup();
void _resize(unsigned int prec); void _resize(precision_t prec);
void _clear(); void _clear();
bigint_t * quantity; bigint_t * quantity;
@ -170,6 +175,9 @@ public:
// test for truth, zero and non-zero // test for truth, zero and non-zero
operator bool() const { operator bool() const {
return nonzero();
}
bool nonzero() const {
return ! zero(); return ! zero();
} }
@ -180,6 +188,8 @@ public:
} }
// conversion methods // conversion methods
long to_long() const;
double to_double() const;
string to_string() const; string to_string() const;
string to_fullstring() const; string to_fullstring() const;
string quantity_string() const; string quantity_string() const;
@ -220,7 +230,7 @@ public:
optional<string> tag() const; optional<string> tag() const;
// general methods // general methods
amount_t round(unsigned int prec) const; amount_t round(precision_t prec) const;
amount_t round() const; amount_t round() const;
amount_t unround() const; amount_t unround() const;
amount_t value(const moment_t& moment) const; amount_t value(const moment_t& moment) const;

View file

@ -138,7 +138,7 @@ typedef std::pair<const string, commodity_t *> commodities_pair;
typedef std::vector<commodity_t *> commodities_array; typedef std::vector<commodity_t *> commodities_array;
class commodity_t class commodity_t : public equality_comparable<commodity_t>
{ {
friend class annotated_commodity_t; friend class annotated_commodity_t;

View file

@ -147,9 +147,9 @@ void export_amount()
.def(! self) .def(! self)
.def(self_ns::int_(self)) .def("__int__", &amount_t::to_long)
.def(self_ns::float_(self)) .def("__float__", &amount_t::to_double)
.def("__nonzero__", &amount_t::nonzero)
.def("__abs__", &amount_t::abs) .def("__abs__", &amount_t::abs)
.def("__str__", &amount_t::to_string) .def("__str__", &amount_t::to_string)
.def("__repr__", &amount_t::to_fullstring) .def("__repr__", &amount_t::to_fullstring)

View file

@ -415,8 +415,8 @@ void BasicAmountTestCase::testIntegerConversion()
amount_t x1(123456L); amount_t x1(123456L);
assertEqual(true, bool(x1)); assertEqual(true, bool(x1));
assertEqual(123456L, long(x1)); assertEqual(123456L, x1.to_long());
assertEqual(123456.0, double(x1)); assertEqual(123456.0, x1.to_double());
assertEqual(string("123456"), x1.to_string()); assertEqual(string("123456"), x1.to_string());
assertEqual(string("123456"), x1.quantity_string()); assertEqual(string("123456"), x1.quantity_string());
@ -428,8 +428,8 @@ void BasicAmountTestCase::testFractionalConversion()
amount_t x1(1234.56); amount_t x1(1234.56);
assertEqual(true, bool(x1)); assertEqual(true, bool(x1));
assertEqual(1234L, long(x1)); assertEqual(1234L, x1.to_long());
assertEqual(1234.56, double(x1)); assertEqual(1234.56, x1.to_double());
assertEqual(string("1234.56"), x1.to_string()); assertEqual(string("1234.56"), x1.to_string());
assertEqual(string("1234.56"), x1.quantity_string()); assertEqual(string("1234.56"), x1.quantity_string());

View file

@ -465,8 +465,8 @@ void CommodityAmountTestCase::testConversion()
amount_t x1("$1234.56"); amount_t x1("$1234.56");
assertEqual(true, bool(x1)); assertEqual(true, bool(x1));
assertEqual(1234L, long(x1)); assertEqual(1234L, x1.to_long());
assertEqual(1234.56, double(x1)); assertEqual(1234.56, x1.to_double());
assertEqual(string("$1234.56"), x1.to_string()); assertEqual(string("$1234.56"), x1.to_string());
assertEqual(string("1234.56"), x1.quantity_string()); assertEqual(string("1234.56"), x1.quantity_string());

View file

@ -177,7 +177,7 @@ class CommodityAmountTestCase(unittest.TestCase):
x9 = amount("123.45€") x9 = amount("123.45€")
x10 = amount("-123.45€") x10 = amount("-123.45€")
self.assertTrue(x0.null()) self.assertTrue(x0.is_null())
self.assertTrue(x0.zero()) self.assertTrue(x0.zero())
self.assertTrue(x0.realzero()) self.assertTrue(x0.realzero())
self.assertTrue(x0.sign() == 0) self.assertTrue(x0.sign() == 0)