/** * Copyright (c) 2021 OceanBase * OceanBase CE is licensed under Mulan PubL v2. * You can use this software according to the terms and conditions of the Mulan PubL v2. * You may obtain a copy of Mulan PubL v2 at: * http://license.coscl.org.cn/MulanPubL-2.0 * THIS SOFTWARE IS PROVIDED ON AN "AS IS" BASIS, WITHOUT WARRANTIES OF ANY KIND, * EITHER EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO NON-INFRINGEMENT, * MERCHANTABILITY OR FIT FOR A PARTICULAR PURPOSE. * See the Mulan PubL v2 for more details. */ #include "lib/utility/utility.h" #include "lib/number/ob_number_v2.h" #include "lib/ob_define.h" #include "lib/utility/ob_print_utils.h" #include "lib/alloc/alloc_struct.h" #include "lib/charset/ob_dtoa.h" #include "lib/utility/ob_fast_convert.h" #include #include #include using namespace oceanbase::common; using namespace oceanbase::common::number; using namespace oceanbase::lib; template const char* format(const T &obj, int16_t scale) { static const int64_t BUFFER_SIZE = 4096; static __thread char buffers[BUFFER_NUM][BUFFER_SIZE]; static __thread uint64_t i = 0; char *buffer = buffers[i++ % BUFFER_NUM]; memset(buffer, 0, BUFFER_SIZE); int64_t length = 0; //int16_t scale = 0; obj.format(buffer, BUFFER_SIZE, length, scale); return buffer; } class LimitedAllocator : public ObIAllocator { public: LimitedAllocator() {}; ~LimitedAllocator() {}; public: void *alloc(const int64_t size) { return alloc(size, default_memattr); }; void *alloc(const int64_t size, const ObMemAttr &attr) { void *ret = NULL; if (size <= (int64_t)sizeof(buffer_)) { ret = buffer_; } return ret; }; void free(void *) {} private: uint32_t buffer_[number::ObNumber::MAX_STORE_LEN + 1]; }; class MockPoly : public number::ObCalcVector { public: MockPoly() : idx_(0) {}; public: template MockPoly(const uint64_t base, const Args&... args) { fill(base, args...); }; public: template void fill(const uint64_t base, const Args&... args) { set_base(base); idx_ = 0; fill_(args...); }; template void fill_(const T &head, const Args&... args) { ensure(idx_ + 1); set(idx_, head); idx_ += 1; fill_(args...); }; void fill_(void) { }; public: MockPoly ref(const int64_t start, const int64_t end) const { MockPoly ret; ret.set_base(this->base()); ret.idx_ = idx_; number::ObCalcVector *ptr = &ret; *ptr = number::ObCalcVector::ref(start, end); return ret; }; int64_t to_string(char* buffer, const int64_t length) const { int64_t pos = 0; databuff_printf(buffer, length, pos, "["); for (int64_t i = 0; i < size(); i++) { databuff_printf(buffer, length, pos, "%lu,", at(i)); } databuff_printf(buffer, length, pos, "]"); return pos; }; public: int64_t idx_; }; //class MockPoly //{ // public: // MockPoly() {}; // ~MockPoly() {}; // template // MockPoly(const uint64_t base, const Args&... args) // { // fill(base, args...); // }; // public: // void resize(const int64_t size) // { // store_.resize(size, 0); // }; // template // void fill(const uint64_t base, const Args&... args) // { // base_ = base; // store_.clear(); // fill_(args...); // }; // template // void fill_(const T &head, const Args&... args) // { // store_.push_back(head); // fill_(args...); // }; // void fill_(void) // { // }; // int64_t to_string(char* buffer, const int64_t length) const // { // int64_t pos = 0; // databuff_printf(buffer, length, pos, "["); // std::vector::const_iterator iter; // for (iter = store_.begin(); iter != store_.end(); iter++) // { // databuff_printf(buffer, length, pos, "%lu,", *iter); // } // databuff_printf(buffer, length, pos, "]"); // return pos; // }; // public: // uint64_t at(const int64_t idx) const // { // return store_[idx]; // }; // uint64_t base() const // { // return base_; // }; // int64_t size() const // { // return store_.size(); // }; // int set(const int64_t idx, const uint64_t digit) // { // int ret = OB_SUCCESS; // if (idx >= (int64_t)store_.size()) // { // ret = OB_BUF_NOT_ENOUGH; // } // else if (digit >= base_) // { // ret = OB_DECIMAL_OVERFLOW_WARN; // } // else // { // store_[idx] = digit & 0x00000000ffffffff; // } // return ret; // }; // void set_base(const uint64_t base) // { // base_ = base; // }; // int ensure(const int64_t size) // { // store_.resize(size, 0); // return OB_SUCCESS; // }; // const MockPoly ref(const int64_t start, const int64_t end) const // { // MockPoly ret; // ret.base_ = base_; // for (int64_t i = start; i <= end; i++) // { // ret.store_.push_back(store_[i]); // } // return ret; // }; // int assign(const MockPoly &other, const int64_t start, const int64_t end) // { // for (int64_t i = start; i <= end; i++) // { // store_[i] = other.store_[i - start]; // } // return OB_SUCCESS; // } // private: // uint64_t base_; // std::vector store_; //}; TEST(ObNumber, poly_mono_mul) { MockPoly poly(10, 4); uint64_t multiplier = 0; MockPoly product(10); product.resize(poly.size() + 1); ObCStringHelper helper; int ret = OB_SUCCESS; multiplier = 1; ret = poly_mono_mul(poly, multiplier, product); EXPECT_EQ(OB_SUCCESS, ret); EXPECT_STREQ(helper.convert(MockPoly(10, 0, 4)), helper.convert(product)); multiplier = 2; ret = poly_mono_mul(poly, multiplier, product); EXPECT_EQ(OB_SUCCESS, ret); EXPECT_STREQ(helper.convert(MockPoly(10, 0, 8)), helper.convert(product)); multiplier = 3; ret = poly_mono_mul(poly, multiplier, product); EXPECT_EQ(OB_SUCCESS, ret); EXPECT_STREQ(helper.convert(MockPoly(10, 1, 2)), helper.convert(product)); product.resize(poly.size()); ret = poly_mono_mul(poly, multiplier, product); EXPECT_EQ(OB_INVALID_ARGUMENT, ret); product.resize(poly.size() + 2); ret = poly_mono_mul(poly, multiplier, product); EXPECT_EQ(OB_INVALID_ARGUMENT, ret); } TEST(ObNumber, poly_mono_div) { MockPoly poly(10, 1, 2); uint64_t divisor = 0; MockPoly quotient(10); uint64_t remainder = 0; quotient.resize(poly.size()); ObCStringHelper helper; int ret = OB_SUCCESS; divisor = 1; ret = poly_mono_div(poly, divisor, quotient, remainder); EXPECT_EQ(OB_SUCCESS, ret); EXPECT_STREQ(helper.convert(MockPoly(10, 1, 2)), helper.convert(quotient)); EXPECT_EQ(0, remainder); divisor = 3; ret = poly_mono_div(poly, divisor, quotient, remainder); EXPECT_EQ(OB_SUCCESS, ret); EXPECT_STREQ(helper.convert(MockPoly(10, 0, 4)), helper.convert(quotient)); EXPECT_EQ(0, remainder); divisor = 5; ret = poly_mono_div(poly, divisor, quotient, remainder); EXPECT_EQ(OB_SUCCESS, ret); EXPECT_STREQ(helper.convert(MockPoly(10, 0, 2)), helper.convert(quotient)); EXPECT_EQ(2, remainder); poly.fill(10, 8); quotient.resize(poly.size()); divisor = 9; ret = poly_mono_div(poly, divisor, quotient, remainder); EXPECT_EQ(OB_SUCCESS, ret); EXPECT_STREQ(helper.convert(MockPoly(10, 0)), helper.convert(quotient)); EXPECT_EQ(8, remainder); quotient.resize(poly.size() - 1); ret = poly_mono_div(poly, divisor, quotient, remainder); EXPECT_EQ(OB_INVALID_ARGUMENT, ret); quotient.resize(poly.size() + 1); ret = poly_mono_div(poly, divisor, quotient, remainder); EXPECT_EQ(OB_INVALID_ARGUMENT, ret); } TEST(ObNumber, poly_poly_add) { MockPoly augend(10, 9, 9, 8); MockPoly addend(10, 1); MockPoly sum(10); ObCStringHelper helper; int ret = OB_SUCCESS; sum.resize(std::max(augend.size(), addend.size()) + 1); ret = poly_poly_add(augend, addend, sum); EXPECT_EQ(OB_SUCCESS, ret); EXPECT_STREQ(helper.convert(MockPoly(10, 0, 9, 9, 9)), helper.convert(sum)); addend.fill(10, 2); sum.resize(std::max(augend.size(), addend.size()) + 1); ret = poly_poly_add(augend, addend, sum); EXPECT_EQ(OB_SUCCESS, ret); EXPECT_STREQ(helper.convert(MockPoly(10, 1, 0, 0, 0)), helper.convert(sum)); addend.fill(10, 9, 9, 9); sum.resize(std::max(augend.size(), addend.size()) + 1); ret = poly_poly_add(augend, addend, sum); EXPECT_EQ(OB_SUCCESS, ret); EXPECT_STREQ(helper.convert(MockPoly(10, 1, 9, 9, 7)), helper.convert(sum)); sum.resize(std::max(augend.size(), addend.size())); ret = poly_poly_add(augend, addend, sum); EXPECT_EQ(OB_INVALID_ARGUMENT, ret); sum.resize(std::max(augend.size(), addend.size()) + 2); ret = poly_poly_add(augend, addend, sum); EXPECT_EQ(OB_INVALID_ARGUMENT, ret); } TEST(ObNumber, poly_poly_sub) { MockPoly minuend(10, 9, 9, 8); MockPoly subtrahend(10, 1); MockPoly remainder(10); bool negative = false; ObCStringHelper helper; int ret = OB_SUCCESS; remainder.resize(std::max(minuend.size(), subtrahend.size())); ret = poly_poly_sub(minuend, subtrahend, remainder, negative); EXPECT_EQ(OB_SUCCESS, ret); EXPECT_STREQ(helper.convert(MockPoly(10, 9, 9, 7)), helper.convert(remainder)); EXPECT_FALSE(negative); subtrahend.fill(10, 9, 9, 9); remainder.resize(std::max(minuend.size(), subtrahend.size())); ret = poly_poly_sub(minuend, subtrahend, remainder, negative); EXPECT_EQ(OB_SUCCESS, ret); EXPECT_STREQ(helper.convert(MockPoly(10, 0, 0, 1)), helper.convert(remainder)); EXPECT_TRUE(negative); subtrahend.fill(10, 9, 9, 9, 9, 9); remainder.resize(std::max(minuend.size(), subtrahend.size())); ret = poly_poly_sub(minuend, subtrahend, remainder, negative); EXPECT_EQ(OB_SUCCESS, ret); EXPECT_STREQ(helper.convert(MockPoly(10, 9, 9, 0, 0, 1)), helper.convert(remainder)); EXPECT_TRUE(negative); remainder.resize(std::max(minuend.size(), subtrahend.size()) - 1); ret = poly_poly_sub(minuend, subtrahend, remainder, negative); EXPECT_EQ(OB_INVALID_ARGUMENT, ret); remainder.resize(std::max(minuend.size(), subtrahend.size()) + 1); ret = poly_poly_sub(minuend, subtrahend, remainder, negative); EXPECT_EQ(OB_INVALID_ARGUMENT, ret); } TEST(ObNumber, poly_poly_mul) { MockPoly multiplicand(10, 9, 9, 9, 9); MockPoly multiplier(10, 1); MockPoly product(10); ObCStringHelper helper; int ret = OB_SUCCESS; product.resize(multiplicand.size() + multiplier.size()); ret = poly_poly_mul(multiplicand, multiplier, product); EXPECT_EQ(OB_SUCCESS, ret); EXPECT_STREQ(helper.convert(MockPoly(10, 0, 9, 9, 9, 9)), helper.convert(product)); multiplier.fill(10, 9, 9); product.resize(multiplicand.size() + multiplier.size()); ret = poly_poly_mul(multiplicand, multiplier, product); EXPECT_EQ(OB_SUCCESS, ret); EXPECT_STREQ(helper.convert(MockPoly(10, 9, 8, 9, 9, 0, 1)), helper.convert(product)); multiplier.fill(10, 9, 9, 9, 9); product.resize(multiplicand.size() + multiplier.size()); ret = poly_poly_mul(multiplicand, multiplier, product); EXPECT_EQ(OB_SUCCESS, ret); EXPECT_STREQ(helper.convert(MockPoly(10, 9, 9, 9, 8, 0, 0, 0, 1)), helper.convert(product)); multiplicand.fill(10, 1, 0, 0, 0); multiplier.fill(10, 1, 0, 0, 0); product.resize(multiplicand.size() + multiplier.size()); ret = poly_poly_mul(multiplicand, multiplier, product); EXPECT_EQ(OB_SUCCESS, ret); EXPECT_STREQ(helper.convert(MockPoly(10, 0, 1, 0, 0, 0, 0, 0, 0)), helper.convert(product)); product.resize(multiplicand.size() + multiplier.size() - 1); ret = poly_poly_mul(multiplicand, multiplier, product); EXPECT_EQ(OB_INVALID_ARGUMENT, ret); product.resize(multiplicand.size() + multiplier.size() + 1); ret = poly_poly_mul(multiplicand, multiplier, product); EXPECT_EQ(OB_INVALID_ARGUMENT, ret); multiplicand.fill(0x0000000100000000, 0xffffffff, 0xffffffff, 0xffffffff, 0xffffffff); multiplier.fill(0x0000000100000000, 0xffffffff, 0xffffffff, 0xffffffff, 0xffffffff); product.fill(0x0000000100000000); product.resize(multiplicand.size() + multiplier.size()); ret = poly_poly_mul(multiplicand, multiplier, product); EXPECT_EQ(OB_SUCCESS, ret); EXPECT_STREQ(helper.convert(MockPoly(0x0000000100000000, 4294967295, 4294967295, 4294967295, 4294967294, 0, 0, 0, 1)), helper.convert(product)); } //TEST(ObNumber, poly_poly_div) //{ // MockPoly dividend(10, 9, 9, 9); // MockPoly divisor(10, 4, 5); // MockPoly quotient(10); // MockPoly remainder(10); // ObCStringHelper helper; // int ret = OB_SUCCESS; // // quotient.resize(dividend.size() - divisor.size() + 1); // remainder.resize(divisor.size()); // ret = poly_poly_div(dividend, divisor, quotient, remainder); // EXPECT_EQ(OB_SUCCESS, ret); // EXPECT_STREQ(helper.convert(MockPoly(10, 2, 2)), helper.convert(quotient)); // EXPECT_STREQ(helper.convert(MockPoly(10, 0, 9)), helper.convert(remainder)); // // dividend.fill(10, 4, 1, 0, 0); // divisor.fill(10, 5, 8, 8); // quotient.resize(dividend.size() - divisor.size() + 1); // remainder.resize(divisor.size()); // ret = poly_poly_div(dividend, divisor, quotient, remainder); // EXPECT_EQ(OB_SUCCESS, ret); // EXPECT_STREQ(helper.convert(MockPoly(10, 0, 6)), helper.convert(quotient)); // EXPECT_STREQ(helper.convert(MockPoly(10, 5, 7, 2)), helper.convert(remainder)); // // dividend.fill(10, 4, 1, 0, 0); // divisor.fill(10, 2, 0); // quotient.resize(dividend.size() - divisor.size() + 1); // remainder.resize(divisor.size()); // ret = poly_poly_div(dividend, divisor, quotient, remainder); // EXPECT_EQ(OB_SUCCESS, ret); // EXPECT_STREQ(helper.convert(MockPoly(10, 2, 0, 5)), helper.convert(quotient)); // EXPECT_STREQ(helper.convert(MockPoly(10, 0, 0)), helper.convert(remainder)); // // dividend.fill(10, 4, 1, 0, 0); // divisor.fill(10, 2); // quotient.resize(dividend.size() - divisor.size() + 1); // remainder.resize(divisor.size()); // ret = poly_poly_div(dividend, divisor, quotient, remainder); // EXPECT_EQ(OB_SUCCESS, ret); // EXPECT_STREQ(helper.convert(MockPoly(10, 2, 0, 5, 0)), helper.convert(quotient)); // EXPECT_STREQ(helper.convert(MockPoly(10, 0)), helper.convert(remainder)); // // dividend.fill(10, 4, 1, 0, 0); // divisor.fill(10, 1); // quotient.resize(dividend.size() - divisor.size() + 1); // remainder.resize(divisor.size()); // ret = poly_poly_div(dividend, divisor, quotient, remainder); // EXPECT_EQ(OB_SUCCESS, ret); // EXPECT_STREQ(helper.convert(MockPoly(10, 4, 1, 0, 0)), helper.convert(quotient)); // EXPECT_STREQ(helper.convert(MockPoly(10, 0)), helper.convert(remainder)); // // dividend.fill(10, 1, 1); // divisor.fill(10, 3); // quotient.resize(dividend.size() - divisor.size() + 1); // remainder.resize(divisor.size()); // ret = poly_poly_div(dividend, divisor, quotient, remainder); // EXPECT_EQ(OB_SUCCESS, ret); // EXPECT_STREQ(helper.convert(MockPoly(10, 0, 3)), helper.convert(quotient)); // EXPECT_STREQ(helper.convert(MockPoly(10, 2)), helper.convert(remainder)); //} class ParseHelperTester : public ObNumberBuilder { public: ParseHelperTester() {}; ~ParseHelperTester() {}; public: int fill(const char *str) { return number_.from(str, allocator_); }; int fill_old(const char *str) { return number_.from_v1(str, strlen(str), allocator_); }; int from(const uint32_t desc, ObCalcVector &vector) { return number_.from(desc, vector, allocator_); }; private: CharArena allocator_; }; std::string negate(const char* str) { std::string ret; if ('-' == str[0]) { ret = &str[1]; } else if (0 != strcmp(str, "0")) { ret = "-"; ret += str; } else { ret = "0"; } return ret; } TEST(ObNumberParseHelper, from_format) { uint32_t digits[number::ObNumber::MAX_STORE_LEN]; number::ObNumber::Desc desc; desc.len_ = 0; desc.reserved_ = 0; desc.sign_ = number::ObNumber::POSITIVE; desc.exp_ = number::ObNumber::EXP_ZERO; ParseHelperTester nph; nph.number_.assign(desc.desc_, digits); int ret = OB_SUCCESS; const int64_t MAX_BUF_SIZE = 256; char buf_v1[MAX_BUF_SIZE]; char buf_v2[MAX_BUF_SIZE]; int64_t pos_v1 = 0; int64_t pos_v2 = 0; char buf_alloc[MAX_BUF_SIZE]; int64_t pos = 0; ObDataBuffer allocator(buf_alloc, MAX_BUF_SIZE); number::ObNumber num1; number::ObNumber num2; EXPECT_EQ(OB_SUCCESS, num1.from_v1("0.1", 3, allocator)); EXPECT_EQ(OB_SUCCESS, num2.from_v2("0.1", 3, allocator)); OB_LOG(INFO, "debug jianhua", K(num1), K(num2)); ASSERT_EQ(0, num1.compare_v1(num2)); allocator.free(); // positive ret = nph.fill("1.1"); // normal EXPECT_EQ(OB_SUCCESS, ret); EXPECT_STREQ("1.1", format(nph.number_, 1)); EXPECT_EQ(0x40, nph.get_exp()); EXPECT_EQ(2, nph.get_length()); ret = nph.fill("0.1"); // normal EXPECT_EQ(OB_SUCCESS, ret); EXPECT_STREQ("0.1", format(nph.number_, 1)); EXPECT_EQ(0x3f, nph.get_exp()); EXPECT_EQ(1, nph.get_length()); ret = nph.fill("1.01"); // normal EXPECT_EQ(OB_SUCCESS, ret); EXPECT_STREQ("1.01", format(nph.number_, 2)); EXPECT_EQ(0x40, nph.get_exp()); EXPECT_EQ(2, nph.get_length()); ret = nph.fill("1.000000001"); // decimal 1digit EXPECT_EQ(OB_SUCCESS, ret); EXPECT_STREQ("1.000000001", format(nph.number_, 9)); EXPECT_EQ(0x40, nph.get_exp()); EXPECT_EQ(2, nph.get_length()); ret = nph.fill("1.000000001234"); // decimal 2digit, decimal first digit not zero EXPECT_EQ(OB_SUCCESS, ret); EXPECT_STREQ("1.000000001234", format(nph.number_, 12)); EXPECT_EQ(0x40, nph.get_exp()); EXPECT_EQ(3, nph.get_length()); ret = nph.fill("1.0000000001234"); // decimal 2digit, decimal first digit is zero EXPECT_EQ(OB_SUCCESS, ret); EXPECT_STREQ("1.0000000001234", format(nph.number_, 13)); EXPECT_EQ(0x40, nph.get_exp()); EXPECT_EQ(3, nph.get_length()); ret = nph.fill("0.000000001"); // decimal 1digit EXPECT_EQ(OB_SUCCESS, ret); EXPECT_STREQ("0.000000001", format(nph.number_, 9)); EXPECT_EQ(0x3f, nph.get_exp()); EXPECT_EQ(1, nph.get_length()); ret = nph.fill("0.000000001234"); // decimal 2digit, decimal first digit not zero EXPECT_EQ(OB_SUCCESS, ret); EXPECT_STREQ("0.000000001234", format(nph.number_, 12)); EXPECT_EQ(0x3f, nph.get_exp()); EXPECT_EQ(2, nph.get_length()); ret = nph.fill("0.0000000001234"); // decimal 2digit, decimal first digit is zero EXPECT_EQ(OB_SUCCESS, ret); EXPECT_STREQ("0.0000000001234", format(nph.number_, 13)); EXPECT_EQ(0x3e, nph.get_exp()); EXPECT_EQ(1, nph.get_length()); ret = nph.fill("00.1"); // extra zero on integer head EXPECT_EQ(OB_SUCCESS, ret); EXPECT_STREQ("0.1", format(nph.number_, 1)); EXPECT_EQ(0x3f, nph.get_exp()); EXPECT_EQ(1, nph.get_length()); ret = nph.fill("0000000000.1"); //extra zero on integer head, integer digit occupy 2digit EXPECT_EQ(OB_SUCCESS, ret); EXPECT_STREQ("0.1", format(nph.number_, 1)); EXPECT_EQ(0x3f, nph.get_exp()); EXPECT_EQ(1, nph.get_length()); ret = nph.fill("123456789.01"); // integer 1digit EXPECT_EQ(OB_SUCCESS, ret); EXPECT_STREQ("123456789.01", format(nph.number_, 2)); EXPECT_EQ(0x40, nph.get_exp()); EXPECT_EQ(2, nph.get_length()); ret = nph.fill("1234567891.01"); // integer 2digit, 1over EXPECT_EQ(OB_SUCCESS, ret); EXPECT_STREQ("1234567891.01", format(nph.number_, 2)); EXPECT_EQ(0x41, nph.get_exp()); EXPECT_EQ(3, nph.get_length()); ret = nph.fill("12345678910.01"); // integer 2digit, 2over EXPECT_EQ(OB_SUCCESS, ret); EXPECT_STREQ("12345678910.01", format(nph.number_, 2)); EXPECT_EQ(0x41, nph.get_exp()); EXPECT_EQ(3, nph.get_length()); ret = nph.fill("123456789100.01"); // integer 2digit, 3over EXPECT_EQ(OB_SUCCESS, ret); EXPECT_STREQ("123456789100.01", format(nph.number_, 2)); EXPECT_EQ(0x41, nph.get_exp()); EXPECT_EQ(3, nph.get_length()); ret = nph.fill("123456789100000000.01"); // integer 2digit, align EXPECT_EQ(OB_SUCCESS, ret); EXPECT_STREQ("123456789100000000.01", format(nph.number_, 2)); EXPECT_EQ(0x41, nph.get_exp()); EXPECT_EQ(3, nph.get_length()); ret = nph.fill("1234567891000000000.01"); // integer 3digit, 1over EXPECT_EQ(OB_SUCCESS, ret); EXPECT_STREQ("1234567891000000000.01", format(nph.number_, 2)); EXPECT_EQ(0x42, nph.get_exp()); EXPECT_EQ(4, nph.get_length()); ret = nph.fill("1234567890000000001.01"); // integer 3digit, 1over EXPECT_EQ(OB_SUCCESS, ret); EXPECT_STREQ("1234567890000000001.01", format(nph.number_, 2)); EXPECT_EQ(0x42, nph.get_exp()); EXPECT_EQ(4, nph.get_length()); ret = nph.fill("12345678910000000000.01"); // integer 3digit, 2over EXPECT_EQ(OB_SUCCESS, ret); EXPECT_STREQ("12345678910000000000.01", format(nph.number_, 2)); EXPECT_EQ(0x42, nph.get_exp()); EXPECT_EQ(4, nph.get_length()); ret = nph.fill("12345678900000000001.01"); // integer 3digit, 2over EXPECT_EQ(OB_SUCCESS, ret); EXPECT_STREQ("12345678900000000001.01", format(nph.number_, 2)); EXPECT_EQ(0x42, nph.get_exp()); EXPECT_EQ(4, nph.get_length()); ret = nph.fill("12345678900000000001"); // integer 3digit, 2over EXPECT_EQ(OB_SUCCESS, ret); EXPECT_STREQ("12345678900000000001", format(nph.number_, 0)); EXPECT_EQ(0x42, nph.get_exp()); EXPECT_EQ(3, nph.get_length()); //////////////////////////////////////////////////////////////////////////////////////////////////// // negative ret = nph.fill("-1.1"); // normal EXPECT_EQ(OB_SUCCESS, ret); EXPECT_STREQ("-1.1", format(nph.number_, 1)); EXPECT_EQ(0x40, nph.get_exp()); EXPECT_EQ(2, nph.get_length()); ret = nph.fill("-0.1"); // normal EXPECT_EQ(OB_SUCCESS, ret); EXPECT_STREQ("-0.1", format(nph.number_, 1)); EXPECT_EQ(0x41, nph.get_exp()); EXPECT_EQ(1, nph.get_length()); ret = nph.fill("-1.01"); // normal EXPECT_EQ(OB_SUCCESS, ret); EXPECT_STREQ("-1.01", format(nph.number_, 2)); EXPECT_EQ(0x40, nph.get_exp()); EXPECT_EQ(2, nph.get_length()); ret = nph.fill("-1.000000001"); // decimal 1digit EXPECT_EQ(OB_SUCCESS, ret); EXPECT_STREQ("-1.000000001", format(nph.number_, 9)); EXPECT_EQ(0x40, nph.get_exp()); EXPECT_EQ(2, nph.get_length()); ret = nph.fill("-1.000000001234"); // decimal 2digit, decimal first digit not zero EXPECT_EQ(OB_SUCCESS, ret); EXPECT_STREQ("-1.000000001234", format(nph.number_, 12)); EXPECT_EQ(0x40, nph.get_exp()); EXPECT_EQ(3, nph.get_length()); ret = nph.fill("-1.0000000001234"); // decimal 2digit, decimal first digit is zero EXPECT_EQ(OB_SUCCESS, ret); EXPECT_STREQ("-1.0000000001234", format(nph.number_, 13)); EXPECT_EQ(0x40, nph.get_exp()); EXPECT_EQ(3, nph.get_length()); ret = nph.fill("-0.000000001"); // decimal 1digit EXPECT_EQ(OB_SUCCESS, ret); EXPECT_STREQ("-0.000000001", format(nph.number_, 9)); EXPECT_EQ(0x41, nph.get_exp()); EXPECT_EQ(1, nph.get_length()); ret = nph.fill("-0.000000001234"); // decimal 2digit, decimal first digit not zero EXPECT_EQ(OB_SUCCESS, ret); EXPECT_STREQ("-0.000000001234", format(nph.number_, 12)); EXPECT_EQ(0x41, nph.get_exp()); EXPECT_EQ(2, nph.get_length()); ret = nph.fill("-0.0000000001234"); // decimal 2digit, decimal first digit is zero EXPECT_EQ(OB_SUCCESS, ret); EXPECT_STREQ("-0.0000000001234", format(nph.number_, 13)); EXPECT_EQ(0x42, nph.get_exp()); EXPECT_EQ(1, nph.get_length()); ret = nph.fill("-00.1"); // extra zero on integer head EXPECT_EQ(OB_SUCCESS, ret); EXPECT_STREQ("-0.1", format(nph.number_, 1)); EXPECT_EQ(0x41, nph.get_exp()); EXPECT_EQ(1, nph.get_length()); ret = nph.fill("-0000000000.1"); //extra zero on integer head, integer digit occupy 2digit EXPECT_EQ(OB_SUCCESS, ret); EXPECT_STREQ("-0.1", format(nph.number_, 1)); EXPECT_EQ(0x41, nph.get_exp()); EXPECT_EQ(1, nph.get_length()); ret = nph.fill("-123456789.01"); // integer 1digit EXPECT_EQ(OB_SUCCESS, ret); EXPECT_STREQ("-123456789.01", format(nph.number_, 2)); EXPECT_EQ(0x40, nph.get_exp()); EXPECT_EQ(2, nph.get_length()); ret = nph.fill("-1234567891.01"); // integer 2digit, 1over EXPECT_EQ(OB_SUCCESS, ret); EXPECT_STREQ("-1234567891.01", format(nph.number_, 2)); EXPECT_EQ(0x3f, nph.get_exp()); EXPECT_EQ(3, nph.get_length()); ret = nph.fill("-12345678910.01"); // integer 2digit, 2over EXPECT_EQ(OB_SUCCESS, ret); EXPECT_STREQ("-12345678910.01", format(nph.number_, 2)); EXPECT_EQ(0x3f, nph.get_exp()); EXPECT_EQ(3, nph.get_length()); ret = nph.fill("-123456789100.01"); // integer 2digit, 3over EXPECT_EQ(OB_SUCCESS, ret); EXPECT_STREQ("-123456789100.01", format(nph.number_, 2)); EXPECT_EQ(0x3f, nph.get_exp()); EXPECT_EQ(3, nph.get_length()); ret = nph.fill("-123456789100000000.01"); // integer 2digit, align EXPECT_EQ(OB_SUCCESS, ret); EXPECT_STREQ("-123456789100000000.01", format(nph.number_, 2)); EXPECT_EQ(0x3f, nph.get_exp()); EXPECT_EQ(3, nph.get_length()); ret = nph.fill("-1234567891000000000.01"); // integer 3digit, 1over EXPECT_EQ(OB_SUCCESS, ret); EXPECT_STREQ("-1234567891000000000.01", format(nph.number_, 2)); EXPECT_EQ(0x3e, nph.get_exp()); EXPECT_EQ(4, nph.get_length()); ret = nph.fill("-1234567890000000001.01"); // integer 3digit, 1over EXPECT_EQ(OB_SUCCESS, ret); EXPECT_STREQ("-1234567890000000001.01", format(nph.number_, 2)); EXPECT_EQ(0x3e, nph.get_exp()); EXPECT_EQ(4, nph.get_length()); ret = nph.fill("-12345678910000000000.01"); // integer 3digit, 2over EXPECT_EQ(OB_SUCCESS, ret); EXPECT_STREQ("-12345678910000000000.01", format(nph.number_, 2)); EXPECT_EQ(0x3e, nph.get_exp()); EXPECT_EQ(4, nph.get_length()); ret = nph.fill("-12345678900000000001.01"); // integer 3digit, 2over EXPECT_EQ(OB_SUCCESS, ret); EXPECT_STREQ("-12345678900000000001.01", format(nph.number_, 2)); EXPECT_EQ(0x3e, nph.get_exp()); EXPECT_EQ(4, nph.get_length()); ret = nph.fill("-12345678900000000001"); // integer 3digit, 2over EXPECT_EQ(OB_SUCCESS, ret); EXPECT_STREQ("-12345678900000000001", format(nph.number_, 0)); EXPECT_EQ(0x3e, nph.get_exp()); EXPECT_EQ(3, nph.get_length()); //////////////////////////////////////////////////////////////////////////////////////////////////// // abnormal ret = nph.fill("9999999999999999999999999999999999999999999994"); EXPECT_EQ(OB_SUCCESS, ret); EXPECT_STREQ("9999999999999999999999999999999999999999999994", format(nph.number_, 0)); ret = nph.fill("9999999999999999999999999999999999999999999995"); EXPECT_EQ(OB_SUCCESS, ret); EXPECT_STREQ("9999999999999999999999999999999999999999999995", format(nph.number_, 0)); ret = nph.fill("1111111112222222223333333334444444445555555556"); EXPECT_EQ(OB_SUCCESS, ret); EXPECT_STREQ("1111111112222222223333333334444444445555555556", format(nph.number_, 0)); ret = nph.fill("111111111222222222333333333444444444555555555"); EXPECT_EQ(OB_SUCCESS, ret); EXPECT_STREQ("111111111222222222333333333444444444555555555", format(nph.number_, 0)); ret = nph.fill("111111111222222222333333333444444444555555555666666666"); EXPECT_EQ(OB_SUCCESS, ret); EXPECT_STREQ("111111111222222222333333333444444444555555555666666666", format(nph.number_, 0)); ret = nph.fill("1111111112222222223333333334444444445555555556666666667"); EXPECT_EQ(OB_SUCCESS, ret); EXPECT_STREQ("1111111112222222223333333334444444445555555556666666667", format(nph.number_, 0)); ret = nph.fill("11111111122222222233333333344444444455555555.5666666/66"); EXPECT_EQ(OB_INVALID_NUMERIC, ret); EXPECT_STREQ("11111111122222222233333333344444444455555555.6", format(nph.number_, 1)); ret = nph.fill("11111111122222222233333333344444444455555555.5666666667"); EXPECT_EQ(OB_SUCCESS, ret); EXPECT_STREQ("11111111122222222233333333344444444455555555.6", format(nph.number_, 1)); ret = nph.fill(".1111111112222222223333333334444444445555555556"); EXPECT_EQ(OB_SUCCESS, ret); EXPECT_STREQ("0.111111111222222222333333333444444444555555556", format(nph.number_, 45)); ret = nph.fill(".111111111222222222333333333444444444555555555"); EXPECT_EQ(OB_SUCCESS, ret); EXPECT_STREQ("0.111111111222222222333333333444444444555555555", format(nph.number_, 45)); ret = nph.fill("111111111.2222222223333333334444444445555555556"); EXPECT_EQ(OB_SUCCESS, ret); EXPECT_STREQ("111111111.222222222333333333444444444555555556", format(nph.number_, 36)); ret = nph.fill("111111111.222222222333333333444444444555555555"); EXPECT_EQ(OB_SUCCESS, ret); EXPECT_STREQ("111111111.222222222333333333444444444555555555", format(nph.number_, 36)); ret = nph.fill("111111111.222222222333333333444444444555555555666666666"); EXPECT_EQ(OB_SUCCESS, ret); EXPECT_STREQ("111111111.222222222333333333444444444555555556", format(nph.number_, 36)); ret = nph.fill("111111111.2222222223333333334444444445555555556666666667"); EXPECT_EQ(OB_SUCCESS, ret); EXPECT_STREQ("111111111.222222222333333333444444444555555556", format(nph.number_, 36)); ret = nph.fill("1111111112.2222222223333333334444444445555555546666666667"); EXPECT_EQ(OB_SUCCESS, ret); EXPECT_STREQ("1111111112.22222222233333333344444444455555555", format(nph.number_, 35)); ret = nph.fill("1111111112.2222222223333333334444444445555555576666666667"); EXPECT_EQ(OB_SUCCESS, ret); EXPECT_STREQ("1111111112.22222222233333333344444444455555556", format(nph.number_, 35)); ret = nph.fill("-9999999999999999999999999999999999999999999994"); EXPECT_EQ(OB_SUCCESS, ret); EXPECT_STREQ("-9999999999999999999999999999999999999999999994", format(nph.number_, 0)); ret = nph.fill("-9999999999999999999999999999999999999999999995"); EXPECT_EQ(OB_SUCCESS, ret); EXPECT_STREQ("-9999999999999999999999999999999999999999999995", format(nph.number_, 0)); ret = nph.fill("-1111111112222222223333333334444444445555555556"); EXPECT_EQ(OB_SUCCESS, ret); EXPECT_STREQ("-1111111112222222223333333334444444445555555556", format(nph.number_, 0)); ret = nph.fill("-111111111222222222333333333444444444555555555"); EXPECT_EQ(OB_SUCCESS, ret); EXPECT_STREQ("-111111111222222222333333333444444444555555555", format(nph.number_, 0)); ret = nph.fill("-111111111222222222333333333444444444555555555666666666"); EXPECT_EQ(OB_SUCCESS, ret); EXPECT_STREQ("-111111111222222222333333333444444444555555555666666666", format(nph.number_, 0)); ret = nph.fill("-1111111112222222223333333334444444445555555556666666667"); EXPECT_EQ(OB_SUCCESS, ret); EXPECT_STREQ("-1111111112222222223333333334444444445555555556666666667", format(nph.number_, 0)); ret = nph.fill("-11111111122222222233333333344444444455555555.566666666"); EXPECT_EQ(OB_SUCCESS, ret); EXPECT_STREQ("-11111111122222222233333333344444444455555555.6", format(nph.number_, 1)); ret = nph.fill("-11111111122222222233333333344444444455555555.5666666667"); EXPECT_EQ(OB_SUCCESS, ret); EXPECT_STREQ("-11111111122222222233333333344444444455555555.6", format(nph.number_, 1)); ret = nph.fill("-.1111111112222222223333333334444444445555555556"); EXPECT_EQ(OB_SUCCESS, ret); EXPECT_STREQ("-0.111111111222222222333333333444444444555555556", format(nph.number_, 45)); ret = nph.fill("-.111111111222222222333333333444444444555555555"); EXPECT_EQ(OB_SUCCESS, ret); EXPECT_STREQ("-0.111111111222222222333333333444444444555555555", format(nph.number_, 45)); ret = nph.fill("-111111111.2222222223333333334444444445555555556"); EXPECT_EQ(OB_SUCCESS, ret); EXPECT_STREQ("-111111111.222222222333333333444444444555555556", format(nph.number_, 36)); ret = nph.fill("-111111111.222222222333333333444444444555555555"); EXPECT_EQ(OB_SUCCESS, ret); EXPECT_STREQ("-111111111.222222222333333333444444444555555555", format(nph.number_, 36)); ret = nph.fill("-111111111.222222222333333333444444444555555555666666666"); EXPECT_EQ(OB_SUCCESS, ret); EXPECT_STREQ("-111111111.222222222333333333444444444555555556", format(nph.number_, 36)); ret = nph.fill("-111111111.2222222223333333334444444445555555556666666667"); EXPECT_EQ(OB_SUCCESS, ret); EXPECT_STREQ("-111111111.222222222333333333444444444555555556", format(nph.number_, 36)); ret = nph.fill("-1111111112.2222222223333333334444444445555555546666666667"); EXPECT_EQ(OB_SUCCESS, ret); EXPECT_STREQ("-1111111112.22222222233333333344444444455555555", format(nph.number_, 35)); ret = nph.fill("-1111111112.2222222223333333334444444445555555576666666667"); EXPECT_EQ(OB_SUCCESS, ret); EXPECT_STREQ("-1111111112.22222222233333333344444444455555556", format(nph.number_, 35)); ret = nph.fill("0.0000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000001"); EXPECT_EQ(OB_SUCCESS, ret); EXPECT_STREQ("0", format(nph.number_, 0)); ret = nph.fill(".0000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000001"); EXPECT_EQ(OB_SUCCESS, ret); EXPECT_STREQ("0", format(nph.number_, 0)); ret = nph.fill("888888888888888888888888888"); EXPECT_EQ(OB_SUCCESS, ret); EXPECT_STREQ("888888888888888888888888888", format(nph.number_, 0)); //代码中限制补零的个数为60为上限,否则报4002错误,所以下面的执行format后,返回错误码为4002 //ret = nph.fill("0.000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000001"); //EXPECT_EQ(OB_SUCCESS, ret); //EXPECT_STREQ("0.000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000001", format(nph.number_, 135)); //ret = nph.fill("0.0000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000010"); //EXPECT_EQ(OB_SUCCESS, ret); //EXPECT_STREQ("0.000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000001", format(nph.number_, 135)); //ret = nph.fill(".000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000001"); //EXPECT_EQ(OB_SUCCESS, ret); //EXPECT_STREQ("0.000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000001", format(nph.number_, 135)); ret = nph.fill("1000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000"); EXPECT_EQ(OB_INTEGER_PRECISION_OVERFLOW, ret); ret = nph.fill("1000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000."); EXPECT_EQ(OB_INTEGER_PRECISION_OVERFLOW, ret); ret = nph.fill("1000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000.0"); EXPECT_EQ(OB_INTEGER_PRECISION_OVERFLOW, ret); ret = nph.fill("999999999999999999999999999999999999999999999000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000"); EXPECT_EQ(OB_SUCCESS, ret); EXPECT_STREQ("999999999999999999999999999999999999999999999000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000", format(nph.number_, 0)); ret = nph.fill("999999999999999999999999999999999999999999999000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000."); EXPECT_EQ(OB_SUCCESS, ret); EXPECT_STREQ("999999999999999999999999999999999999999999999000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000", format(nph.number_, 0)); ret = nph.fill("999999999999999999999999999999999999999999999000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000.0"); EXPECT_EQ(OB_SUCCESS, ret); EXPECT_STREQ("999999999999999999999999999999999999999999999000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000", format(nph.number_, 0)); // special ret = nph.fill("0"); EXPECT_EQ(OB_SUCCESS, ret); EXPECT_STREQ("0", format(nph.number_, 0)); ret = nph.fill("0.0"); EXPECT_EQ(OB_SUCCESS, ret); EXPECT_STREQ("0", format(nph.number_, 0)); ret = nph.fill("0.0000000000"); EXPECT_EQ(OB_SUCCESS, ret); EXPECT_STREQ("0", format(nph.number_, 0)); ret = nph.fill("0000000000.0"); EXPECT_EQ(OB_SUCCESS, ret); EXPECT_STREQ("0", format(nph.number_, 0)); ret = nph.fill("-0.0"); EXPECT_EQ(OB_SUCCESS, ret); EXPECT_STREQ("0", format(nph.number_, 0)); ret = nph.fill("-0.0000000000"); EXPECT_EQ(OB_SUCCESS, ret); EXPECT_STREQ("0", format(nph.number_, 0)); ret = nph.fill("-0000000000.0"); EXPECT_EQ(OB_SUCCESS, ret); EXPECT_STREQ("0", format(nph.number_, 0)); } #define EQ(str1, str2) \ EXPECT_EQ(OB_SUCCESS, nph1.fill(str1)); \ EXPECT_EQ(OB_SUCCESS, nph2.fill(str2)); \ EXPECT_EQ(nph1.number_, nph2.number_); \ EXPECT_LE(nph1.number_, nph2.number_); \ EXPECT_GE(nph1.number_, nph2.number_); \ EXPECT_EQ(OB_SUCCESS, nph3.fill_old(str2)); \ EXPECT_EQ(nph3.number_, nph1.number_); \ EXPECT_LE(nph1.number_, nph3.number_); \ EXPECT_GE(nph1.number_, nph3.number_); \ EXPECT_EQ(OB_SUCCESS, nph3.fill_old(str1)); \ EXPECT_EQ(nph3.number_, nph1.number_); \ EXPECT_EQ(nph3.number_, nph2.number_); \ EXPECT_LE(nph3.number_, nph2.number_); \ EXPECT_GE(nph3.number_, nph2.number_); #define NE(str1, str2) \ EXPECT_EQ(OB_SUCCESS, nph1.fill(str1)); \ EXPECT_EQ(OB_SUCCESS, nph2.fill(str2)); \ EXPECT_NE(nph1, nph2); \ EXPECT_EQ(OB_SUCCESS, nph3.fill_old(str2)); \ EXPECT_EQ(nph3, nph2); \ EXPECT_NE(nph1, nph3); \ EXPECT_EQ(OB_SUCCESS, nph3.fill_old(str1)); \ EXPECT_EQ(nph3, nph1); \ EXPECT_NE(nph3, nph2); #define LT(str1, str2) \ EXPECT_EQ(OB_SUCCESS, nph1.fill(str1)); \ EXPECT_EQ(OB_SUCCESS, nph2.fill(str2)); \ EXPECT_TRUE(nph1.number_ < nph2.number_); \ EXPECT_TRUE(nph1.number_ <= nph2.number_); \ EXPECT_EQ(OB_SUCCESS, nph3.fill_old(str2)); \ EXPECT_TRUE(nph3.number_ == nph2.number_); \ EXPECT_TRUE(nph1.number_ < nph3.number_); \ EXPECT_TRUE(nph1.number_ <= nph3.number_); \ EXPECT_EQ(OB_SUCCESS, nph3.fill_old(str1)); \ EXPECT_TRUE(nph3.number_ == nph1.number_); \ EXPECT_TRUE(nph3.number_ < nph2.number_); \ EXPECT_TRUE(nph3.number_ <= nph2.number_); \ #define LE(str1, str2) \ EXPECT_EQ(OB_SUCCESS, nph1.fill(str1)); \ EXPECT_EQ(OB_SUCCESS, nph2.fill(str2)); \ EXPECT_TRUE(nph1.number_ <= nph2.number_); \ EXPECT_EQ(OB_SUCCESS, nph3.fill_old(str2)); \ EXPECT_TRUE(nph3.number_ == nph2.number_); \ EXPECT_TRUE(nph1.number_ <= nph3.number_); \ EXPECT_EQ(OB_SUCCESS, nph3.fill_old(str1)); \ EXPECT_TRUE(nph3.number_ == nph1.number_); \ EXPECT_TRUE(nph3.number_ <= nph2.number_); #define GT(str1, str2) \ EXPECT_EQ(OB_SUCCESS, nph1.fill(str1)); \ EXPECT_EQ(OB_SUCCESS, nph2.fill(str2)); \ EXPECT_TRUE(nph1.number_ > nph2.number_); \ EXPECT_TRUE(nph1.number_ >= nph2.number_); \ EXPECT_EQ(OB_SUCCESS, nph3.fill_old(str2)); \ EXPECT_TRUE(nph3.number_ == nph2.number_); \ EXPECT_TRUE(nph1.number_ > nph3.number_); \ EXPECT_TRUE(nph1.number_ >= nph3.number_); \ EXPECT_EQ(OB_SUCCESS, nph3.fill_old(str1)); \ EXPECT_TRUE(nph3.number_ == nph1.number_); \ EXPECT_TRUE(nph3.number_ > nph2.number_); \ EXPECT_TRUE(nph2.number_ >= nph2.number_); #define GE(str1, str2) \ EXPECT_EQ(OB_SUCCESS, nph1.fill(str1)); \ EXPECT_EQ(OB_SUCCESS, nph2.fill(str2)); \ EXPECT_TRUE(nph1.number_ == nph2.number_); \ EXPECT_EQ(OB_SUCCESS, nph3.fill(str2)); \ EXPECT_TRUE(nph3.number_ == nph2.number_); \ EXPECT_TRUE(nph1.number_ >= nph3.number_); \ EXPECT_EQ(OB_SUCCESS, nph3.fill(str1)); \ EXPECT_TRUE(nph3.number_ == nph1.number_); \ EXPECT_TRUE(nph3.number_ >= nph2.number_); \ TEST(ObNumberParseHelper, compare) { uint32_t digits1[number::ObNumber::MAX_STORE_LEN]; uint32_t digits2[number::ObNumber::MAX_STORE_LEN]; uint32_t digits3[number::ObNumber::MAX_STORE_LEN]; number::ObNumber::Desc desc; ParseHelperTester nph1; ParseHelperTester nph2; ParseHelperTester nph3; desc.len_ = 0; desc.reserved_ = 0; desc.sign_ = number::ObNumber::POSITIVE; desc.exp_ = number::ObNumber::EXP_ZERO; nph1.number_.assign(desc.desc_, digits1); desc.len_ = 0; desc.reserved_ = 0; desc.sign_ = number::ObNumber::POSITIVE; desc.exp_ = number::ObNumber::EXP_ZERO; nph2.number_.assign(desc.desc_, digits2); desc.len_ = 0; desc.reserved_ = 0; desc.sign_ = number::ObNumber::POSITIVE; desc.exp_ = number::ObNumber::EXP_ZERO; nph3.number_.assign(desc.desc_, digits3); EQ("0.0", "0.0"); EQ("0000000000.0", "0.0000000000"); EQ("-0.0", "-0.0"); EQ("-0000000000.0", "-0.0000000000"); EQ("-0.0", "0.0"); EQ("-0000000000.0", "0.0000000000"); EQ("1.1", "1.1"); EQ("0000000001.1", "1.1000000000"); EQ("-1.1", "-1.1"); EQ("-0000000001.1", "-1.1000000000"); LT("0", "1.1"); GT("1.1", "0"); GT("0", "-1.1"); LT("-1.1", "0"); LT("-1.1", "-1"); LT("1.0000000001", "1.0000000002"); GT("-1.0000000001", "-1.0000000002"); LT("1000000000.00000000001", "1000000000.00000000002"); GT("-1000000000.00000000001", "-1000000000.00000000002"); EQ("1234567890.1234567890", "1234567890.1234567890"); EQ("-1234567890.1234567890", "-1234567890.1234567890"); LT("-1234567890.1234567890", "1234567890.1234567890"); GT("1234567890.1234567890", "-1234567890.1234567890"); LT("1.234", "1.2345"); GT("1.2345", "1.234"); GT("-1.234", "-1.2345"); LT("-1.2345", "-1.234"); LT("1.234567891", "1.2345678912"); GT("1.2345678912", "1.234567891"); GT("-1.234567891", "-1.2345678912"); LT("-1.2345678912", "-1.234567891"); LT("300000000000000000000", "400000000000000000000"); LT("3000000000000000000000000000000", "4000000000000000000000000000000"); LT("3000000000000000000000000000000000000000", "4000000000000000000000000000000000000000"); LT("0", "0.000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000001"); GT("0.000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000001", "0"); LT("0.000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000001", "0.0000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000011"); GT("0.0000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000011", "0.000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000001"); } TEST(ObNumberParseHelper, precision) { uint32_t digits[number::ObNumber::MAX_STORE_LEN]; number::ObNumber::Desc desc; desc.len_ = 0; desc.reserved_ = 0; desc.sign_ = number::ObNumber::POSITIVE; desc.exp_ = number::ObNumber::EXP_ZERO; ParseHelperTester nph; nph.number_.assign(desc.desc_, digits); int ret = OB_SUCCESS; nph.fill("123123456789.1234"); ret = nph.number_.check_and_round(15, 3); EXPECT_EQ(OB_SUCCESS, ret); nph.fill("1234123456789.1234"); ret = nph.number_.check_and_round(15, 3); EXPECT_EQ(OB_INTEGER_PRECISION_OVERFLOW, ret); nph.fill("0.0000000000001234"); ret = nph.number_.check_and_round(3, 15); EXPECT_EQ(OB_SUCCESS, ret); nph.fill("1.0000000000001234"); ret = nph.number_.check_and_round(3, 15); EXPECT_EQ(OB_INTEGER_PRECISION_OVERFLOW, ret); nph.fill("0.000000000001234"); ret = nph.number_.check_and_round(3, 15); EXPECT_EQ(OB_DECIMAL_PRECISION_OVERFLOW, ret); nph.fill("0.0000000019"); ret = nph.number_.check_and_round(18, 9); EXPECT_EQ(OB_SUCCESS, ret); } #define SCALE(res, str, p, s) \ do { \ nph.fill(str); \ ret = nph.number_.check_and_round(p, s); \ EXPECT_EQ(OB_SUCCESS, ret); \ ASSERT_STREQ(res, format(nph.number_, s)); \ \ nph.fill(negate(str).c_str()); \ ret = nph.number_.check_and_round(p, s); \ EXPECT_EQ(OB_SUCCESS, ret); \ ASSERT_STREQ(negate(res).c_str(), format(nph.number_, s)); \ } while (0) TEST(ObNumberParseHelper, scale) { uint32_t digits[number::ObNumber::MAX_STORE_LEN]; number::ObNumber::Desc desc; desc.len_ = 0; desc.reserved_ = 0; desc.sign_ = number::ObNumber::POSITIVE; desc.exp_ = number::ObNumber::EXP_ZERO; ParseHelperTester nph; nph.number_.assign(desc.desc_, digits); int ret = OB_SUCCESS; SCALE("1.123", "1.1234", 32, 3); SCALE("1.124", "1.1235", 32, 3); SCALE("1.124", "1.1236", 32, 3); SCALE("-1.123", "-1.1234", 32, 3); SCALE("-1.124", "-1.1235", 32, 3); SCALE("-1.124", "-1.1236", 32, 3); SCALE("0.123", "0.1234", 32, 3); SCALE("0.124", "0.1235", 32, 3); SCALE("0.124", "0.1236", 32, 3); SCALE("-0.123", "-0.1234", 32, 3); SCALE("-0.124", "-0.1235", 32, 3); SCALE("-0.124", "-0.1236", 32, 3); SCALE("1.000000000123", "1.0000000001234", 32, 12); SCALE("1.000000000124", "1.0000000001235", 32, 12); SCALE("1.000000000124", "1.0000000001236", 32, 12); SCALE("-1.000000000123", "-1.0000000001234", 32, 12); SCALE("-1.000000000124", "-1.0000000001235", 32, 12); SCALE("-1.000000000124", "-1.0000000001236", 32, 12); SCALE("0.000000000123", "0.0000000001234", 32, 12); SCALE("0.000000000000000000123", "0.0000000000000000001234", 32, 21); SCALE("0.000000000124", "0.0000000001235", 32, 12); SCALE("0.000000000124", "0.0000000001236", 32, 12); SCALE("-0.000000000123", "-0.0000000001234", 32, 12); SCALE("-0.000000000124", "-0.0000000001235", 32, 12); SCALE("-0.000000000124", "-0.0000000001236", 32, 12); SCALE("0.000000000000000123123", "0.0000000000000001231234", 32, 21); SCALE("0.000000000000000123124", "0.0000000000000001231235", 32, 21); SCALE("0.000000000000000123124", "0.0000000000000001231236", 32, 21); SCALE("-0.000000000000000123123", "-0.0000000000000001231234", 32, 21); SCALE("-0.000000000000000123124", "-0.0000000000000001231235", 32, 21); SCALE("-0.000000000000000123124", "-0.0000000000000001231236", 32, 21); SCALE("1.000000001123", "1.0000000011234", 32, 12); SCALE("1.000000001124", "1.0000000011235", 32, 12); SCALE("1.000000001124", "1.0000000011236", 32, 12); SCALE("0.000000001123", "0.0000000011234", 32, 12); SCALE("0.000000001124", "0.0000000011235", 32, 12); SCALE("0.000000001124", "0.0000000011236", 32, 12); SCALE("-0.000000001123", "-0.0000000011234", 32, 12); SCALE("-0.000000001124", "-0.0000000011235", 32, 12); SCALE("-0.000000001124", "-0.0000000011236", 32, 12); SCALE("1.000000009", "1.0000000094", 32, 9); SCALE("1.00000001", "1.0000000095", 32, 8); SCALE("1.00000001", "1.0000000096", 32, 8); SCALE("0.000000009", "0.0000000094", 32, 9); SCALE("0.00000001", "0.0000000095", 32, 8); SCALE("0.00000001", "0.0000000096", 32, 8); SCALE("-0.000000009", "-0.0000000094", 32, 9); SCALE("-0.00000001", "-0.0000000095", 32, 8); SCALE("-0.00000001", "-0.0000000096", 32, 8); SCALE("1.000000000000000009", "1.0000000000000000094", 32, 18); SCALE("1.00000000000000001", "1.0000000000000000095", 32, 17); SCALE("1.00000000000000001", "1.0000000000000000096", 32, 17); SCALE("0.000000000000000009", "0.0000000000000000094", 32, 18); SCALE("0.00000000000000001", "0.0000000000000000095", 32, 17); SCALE("0.00000000000000001", "0.0000000000000000096", 32, 17); SCALE("1", "1.000000000000000009", 32, 0); SCALE("1", "1.00000000000000001", 32, 0); SCALE("1", "1.00000000000000001", 32, 0); SCALE("-1", "-1.000000000000000009", 32, 0); SCALE("-1", "-1.00000000000000001", 32, 0); SCALE("-1", "-1.00000000000000001", 32, 0); SCALE("0", "-0.000000000000000009", 32, 0); SCALE("0", "-0.00000000000000001", 32, 0); SCALE("0", "-0.00000000000000001", 32, 0); SCALE("0", "-0.00009", 32, 0); SCALE("0", "-0.00009", 32, 0); SCALE("0", "-0.00009", 32, 0); SCALE("-1", "-0.500000001", 32, 0); SCALE("1", "0.500000001", 32, 0); SCALE("0", "-0.499999999", 32, 0); SCALE("0", "0.499999999", 32, 0); SCALE("1.0000000001", "1.00000000014234567891", 32, 10); SCALE("1.0000000002", "1.00000000015234567891", 32, 10); SCALE("1.0000000002", "1.00000000016234567891", 32, 10); SCALE("0.0000000001", "0.00000000014234567891", 32, 10); SCALE("0.0000000002", "0.00000000015234567891", 32, 10); SCALE("0.0000000002", "0.00000000016234567891", 32, 10); SCALE("1.000000001", "1.0000000014234567891", 32, 9); SCALE("1.000000002", "1.0000000015234567891", 32, 9); SCALE("1.000000002", "1.0000000016234567891", 32, 9); SCALE("0.000000001", "0.0000000014234567891", 32, 9); SCALE("0.000000002", "0.0000000015234567891", 32, 9); SCALE("0.000000002", "0.0000000016234567891", 32, 9); SCALE("1.00000001", "1.000000014234567891", 32, 8); SCALE("1.00000002", "1.000000015234567891", 32, 8); SCALE("1.00000002", "1.000000016234567891", 32, 8); SCALE("0.00000001", "0.000000014234567891", 32, 8); SCALE("0.00000002", "0.000000015234567891", 32, 8); SCALE("0.00000002", "0.000000016234567891", 32, 8); SCALE("1200", "1245.1234", 32, -2); SCALE("1300", "1256.1234", 32, -2); SCALE("1300", "1267.1234", 32, -2); SCALE("1200", "1245", 32, -2); SCALE("1300", "1256", 32, -2); SCALE("1300", "1267", 32, -2); SCALE("1234567891200", "1234567891245.1234", 32, -2); SCALE("1234567891300", "1234567891256.1234", 32, -2); SCALE("1234567891300", "1234567891267.1234", 32, -2); SCALE("1234567891200", "1234567891245", 32, -2); SCALE("1234567891300", "1234567891256", 32, -2); SCALE("1234567891300", "1234567891267", 32, -2); SCALE("1000000000", "1423456789.1234", 32, -9); SCALE("2000000000", "1523456789.1234", 32, -9); SCALE("2000000000", "1623456789.1234", 32, -9); SCALE("1000000000", "1423456789", 32, -9); SCALE("2000000000", "1523456789", 32, -9); SCALE("2000000000", "1623456789", 32, -9); SCALE("1000000000000000000", "1423456789123456789.1234", 32, -18); SCALE("2000000000000000000", "1523456789123456789.1234", 32, -18); SCALE("2000000000000000000", "1623456789123456789.1234", 32, -18); SCALE("-1000000000000000000", "-1423456789123456789.1234", 32, -18); SCALE("-2000000000000000000", "-1523456789123456789.1234", 32, -18); SCALE("-2000000000000000000", "-1623456789123456789.1234", 32, -18); SCALE("1000000000000000000", "1423456789123456789", 32, -18); SCALE("2000000000000000000", "1523456789123456789", 32, -18); SCALE("2000000000000000000", "1623456789123456789", 32, -18); SCALE("10000000000", "14234567891.1234", 32, -10); SCALE("20000000000", "15234567891.1234", 32, -10); SCALE("20000000000", "16234567891.1234", 32, -10); SCALE("10000000000", "14234567891", 32, -10); SCALE("20000000000", "15234567891", 32, -10); SCALE("20000000000", "16234567891", 32, -10); SCALE("-10000000000", "-14234567891", 32, -10); SCALE("-20000000000", "-15234567891", 32, -10); SCALE("-20000000000", "-16234567891", 32, -10); SCALE("1", "1.4", 32, 0); SCALE("2", "1.5", 32, 0); SCALE("2", "1.6", 32, 0); SCALE("1234567891", "1234567891.4", 32, 0); SCALE("1234567892", "1234567891.5", 32, 0); SCALE("1234567892", "1234567891.6", 32, 0); SCALE("-1234567891", "-1234567891.4", 32, 0); SCALE("-1234567892", "-1234567891.5", 32, 0); SCALE("-1234567892", "-1234567891.6", 32, 0); SCALE("0", "1.4", 32, -1); SCALE("0", "1.5", 32, -1); SCALE("0", "1.6", 32, -1); SCALE("0", "4234567891.4", 32, -10); SCALE("10000000000", "5234567891.5", 32, -10); SCALE("10000000000", "6234567891.6", 32, -10); SCALE("0.0000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000001", "0.00000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000014", 38, 127); SCALE("0.0000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000002", "0.00000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000015", 38, 127); SCALE("0.0000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000002", "0.00000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000016", 38, 127); SCALE("-0.0000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000001", "-0.00000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000014", 38, 127); SCALE("-0.0000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000002", "-0.00000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000015", 38, 127); SCALE("-0.0000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000002", "-0.00000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000016", 38, 127); SCALE("0", "0.00000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000004", 38, 0); SCALE("0.0000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000001", "0.00000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000005", 38, 127); SCALE("0.0000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000001", "0.00000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000006", 38, 127); SCALE("11111111111111111111111111111111111111000000000000000000000000000000000000000000000000000000000000000000000000000000000000", "11111111111111111111111111111111111111400000000000000000000000000000000000000000000000000000000000000000000000000000000000.0", 38, -84); SCALE("11111111111111111111111111111111111112000000000000000000000000000000000000000000000000000000000000000000000000000000000000", "11111111111111111111111111111111111111500000000000000000000000000000000000000000000000000000000000000000000000000000000000.0", 38, -84); SCALE("11111111111111111111111111111111111112000000000000000000000000000000000000000000000000000000000000000000000000000000000000", "11111111111111111111111111111111111111600000000000000000000000000000000000000000000000000000000000000000000000000000000000.0", 38, -84); SCALE("19999", "19999.4", 5, 0); SCALE("20000", "19999.5", 5, 0); SCALE("20000", "19999.6", 5, 0); SCALE("19999999999999", "19999999999999.4", 14, 0); SCALE("20000000000000", "19999999999999.5", 14, 0); SCALE("20000000000000", "19999999999999.6", 14, 0); nph.fill("99999.5"); ret = nph.number_.check_and_round(5, 0); EXPECT_EQ(OB_INTEGER_PRECISION_OVERFLOW, ret); nph.fill("99999999999999.5"); ret = nph.number_.check_and_round(14, 0); EXPECT_EQ(OB_INTEGER_PRECISION_OVERFLOW, ret); nph.fill("999999999.5"); ret = nph.number_.check_and_round(9, 0); EXPECT_EQ(OB_INTEGER_PRECISION_OVERFLOW, ret); nph.fill("0.0000000005"); ret = nph.number_.check_and_round(0, 9); EXPECT_EQ(OB_DECIMAL_PRECISION_OVERFLOW, ret); nph.fill("500000000"); ret = nph.number_.check_and_round(0, -8); EXPECT_EQ(OB_INTEGER_PRECISION_OVERFLOW, ret); SCALE("1000000000", "999999999.5", 10, 0); SCALE("1000000000000000000", "999999999999999999.5", 19, 0); SCALE("1000000000000000000000000000000000000000000000", "999999999999999999999999999999999999999999999", 46, -1); SCALE("1000000000", "500000000", 1, -9); SCALE("1", "0.9999999995", 10, 0); SCALE("1", "0.9999999999999999995", 10, 0); SCALE("1", "0.999999999999999999999999999999999999999999999", 10, 0); SCALE("0.000000001", "0.000000000999999999999999999999999999999999999", 11, 9); SCALE("1", "0.5", 10, 0); SCALE("1000000000", "999999999.5", 10, 0); SCALE("0", "0", 1, 0); //SCALE("1", "1", INT64_MAX, INT64_MAX); } TEST(ObNumberParseHelper, vector) { uint32_t digits[number::ObNumber::MAX_STORE_LEN]; number::ObNumber::Desc desc; desc.len_ = 0; desc.reserved_ = 0; desc.sign_ = number::ObNumber::POSITIVE; desc.exp_ = number::ObNumber::EXP_ZERO; ParseHelperTester nph; nph.number_.assign(desc.desc_, digits); int ret = OB_SUCCESS; ObCalcVector cv; ObCalcVector cv1; ret = nph.fill("0.00000000031415"); desc.desc_ = nph.number_.get_desc_value(); desc.exp_ = (0x7f) & (desc.exp_ + 3); cv1.init(desc.desc_, nph.number_.get_digits()); ObCalcVector cv2; ret = nph.fill("1000000000.31415"); desc = nph.number_.get_desc(); desc.exp_ = (0x7f) & (desc.exp_ + 3); cv2.init(desc.desc_, nph.number_.get_digits()); ObCalcVector cvs; cvs.ensure(std::max(cv1.size(), cv2.size()) + 1); poly_poly_add(cv1, cv2, cvs); desc.len_ = (0x7f) & cvs.size(); desc.exp_ = (0x7f) & (desc.exp_ - 3); ret = nph.from(desc.desc_, cvs); EXPECT_EQ(OB_SUCCESS, ret); EXPECT_STREQ("1000000000.31415000031415", format(nph.number_, 14)); cv.ensure(6); cv.set(0, 999999999); cv.set(1, 999999999); cv.set(2, 999999999); cv.set(3, 999999999); cv.set(4, 999999999); cv.set(5, 500000000); desc.exp_ = number::ObNumber::EXP_ZERO + 4; desc.sign_ = number::ObNumber::POSITIVE; ret = nph.from(desc.desc_, cv); EXPECT_EQ(OB_SUCCESS, ret); EXPECT_STREQ("1000000000000000000000000000000000000000000000", format(nph.number_, 0)); desc.exp_ = number::ObNumber::EXP_ZERO + 3; desc.sign_ = number::ObNumber::POSITIVE; ret = nph.from(desc.desc_, cv); EXPECT_EQ(OB_SUCCESS, ret); EXPECT_STREQ("1000000000000000000000000000000000000", format(nph.number_, 0)); desc.exp_ = number::ObNumber::EXP_ZERO; desc.sign_ = number::ObNumber::NEGATIVE; ret = nph.from(desc.desc_, cv); EXPECT_EQ(OB_SUCCESS, ret); EXPECT_STREQ("-1000000000", format(nph.number_, 0)); desc.exp_ = number::ObNumber::EXP_ZERO - 1; desc.sign_ = number::ObNumber::NEGATIVE; ret = nph.from(desc.desc_, cv); EXPECT_EQ(OB_SUCCESS, ret); EXPECT_STREQ("-1000000000000000000", format(nph.number_, 0)); cv.ensure(6); cv.set(0, 999999999); cv.set(1, 999999999); cv.set(2, 999999999); cv.set(3, 999999999); cv.set(4, 999999999); cv.set(5, 400000000); desc.exp_ = number::ObNumber::EXP_ZERO + 4; desc.sign_ = number::ObNumber::POSITIVE; ret = nph.from(desc.desc_, cv); EXPECT_EQ(OB_SUCCESS, ret); EXPECT_STREQ("999999999999999999999999999999999999999999999", format(nph.number_, 0)); desc.exp_ = number::ObNumber::EXP_ZERO + 3; desc.sign_ = number::ObNumber::POSITIVE; ret = nph.from(desc.desc_, cv); EXPECT_EQ(OB_SUCCESS, ret); EXPECT_STREQ("999999999999999999999999999999999999.999999999", format(nph.number_, 9)); desc.exp_ = number::ObNumber::EXP_ZERO; desc.sign_ = number::ObNumber::NEGATIVE; ret = nph.from(desc.desc_, cv); EXPECT_EQ(OB_SUCCESS, ret); EXPECT_STREQ("-999999999.999999999999999999999999999999999999", format(nph.number_, 36)); desc.exp_ = number::ObNumber::EXP_ZERO - 1; desc.sign_ = number::ObNumber::NEGATIVE; ret = nph.from(desc.desc_, cv); EXPECT_EQ(OB_SUCCESS, ret); EXPECT_STREQ("-999999999999999999.999999999999999999999999999", format(nph.number_, 27)); } #define ADD(n1, n2, res, scale) \ do { \ int ret = OB_SUCCESS; \ CharArena allocator; \ LimitedAllocator la; \ number::ObNumber augend; \ number::ObNumber addend; \ number::ObNumber sum; \ number::ObNumber result; \ \ ret = augend.from(n1, allocator); \ EXPECT_EQ(OB_SUCCESS, ret); \ ret = addend.from(n2, allocator); \ EXPECT_EQ(OB_SUCCESS, ret); \ ret = result.from(res, allocator); \ EXPECT_EQ(OB_SUCCESS, ret); \ ret = augend.add_v3(addend, sum, la); \ EXPECT_EQ(OB_SUCCESS, ret); \ EXPECT_STREQ(res, format(sum, scale)); \ if (scale > 0) {\ EXPECT_EQ(0, result.compare_v1(sum)); \ }\ \ ret = augend.from(negate(n1).c_str(), allocator); \ EXPECT_EQ(OB_SUCCESS, ret); \ ret = addend.from(negate(n2).c_str(), allocator); \ EXPECT_EQ(OB_SUCCESS, ret); \ ret = result.from(negate(res).c_str(), allocator); \ EXPECT_EQ(OB_SUCCESS, ret); \ ret = augend.add_v3(addend, sum, la); \ EXPECT_EQ(OB_SUCCESS, ret); \ EXPECT_STREQ(negate(res).c_str(), format(sum, scale)); \ if (scale > 0) {\ EXPECT_EQ(0, result.compare_v1(sum)); \ }\ } while (0) TEST(ObNumberCalc, add) { ADD("1.1", "1.2", "2.3", 1); ADD("9.9", "9.9", "19.8", 1); ADD("9999999999.9999999999", "9999999999.9999999999", "19999999999.9999999998", 10); ADD("1", "0.0000000002", "1.0000000002", 10); ADD("1", "0.9999999999999999999999999999999999994", "1.9999999999999999999999999999999999994", 37); ADD("1", "0.9999999999999999999999999999999999995", "1.9999999999999999999999999999999999995", 37); ADD("1", "0.99999999999999999999999999999999999955555555", "1.99999999999999999999999999999999999955555555", 44); ADD("1", "0.999999999999999999999999999999999999999999995", "2", 0); ADD("1", "0", "1", 0); ADD("0", "1", "1", 0); ADD("999999999", "1", "1000000000", 0); ADD("999999999", "999999999", "1999999998", 0); ADD("999999999999999999999999999999999999999999999000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000", "0.000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000001", "999999999999999999999999999999999999999999999000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000", 0); ADD("999999999999999999999999999999999999999999999000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000", "0.000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000999999999999999999999999999999999999999999999", "999999999999999999999999999999999999999999999000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000", 0); ADD("0.0000000000000000001234", "0.0000000001234", "0.0000000001234000001234", 22); ADD("0.000000000000000000000000000000000000000000001", "0.999999999999999999999999999999999999999999999", "1", 0); ADD("-1.1", "1.2", "0.1", 1); ADD("-1", "0.0000000002", "-0.9999999998", 10); ADD("-1", "0.9999999999999999999999999999999999994", "-0.0000000000000000000000000000000000006", 37); ADD("-1", "0.999999999999999999999999999999999999999999999", "-0.000000000000000000000000000000000000000000001", 45); ADD("-1", "0.0000000000000000000000000000000000000000000001", "-1", 0); ADD("-1", "0", "-1", 0); ADD("0", "-1", "-1", 0); ADD("-999999999999999999999999999999999999999999999000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000", "0.000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000001", "-999999999999999999999999999999999999999999999000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000", 0); ADD("-999999999999999999999999999999999999999999999000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000", "0.000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000999999999999999999999999999999999999999999999", "-999999999999999999999999999999999999999999999000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000", 0); ADD("-0.0000000000000000001234", "0.0000000001234", "0.0000000001233999998766", 22); ADD("-0.000000000000000000000000000000000000000000001", "0.999999999999999999999999999999999999999999999", "0.999999999999999999999999999999999999999999998", 45); ADD("999999999999999995", "5", "1000000000000000000", 0); ADD("0.999999999999999995", "0.000000000000000005", "1", 0); ADD("99999999999999999999999999999999999999999999999999999999999999999999.9999", "7.99365542543574543435457546878765465768765465476876465746876576576576577", "100000000000000000000000000000000000000000000000000000000000000000008", 0); ADD("99999999999999999999999999999999999999999999999999999999999999999999.9999", "-1", "99999999999999999999999999999999999999999999999999999999999999999999", 0); ADD("0.9902819038736083899663929849555723217376", "0.000000000000000000000000000000000000000000000000000000000025578569366733499189294448246024165831350", "1", 0); CharArena allocator; LimitedAllocator la; number::ObNumber augend; number::ObNumber addend; number::ObNumber sum; int ret = augend.from("500000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000", allocator); EXPECT_EQ(OB_SUCCESS, ret); ret = addend.from("500000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000", allocator); EXPECT_EQ(OB_SUCCESS, ret); ret = augend.add_v3(addend, sum, la); EXPECT_EQ(OB_INTEGER_PRECISION_OVERFLOW, ret); } #define NEGATE(v, scale) \ do { \ int ret = OB_SUCCESS; \ CharArena allocator; \ number::ObNumber orig; \ number::ObNumber res; \ ret = orig.from(v, allocator); \ EXPECT_EQ(OB_SUCCESS, ret); \ \ LimitedAllocator la1; \ ret = orig.negate(res, la1); \ EXPECT_EQ(OB_SUCCESS, ret); \ EXPECT_STREQ(negate(v).c_str(), format(res, scale)); \ \ LimitedAllocator la2; \ ret = res.negate(res, la2); \ EXPECT_EQ(OB_SUCCESS, ret); \ EXPECT_STREQ(format(orig, scale), format(res, scale)); \ } while (0) TEST(ObNumberCalc, negate) { NEGATE("0", 0); NEGATE("1", 0); NEGATE("999999999999999999999999999999999999999999999", 0); NEGATE("0.999999999999999999999999999999999999999999999", 45); NEGATE("999999999.999999999999999999999999999999999999", 36); } #define SUB(n1, n2, res, scale) \ do { \ int ret = OB_SUCCESS; \ CharArena allocator; \ LimitedAllocator la; \ number::ObNumber minuend; \ number::ObNumber subtrahend; \ number::ObNumber remainder; \ \ ret = minuend.from(n1, allocator); \ EXPECT_EQ(OB_SUCCESS, ret); \ ret = subtrahend.from(n2, allocator); \ EXPECT_EQ(OB_SUCCESS, ret); \ ret = minuend.sub_v3(subtrahend, remainder, la); \ EXPECT_EQ(OB_SUCCESS, ret); \ EXPECT_STREQ(res, format(remainder, scale)); \ \ ret = minuend.from(negate(n1).c_str(), allocator); \ EXPECT_EQ(OB_SUCCESS, ret); \ ret = subtrahend.from(negate(n2).c_str(), allocator); \ EXPECT_EQ(OB_SUCCESS, ret); \ ret = minuend.sub_v3(subtrahend, remainder, la); \ EXPECT_EQ(OB_SUCCESS, ret); \ EXPECT_STREQ(negate(res).c_str(), format(remainder, scale)); \ } while (0) TEST(ObNumberCalc, sub) { SUB("1", "0.999999999999999999999999999999999999940000000000000000000000000000000000000900000", "0.000000000000000000000000000000000000060000000", 45); SUB("1", "-1", "2", 0); SUB("1", "-1351298048", "1351298049", 0); SUB("3", "2", "1", 0); SUB("2", "3", "-1", 0); SUB("3", "0", "3", 0); SUB("0", "3", "-3", 0); SUB("3", "3", "0", 0); SUB("1", "0.999999999999999999999999999999999999999999999", "0.000000000000000000000000000000000000000000001", 45); SUB("0.999999999999999999999999999999999999999999999", "1", "-0.000000000000000000000000000000000000000000001", 45); SUB("999999999999999999999999999999999999999999999000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000", "0.000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000001", "999999999999999999999999999999999999999999999000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000", 0); SUB("999999999999999999999999999999999999999999999000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000", "0.000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000999999999999999999999999999999999999999999999", "999999999999999999999999999999999999999999999000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000", 0); SUB("-3", "2", "-5", 0); SUB("-2", "3", "-5", 0); SUB("-3", "0", "-3", 0); SUB("0", "-3", "3", 0); SUB("-3", "3", "-6", 0); SUB("-1", "0.999999999999999999999999999999999999999999999", "-2", 0); SUB("-0.999999999999999999999999999999999999999999999", "1", "-2", 0); SUB("-999999999999999999999999999999999999999999999000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000", "0.000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000001", "-999999999999999999999999999999999999999999999000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000", 0); SUB("-999999999999999999999999999999999999999999999000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000", "0.000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000999999999999999999999999999999999999999999999", "-999999999999999999999999999999999999999999999000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000", 0); SUB("1000000000000000005", "5", "1000000000000000000", 0); SUB("1.000000000000000005", "0.000000000000000005", "1", 0); SUB("99999999999999999999999999999999999999999999999999999999999999999999.9999", "7.99365542543574543435457546878765465768765465476876465746876576576576577", "99999999999999999999999999999999999999999999999999999999999999999992", 0); SUB("99999999999999999999999999999999999999999999999999999999999999999999.9999", "1", "99999999999999999999999999999999999999999999999999999999999999999999", 0); } #define MUL(n1, n2, res, scale) \ do { \ int ret = OB_SUCCESS; \ CharArena allocator; \ LimitedAllocator la; \ number::ObNumber multiplicand; \ number::ObNumber multiplier; \ number::ObNumber product; \ \ ret = multiplicand.from(n1, allocator); \ EXPECT_EQ(OB_SUCCESS, ret); \ ret = multiplier.from(n2, allocator); \ EXPECT_EQ(OB_SUCCESS, ret); \ ret = multiplicand.mul_v3(multiplier, product, la); \ EXPECT_EQ(OB_SUCCESS, ret); \ EXPECT_STREQ(res, format(product, scale)); \ \ ret = multiplicand.from(negate(n1).c_str(), allocator); \ EXPECT_EQ(OB_SUCCESS, ret); \ ret = multiplier.from(negate(n2).c_str(), allocator); \ EXPECT_EQ(OB_SUCCESS, ret); \ ret = multiplicand.mul_v3(multiplier, product, la); \ EXPECT_EQ(OB_SUCCESS, ret); \ EXPECT_STREQ(res, format(product, scale)); \ } while (0) TEST(ObNumberCalc, mul) { MUL("2", "3", "6", 0); MUL("2", "500000000", "1000000000", 0); MUL("999999999999999999999999999999999999999999999000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000", "0.000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000001", "0.999999999999999999999999999999999999999999999", 45); MUL("0.999999999999999999999999999999999999999999999", "2", "2", 0); MUL("0.999999999999999999999999999999999999999999999", "0.000000001", "0.000000000999999999999999999999999999999999999999999999", 54); MUL("-2", "3", "-6", 0); MUL("-2", "500000000", "-1000000000", 0); MUL("-999999999999999999999999999999999999999999999000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000", "0.000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000001", "-0.999999999999999999999999999999999999999999999", 45); MUL("-0.999999999999999999999999999999999999999999999", "2", "-2", 0); MUL("-0.999999999999999999999999999999999999999999999", "0.000000001", "-0.000000000999999999999999999999999999999999999999999999", 54); MUL("99999999999999999999999999999999999999999999999999999999999999999999.9999", "1.99365542543574543435457546878765465768765465476876465746876576576576577", "199365542543574543435457546878765465768765465476876465746876576576576.577000000000000000000000000000000000000000000000000000", 54); CharArena allocator; number::ObNumber multiplicand; number::ObNumber multiplier; number::ObNumber product; int ret = multiplicand.from("500000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000", allocator); EXPECT_EQ(OB_SUCCESS, ret); ret = multiplier.from("2", allocator); EXPECT_EQ(OB_SUCCESS, ret); ret = multiplicand.mul(multiplier, product, allocator); EXPECT_EQ(OB_INTEGER_PRECISION_OVERFLOW, ret); } #define DIV(n1, n2, res, scale) \ do { \ int ret = OB_SUCCESS; \ CharArena allocator; \ LimitedAllocator la; \ number::ObNumber dividend; \ number::ObNumber divisor; \ number::ObNumber quotient; \ \ ret = dividend.from(n1, allocator); \ EXPECT_EQ(OB_SUCCESS, ret); \ ret = divisor.from(n2, allocator); \ EXPECT_EQ(OB_SUCCESS, ret); \ ret = dividend.div_v3(divisor, quotient, la); \ EXPECT_EQ(OB_SUCCESS, ret); \ ASSERT_STREQ(res, format(quotient, scale)); \ \ ret = dividend.from(negate(n1).c_str(), allocator); \ EXPECT_EQ(OB_SUCCESS, ret); \ ret = divisor.from(negate(n2).c_str(), allocator); \ EXPECT_EQ(OB_SUCCESS, ret); \ ret = dividend.div_v3(divisor, quotient, la); \ EXPECT_EQ(OB_SUCCESS, ret); \ ASSERT_STREQ(res, format(quotient, scale)); \ } while (0) TEST(ObNumberCalc, div) { DIV("1", "3", "0.333333333333333333333333333333333333333333333", 45); DIV("1000000000", "3", "333333333.333333333333333333333333333333333333", 36); DIV("1000000000000000000", "3", "333333333333333333.333333333333333333333333333", 27); DIV("1000000000000000000", "3000000000", "333333333.333333333333333333333333333333333333", 36); DIV("1", "3000000000", "0.000000000333333333333333333333333333333333333333333333", 54); DIV("1", "3000000000000000000", "0.000000000000000000333333333333333333333333333333333333333333333", 63); DIV("1", "3000000000000000000000000000", "0.000000000000000000000000000333333333333333333333333333333333333333333333", 72); //代码中限制补零的个数为60为上限,否则报4002错误,所以下面的执行format后,返回错误码为4002 //DIV("1", // "300000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000", // "0.000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000003333333333333333333333333333333333333", 171); DIV("1", "0.000000002", "500000000", 0); DIV("1", "0.000000000000000002", "500000000000000000", 0); DIV("0.1", "0.000000000000000002", "50000000000000000", 0); DIV("0.0000000001", "0.000000000000000002", "50000000", 0); DIV("1", "0.000000003", "333333333.333333333333333333333333333333333333", 36); DIV("1", "0.000000000000000003", "333333333333333333.333333333333333333333333333", 27); DIV("0.1", "0.000000000000000003", "33333333333333333.3333333333333333333333333333", 28); DIV("0.0000000001", "0.000000000000000003", "33333333.3333333333333333333333333333333333333", 37); DIV("1", "7", "0.142857142857142857142857142857142857142857143", 45); DIV("-1", "7", "-0.142857142857142857142857142857142857142857143", 45); DIV("1", "-7", "-0.142857142857142857142857142857142857142857143", 45); DIV("111111111.222222222333333333444444444555555555", "333333333.444444444555555555", "0.333333333555555555703703704469135802868312755", 45); DIV("111111111222222222.333333333444444444555555555", "333333333.444444444555555555", "333333333.555555555703703704469135802868312755", 36); DIV("0.6", "2.166190378969060094174830575509116615300000000", "0.276983964948433490291384292515194359", 36); DIV("99999999999999999999999999999999999999999999999999999999999999999999.9999", "1.99365542543574543435457546878765465768765465476876465746876576576576577", "50159119135716942081585699790616486181246465997952174780397060972446.202400000000000000000000000000000000000000000", 45); DIV("1089999999", "2044381910", "0.533168481714847496376056272186442894126371916", 45); } #define POWER(n1, n2, res, scale) \ do { \ int ret = OB_SUCCESS; \ CharArena allocator; \ LimitedAllocator la; \ number::ObNumber base; \ number::ObNumber exponent; \ number::ObNumber value; \ \ ret = base.from(n1, allocator); \ EXPECT_EQ(OB_SUCCESS, ret); \ ret = exponent.from(n2, allocator); \ EXPECT_EQ(OB_SUCCESS, ret); \ ret = base.power(exponent, value, la); \ EXPECT_EQ(OB_SUCCESS, ret); \ ASSERT_STREQ(res, format(value, scale)); \ } while (0) TEST(ObNumberCalc, power) { POWER("2", "10", "1024", 0); POWER("2", "0.5", "1.414213562373095048801688724209698079", 36); POWER("2", "0.5", "1.414213562373095048801688724209698078569671875", 45); POWER("10", "-100", "0.0000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000001", 100); POWER("0.98", "50", "0.364169680087117065217373981453183629371404595", 45); POWER("0.98", "60", "0.297553142692120732008661162497860252541618135", 45); POWER("0.98", "70", "0.243122581497661847070621529790268289060887739", 45); } #define EXP(arg, res, scale) \ do { \ int ret = OB_SUCCESS; \ CharArena allocator; \ LimitedAllocator la; \ number::ObNumber exponent; \ number::ObNumber value; \ number::ObNumber power_with_e_value; \ number::ObNumber ln_value; \ number::ObNumber e; \ ret = e.from("2.7182818284590452353602874713526624977572470936999595749669", allocator); \ EXPECT_EQ(OB_SUCCESS, ret); \ \ ret = exponent.from(arg, allocator); \ EXPECT_EQ(OB_SUCCESS, ret); \ ret = exponent.e_power(value, la); \ EXPECT_EQ(OB_SUCCESS, ret); \ ASSERT_STREQ(res, format(value, scale)); \ ret = e.power(exponent, power_with_e_value, la); \ EXPECT_EQ(OB_SUCCESS, ret); \ ASSERT_STREQ(res, format(power_with_e_value, scale)); \ ret = value.ln(ln_value, la); \ EXPECT_EQ(OB_SUCCESS, ret); \ ASSERT_STREQ(format(ln_value, scale), format(exponent, scale)); \ } while (0) TEST(ObNumberCalc, exp) { EXP("1", "2.7182818285", 10); EXP("5", "148.4131591026", 10); EXP("1", "2.71828182845904523536028747135266249775724709", 44); EXP("5", "148.413159102576603421115580040552279623487668", 42); EXP("-1", "0.367879441171442321595523770161460867445811131", 45); EXP("-5", "0.00673794699908546709663604842314842424884958503", 47); EXP("100", "26881171418161354484126255515800135873611118.8", 1); EXP("-30", "0.00000000000009357622968840174604915832223378706744958", 53); } #define LN(arg, res, scale) \ do { \ int ret = OB_SUCCESS; \ CharArena allocator; \ LimitedAllocator la; \ number::ObNumber arg_num; \ number::ObNumber value; \ number::ObNumber exp_value; \ \ ret = arg_num.from(arg, allocator); \ EXPECT_EQ(OB_SUCCESS, ret); \ ret = arg_num.ln(value, la); \ EXPECT_EQ(OB_SUCCESS, ret); \ ASSERT_STREQ(res, format(value, scale)); \ ret = value.e_power(exp_value, la); \ EXPECT_EQ(OB_SUCCESS, ret); \ ASSERT_STREQ(format(exp_value, scale), format(arg_num, scale)); \ } while (0) TEST(ObNumberCalc, ln) { LN("0.52", "-0.653926467406664013148031224887056674245934764", 45); LN("1.00000001", "0.00000000999999995000000033333333083333335333333316667", 53); LN("1", "0.00000000000000000000000000000000000000000000", 44); LN("5", "1.60943791243410037460075933322618763952560135", 44); LN("5.5", "1.70474809223842523464471145650695273174620672", 44); // exp^ln_value get 9999999999999999999999999999999999999999.99999999999999999999999999786061 // LN("10000000000000000000000000000000000000000", "92.1034037197618273607196581873745683040440595", 43); // exp^ln_value get "999999999999999999999999999999.9999999999999999999999999999999999999247660" // LN("1000000000000000000000000000000", "69.0775527898213705205397436405309262280330447", 43); LN("100000000000000000000", "46.0517018598809136803598290936872841520220298", 43); } #define LOG(n1, n2, res, scale) \ do { \ int ret = OB_SUCCESS; \ CharArena allocator; \ LimitedAllocator la; \ number::ObNumber base; \ number::ObNumber x; \ number::ObNumber value; \ number::ObNumber power_base_value; \ \ ret = base.from(n1, allocator); \ EXPECT_EQ(OB_SUCCESS, ret); \ ret = x.from(n2, allocator); \ EXPECT_EQ(OB_SUCCESS, ret); \ ret = x.log(base, value, la); \ EXPECT_EQ(OB_SUCCESS, ret); \ ASSERT_STREQ(res, format(value, scale)); \ ret = base.power(value, power_base_value, la); \ EXPECT_EQ(OB_SUCCESS, ret); \ ASSERT_STREQ(format(power_base_value, scale), format(x, scale)); \ } while (0) TEST(ObNumberCalc, log) { LOG("2", "1024", "10", 0); LOG("10", "100000", "5", 0); LOG("7", "0.01", "-2.366589324909876653635857123293718296330889046", 45); LOG("0.2", "100", "-2.861353116146786101340213137527931264139583864", 45); LOG("0.2", "0.0001", "5.722706232293572202680426275055862528279167728", 45); } #define SQRT(arg, res, scale) \ do { \ int ret = OB_SUCCESS; \ CharArena allocator; \ LimitedAllocator la; \ number::ObNumber arg_num; \ number::ObNumber value; \ number::ObNumber power_point_five_value; \ number::ObNumber point_five; \ ret = point_five.from("0.5", allocator); \ EXPECT_EQ(OB_SUCCESS, ret); \ \ ret = arg_num.from(arg, allocator); \ EXPECT_EQ(OB_SUCCESS, ret); \ ret = arg_num.sqrt(value, la); \ EXPECT_EQ(OB_SUCCESS, ret); \ ASSERT_STREQ(res, format(value, scale)); \ ret = arg_num.power(point_five, power_point_five_value, la); \ EXPECT_EQ(OB_SUCCESS, ret); \ ASSERT_STREQ(format(power_point_five_value, scale), format(value, scale)); \ } while (0) TEST(ObNumberCalc, sqrt) { SQRT("0.001", "0.0316227766016837933199889354443271853371955514", 46); SQRT("0.233332333323333", "0.483044856429848552010281859139251460607774146", 45); SQRT("2", "1.41421356237309504880168872420969807856967188", 44); SQRT("5", "2.23606797749978969640917366873127623544061836", 44); SQRT("23332333.23332333", "4830.35539410127979042754285628563348297762838", 41); SQRT("1000000000000001", "31622776.6016838091313772362862198924845880631", 37); } #define REM(n1, n2, res, scale) \ do { \ int ret = OB_SUCCESS; \ CharArena allocator; \ number::ObNumber dividend; \ number::ObNumber divisor; \ number::ObNumber remainder; \ \ ret = dividend.from(n1, allocator); \ EXPECT_EQ(OB_SUCCESS, ret); \ ret = divisor.from(n2, allocator); \ EXPECT_EQ(OB_SUCCESS, ret); \ ret = dividend.rem(divisor, remainder, allocator); \ EXPECT_EQ(OB_SUCCESS, ret); \ EXPECT_STREQ(res, format(remainder, scale)); \ \ ret = dividend.rem(divisor.negate(), remainder, allocator); \ EXPECT_EQ(OB_SUCCESS, ret); \ EXPECT_STREQ(res, format(remainder, scale)); \ \ ret = dividend.negate().rem(divisor, remainder, allocator); \ EXPECT_EQ(OB_SUCCESS, ret); \ EXPECT_STREQ(res, format(remainder.negate(), scale)); \ \ ret = dividend.negate().rem(divisor.negate(), remainder, allocator); \ EXPECT_EQ(OB_SUCCESS, ret); \ EXPECT_STREQ(res, format(remainder.negate(), scale)); \ } while (0) TEST(ObNumberCalc, rem) { REM("1000000000", "3", "1", 0); REM("1000000000000000000", "3", "1", 0); REM("1000000000000000000", "3000000000", "1000000000", 0); REM("4000000000", "3000000000", "1000000000", 0); REM("4000000000000000000", "3000000000000000000", "1000000000000000000", 0); REM("4000000000000000000000000000", "3000000000000000000000000000", "1000000000000000000000000000", 0); REM("4000000000000000000000000000000", "3000000000000000000000000000000", "1000000000000000000000000000000", 0); REM("4000000000000000000000000000000000", "3000000000000000000000000000000000", "1000000000000000000000000000000000", 0); REM("400000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000", "300000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000", "100000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000", 0); REM("1.000000001", "0.000000002", "0.000000001", 9); REM("1.000000000000000001", "0.000000000000000002", "0.000000000000000001", 18); REM("1.000000000000000000001", "0.000000000000000000002", "0.000000000000000000001", 21); REM("1.00000000000000000000000000000000000000000001", "0.00000000000000000000000000000000000000000002", "0.00000000000000000000000000000000000000000001", 44); REM("1", "0.000000003", "0.000000001", 9); REM("1", "0.000000000000000003", "0.000000000000000001", 18); REM("0.1", "0.000000000000000003", "0.000000000000000001", 18); REM("0.0000000001", "0.000000000000000003", "0.000000000000000001", 18); REM("1", "7", "1", 0); REM("-1", "7", "-1", 0); REM("1", "-7", "1", 0); } #define INIT_NUMBER12 \ ObNumber num1;\ ObNumber num2;\ const uint32_t test_data = data[k][j];\ for (int64_t m = 0; m < i; ++m) {\ digits1[m] = data[k][j] == 0 ? 2 : data[k][j];\ digits2[m] = data[k][j] <= 1 ? 1 : data[k][j] - 1;\ }\ ObNumberDesc desc;\ desc.len_ = i;\ desc.exp_ = (ObNumber::EXP_ZERO) & 0x7f;\ desc.sign_ = ObNumber::POSITIVE;\ num1.assign(desc.desc_, digits1);\ num2.assign(desc.desc_, digits2);\ tmp +=(num1.is_zero() + num2.is_zero()); TEST(ObNumber, format_number_format_cmp) { const int64_t MAX_TEST_COUNT = 10000; const int64_t MAX_BUF_SIZE = 256; char buf_v1[MAX_BUF_SIZE]; char buf_v2[MAX_BUF_SIZE]; int64_t pos_v1 = 0; int64_t pos_v2 = 0; char buf_alloc[MAX_BUF_SIZE]; int64_t pos = 0; int64_t get_range_beg = 0; int64_t get_range_cost = 0; int64_t begin_value = 0; int64_t end_value = 0; ObDataBuffer allocator(buf_alloc, MAX_BUF_SIZE); number::ObNumber tmp_number; int64_t tmp_value = 0; tmp_number.from(tmp_value, allocator); EXPECT_EQ(OB_INVALID_ARGUMENT, tmp_number.format_v2(buf_v1, -1, pos_v1, -1)); for (int64_t j = 0; j < 3; ++j) { EXPECT_EQ(OB_SIZE_OVERFLOW, tmp_number.format_v2(buf_v1, j, pos_v1, -1)); } EXPECT_EQ(OB_SUCCESS, tmp_number.format_v2(buf_v1, 3, pos_v1, -1)); EXPECT_EQ(pos_v1, 1); EXPECT_EQ(0, strcmp(buf_v1, "0")); allocator.free(); pos_v1 = 0; tmp_value = -1; tmp_number.from(tmp_value, allocator); EXPECT_EQ(OB_INVALID_ARGUMENT, tmp_number.format_v2(buf_v1, -1, pos_v1, -1)); for (int64_t j = 0; j < 12; ++j) { EXPECT_EQ(OB_SIZE_OVERFLOW, tmp_number.format_v2(buf_v1, j, pos_v1, -1)); } EXPECT_EQ(OB_SUCCESS, tmp_number.format_v2(buf_v1, 12, pos_v1, -1)); EXPECT_EQ(pos_v1, 2); EXPECT_EQ(0, strcmp(buf_v1, "-1")); allocator.free(); pos_v1 = 0; tmp_value = -1000000000; tmp_number.from(tmp_value, allocator); EXPECT_EQ(OB_INVALID_ARGUMENT, tmp_number.format_v2(buf_v1, -1, pos_v1, -1)); for (int64_t j = 0; j < 21; ++j) { EXPECT_EQ(OB_SIZE_OVERFLOW, tmp_number.format_v2(buf_v1, j, pos_v1, -1)); } EXPECT_EQ(OB_SUCCESS, tmp_number.format_v2(buf_v1, 21, pos_v1, -1)); EXPECT_EQ(pos_v1, 11); EXPECT_EQ(0, strcmp(buf_v1, "-1000000000")); allocator.free(); pos_v1 = 0; tmp_number.from("-123.456", allocator); EXPECT_EQ(OB_INVALID_ARGUMENT, tmp_number.format_v2(buf_v1, -1, pos_v1, -1)); for (int64_t j = 0; j < 21; ++j) { EXPECT_EQ(OB_SIZE_OVERFLOW, tmp_number.format_v2(buf_v1, j, pos_v1, -1)); } EXPECT_EQ(OB_SUCCESS, tmp_number.format_v2(buf_v1, 21, pos_v1, -1)); EXPECT_EQ(pos_v1, 8); EXPECT_EQ(0, strcmp(buf_v1, "-123.456")); allocator.free(); pos_v1 = 0; for (int64_t j = 0; j < 20; ++j) { begin_value = pow(10, j); end_value = pow(10, j + 1) - 1; _OB_LOG(INFO, "test numer %ld, [%ld, %ld]", j + 1, begin_value, end_value); number::ObNumber num1; int64_t add_step = (end_value - begin_value) / MAX_TEST_COUNT; if (add_step < 17) { add_step = 17; } for (int64_t i = begin_value, k = 0; k < MAX_TEST_COUNT && i < end_value - begin_value; i += add_step, ++k) { num1.from(i * (1 == i%2 ? 1 : -1), allocator); pos_v1 = 0; pos_v2 = 0; EXPECT_EQ(OB_SUCCESS, num1.format_v1(buf_v1, MAX_BUF_SIZE, pos_v1, -1)); EXPECT_EQ(OB_SUCCESS, num1.format_v2(buf_v2, MAX_BUF_SIZE, pos_v2, -1)); _OB_LOG(INFO, "debug jianhua, j=%ld, i=%ld, k=%ld, l1=%ld, l2=%ld, v1=%.*s, v2=%.*s", j ,i, k, pos_v1, pos_v2, static_cast(pos_v1), buf_v1, static_cast(pos_v2), buf_v2); EXPECT_EQ(pos_v1, pos_v2); EXPECT_EQ(0, memcmp(buf_v1, buf_v2, pos_v1)); allocator.free(); if (i > end_value) { i -= (end_value - begin_value); } } } for (int64_t j = 0; j < 20; ++j) { begin_value = pow(10, j); end_value = pow(10, j + 1) - 1; _OB_LOG(INFO, "test numer %ld, [%ld, %ld]", j + 1, begin_value, end_value); number::ObNumber num1; int64_t add_step = (end_value - begin_value) / MAX_TEST_COUNT; if (add_step < 17) { add_step = 17; } for (int64_t i = begin_value, k = 0; k < MAX_TEST_COUNT && i < end_value - begin_value; i += add_step, ++k) { for (int64_t m = 0; m < j; ++m) { ObFastFormatInt ffi(i * (1 == i%2 ? 1 : -1)); *const_cast(ffi.ptr() + m) = '.'; { ObLogger::ObTraceLogGuard trace_log_guard; // prevent printing log num1.from_v1(ffi.ptr(), ffi.length(), allocator); } pos_v1 = 0; pos_v2 = 0; EXPECT_EQ(OB_SUCCESS, num1.format_v1(buf_v1, MAX_BUF_SIZE, pos_v1, -1)); EXPECT_EQ(OB_SUCCESS, num1.format_v2(buf_v2, MAX_BUF_SIZE, pos_v2, -1)); // _OB_LOG(INFO, "debug jianhua decimal, j=%ld, i=%ld, k=%ld, ffi=%.*s, l1=%ld, l2=%ld, v1=%.*s, v2=%.*s", j, i, k, ffi.length(), ffi.ptr(), pos_v1, pos_v2, pos_v1, buf_v1, pos_v2, buf_v2); EXPECT_EQ(pos_v1, pos_v2); EXPECT_EQ(0, memcmp(buf_v1, buf_v2, pos_v1)); allocator.free(); } char tmp_buf[MAX_BUF_SIZE]; snprintf(tmp_buf, MAX_BUF_SIZE, "%.17lf", (i * (1 == i%2 ? 1 : -1)) * pow(10, 20)); { ObLogger::ObTraceLogGuard trace_log_guard; // prevent printing log num1.from_v1(tmp_buf, strlen(tmp_buf), allocator); } pos_v1 = 0; pos_v2 = 0; EXPECT_EQ(OB_SUCCESS, num1.format_v1(buf_v1, MAX_BUF_SIZE, pos_v1, -1)); EXPECT_EQ(OB_SUCCESS, num1.format_v2(buf_v2, MAX_BUF_SIZE, pos_v2, -1)); _OB_LOG(INFO, "debug jianhua decimal, j=%ld, i=%ld, k=%ld, v1=%s", j, i, k, tmp_buf); EXPECT_EQ(pos_v1, pos_v2); EXPECT_EQ(0, memcmp(buf_v1, buf_v2, pos_v1)); allocator.free(); if (i > end_value) { i -= (end_value - begin_value); } } } //200 char const_number_str[] = "98765432109876543210987654321098765432109876543210987654321098765432109876543210987654321098765432109876543210987654321098765432109876543210987654321098765432109876543210987654321098765432109876543210"; ObString const_number_string(const_number_str); for (int64_t j = 1; j < 80; ++j) { number::ObNumber num1, num2, value_old,value_new; for (int64_t i = 0; i < 11; ++i) { allocator.free(); { ObLogger::ObTraceLogGuard trace_log_guard; // prevent printing log num1.from_v1(const_number_str + i, j, allocator); } pos_v1 = 0; pos_v2 = 0; EXPECT_EQ(OB_SUCCESS, num1.format_v1(buf_v1, MAX_BUF_SIZE, pos_v1, -1)); EXPECT_EQ(OB_SUCCESS, num1.format_v2(buf_v2, MAX_BUF_SIZE, pos_v2, -1)); // _OB_LOG(INFO, "debug jianhua decimal, j=%ld, i=%ld, ffi=%.*s, l1=%ld, l2=%ld, v1=%.*s, v2=%.*s", j, i, j, const_number_str + i, pos_v1, pos_v2, pos_v1, buf_v1, pos_v2, buf_v2); EXPECT_EQ(pos_v1, pos_v2); EXPECT_EQ(0, memcmp(buf_v1, buf_v2, pos_v1)); } } } TEST(ObNumber, number_format_perf) { const int64_t MAX_TEST_COUNT = 10000; const int64_t MAX_BUF_SIZE = 256; char buf[MAX_BUF_SIZE]; char buf_v2[MAX_BUF_SIZE]; char buf_alloc[MAX_BUF_SIZE]; int64_t pos = 0; int64_t get_range_beg = 0; int64_t get_range_cost = 0; int64_t base_get_range_cost = 0; int64_t begin_value = 0; int64_t end_value = 0; ObDataBuffer allocator(buf_alloc, MAX_BUF_SIZE); for (int64_t j = 0; j < 20; ++j) { begin_value = pow(10, j); end_value = pow(10, j + 1) - 1; _OB_LOG(INFO, "test numer %ld, [%ld, %ld]", j + 1, begin_value, end_value); get_range_beg = ObTimeUtility::current_time(); number::ObNumber num1; for (int64_t i = begin_value, k = 0; k < MAX_TEST_COUNT; ++i, ++k) { num1.from(i, allocator); pos = 0; allocator.free(); if (i > end_value) { i -= (end_value - begin_value); } } base_get_range_cost = ObTimeUtility::current_time() - get_range_beg; get_range_beg = ObTimeUtility::current_time(); for (int64_t i = begin_value, k = 0; k < MAX_TEST_COUNT; ++i, ++k) { num1.from(i, allocator); pos = 0; num1.format_v1(buf, MAX_BUF_SIZE, pos, -1); allocator.free(); if (i > end_value) { i -= (end_value - begin_value); } } get_range_cost = ObTimeUtility::current_time() - get_range_beg - base_get_range_cost; _OB_LOG(INFO, "format, cost time: %f us", (double)get_range_cost / MAX_TEST_COUNT); get_range_beg = ObTimeUtility::current_time(); for (int64_t i = begin_value, k = 0; k < MAX_TEST_COUNT; ++i, ++k) { num1.from(i, allocator); pos = 0; num1.format_v1(buf, MAX_BUF_SIZE, pos, -1); allocator.free(); if (i > end_value) { i -= (end_value - begin_value); } } get_range_cost = ObTimeUtility::current_time() - get_range_beg - base_get_range_cost; _OB_LOG(INFO, "format_v1, cost time: %f us", (double)get_range_cost / MAX_TEST_COUNT); get_range_beg = ObTimeUtility::current_time(); for (int64_t i = begin_value, k = 0; k < MAX_TEST_COUNT; ++i, ++k) { num1.from(i, allocator); pos = 0; num1.format_v2(buf, MAX_BUF_SIZE, pos, -1); allocator.free(); if (i > end_value) { i -= (end_value - begin_value); } } get_range_cost = ObTimeUtility::current_time() - get_range_beg - base_get_range_cost; _OB_LOG(INFO, "format_v2, cost time: %f us", (double)get_range_cost / MAX_TEST_COUNT); } for (int64_t j = 0; j < 20; ++j) { begin_value = pow(10, j); end_value = pow(10, j + 1) - 1; _OB_LOG(INFO, "test numer decimal %ld, [%ld, %ld]", j + 1, begin_value, end_value); get_range_beg = ObTimeUtility::current_time(); number::ObNumber num1; for (int64_t i = begin_value, k = 0; k < MAX_TEST_COUNT; ++i, ++k) { ObFastFormatInt ffi(i); *const_cast(ffi.ptr() + ffi.length() / 2) = '.'; num1.from_v1(ffi.ptr(), ffi.length(), allocator); pos = 0; allocator.free(); if (i > end_value) { i -= (end_value - begin_value); } } base_get_range_cost = ObTimeUtility::current_time() - get_range_beg; get_range_beg = ObTimeUtility::current_time(); for (int64_t i = begin_value, k = 0; k < MAX_TEST_COUNT; ++i, ++k) { ObFastFormatInt ffi(i); *const_cast(ffi.ptr() + ffi.length() / 2) = '.'; num1.from_v1(ffi.ptr(), ffi.length(), allocator); pos = 0; num1.format_v1(buf, MAX_BUF_SIZE, pos, -1); allocator.free(); if (i > end_value) { i -= (end_value - begin_value); } } get_range_cost = ObTimeUtility::current_time() - get_range_beg - base_get_range_cost; _OB_LOG(INFO, "format, cost time: %f us", (double)get_range_cost / MAX_TEST_COUNT); get_range_beg = ObTimeUtility::current_time(); for (int64_t i = begin_value, k = 0; k < MAX_TEST_COUNT; ++i, ++k) { ObFastFormatInt ffi(i); *const_cast(ffi.ptr() + ffi.length() / 2) = '.'; num1.from_v1(ffi.ptr(), ffi.length(), allocator); pos = 0; num1.format_v2(buf, MAX_BUF_SIZE, pos, -1); allocator.free(); if (i > end_value) { i -= (end_value - begin_value); } } get_range_cost = ObTimeUtility::current_time() - get_range_beg - base_get_range_cost; _OB_LOG(INFO, "format_v2, cost time: %f us", (double)get_range_cost / MAX_TEST_COUNT); } } TEST(ObNumber, DISABLED_number_format_from_perf_v2) { const int64_t MAX_TEST_COUNT = 10000; const int64_t MAX_TEST_COUNT2 = 40; const int64_t MAX_BUF_SIZE = 256; char buf_alloc[MAX_BUF_SIZE]; char buf_alloc2[MAX_BUF_SIZE]; int64_t tmp = 0; int64_t pos = 0; int64_t get_range_beg = 0; int64_t base_get_range_cost = 0; int64_t base_get_range_cost2 = 0; int64_t get_range_cost = 0; int64_t get_range_cost2 = 0; int64_t begin_value = 0; int64_t end_value = 0; ObDataBuffer allocator(buf_alloc, MAX_BUF_SIZE); const int64_t MAX_DOUBLE_STRICT_PRINT_SIZE2 = 512; char buf[MAX_DOUBLE_STRICT_PRINT_SIZE2] = {}; int64_t length = 0 ; enum TestType { TT_MIN = 0, TT_MAX, TT_RAND, TT_MAX_TYPE }; uint32_t data[TestType::TT_MAX_TYPE][MAX_TEST_COUNT]; for (int64_t i = 0; i < MAX_TEST_COUNT; ++i) { data[TestType::TT_MIN][i] = rand() % 100; data[TestType::TT_MAX][i] = rand() % 100 + ObNumber::BASE - 120; data[TestType::TT_RAND][i] = rand() % ObNumber::BASE; } uint32_t digits1[ObNumber::MAX_STORE_LEN]; uint32_t digits2[ObNumber::MAX_STORE_LEN]; for (int64_t k = 0; k < TestType::TT_MAX_TYPE; ++k) { for (int64_t i = 1; i <= ObNumber::MAX_STORE_LEN; ++i) { _OB_LOG(INFO, "test numer k=%ld, i=%ld", k, i); ObNumber value_v1; base_get_range_cost = 0; for (int64_t r = 1; r <= MAX_TEST_COUNT2; ++r) { get_range_beg = ObTimeUtility::current_time(); for (int64_t j = 0; j < MAX_TEST_COUNT; ++j) { INIT_NUMBER12; // EXPECT_EQ(OB_SUCCESS, num1.add_v1(num2, value_v1, allocator)); allocator.free(); } base_get_range_cost += (ObTimeUtility::current_time() - get_range_beg); } base_get_range_cost2 = base_get_range_cost / MAX_TEST_COUNT2; get_range_cost = 0; for (int64_t r = 1; r <= MAX_TEST_COUNT2; ++r) { get_range_beg = ObTimeUtility::current_time(); for (int64_t j = 0; j < MAX_TEST_COUNT; ++j) { INIT_NUMBER12; pos = 0; num1.format_v1(buf, MAX_BUF_SIZE, pos, -1); } get_range_cost += (ObTimeUtility::current_time() - get_range_beg); } get_range_cost2 = get_range_cost/MAX_TEST_COUNT2 - base_get_range_cost2; _OB_LOG(INFO, "format_v1, cost time: %f us", (double)get_range_cost2 / MAX_TEST_COUNT); get_range_cost = 0; for (int64_t r = 1; r <= MAX_TEST_COUNT2; ++r) { get_range_beg = ObTimeUtility::current_time(); for (int64_t j = 0; j < MAX_TEST_COUNT; ++j) { INIT_NUMBER12; pos = 0; num1.format_v2(buf, MAX_BUF_SIZE, pos, -1); } get_range_cost += (ObTimeUtility::current_time() - get_range_beg); } get_range_cost2 = get_range_cost/MAX_TEST_COUNT2 - base_get_range_cost2; _OB_LOG(INFO, "format_v2, cost time: %f us", (double)get_range_cost2 / MAX_TEST_COUNT); base_get_range_cost2 = get_range_cost2; get_range_cost = 0; for (int64_t r = 1; r <= MAX_TEST_COUNT2; ++r) { get_range_beg = ObTimeUtility::current_time(); for (int64_t j = 0; j < MAX_TEST_COUNT; ++j) { INIT_NUMBER12; pos = 0; num1.format_v2(buf, MAX_BUF_SIZE, pos, -1); num1.from_v1(buf, pos, allocator); allocator.free(); } get_range_cost += (ObTimeUtility::current_time() - get_range_beg); } get_range_cost2 = get_range_cost/MAX_TEST_COUNT2 - base_get_range_cost2; _OB_LOG(INFO, "from_v1, cost time: %f us", (double)get_range_cost2 / MAX_TEST_COUNT); get_range_cost = 0; for (int64_t r = 1; r <= MAX_TEST_COUNT2; ++r) { get_range_beg = ObTimeUtility::current_time(); for (int64_t j = 0; j < MAX_TEST_COUNT; ++j) { INIT_NUMBER12; pos = 0; num1.format_v2(buf, MAX_BUF_SIZE, pos, -1); num1.from_v2(buf, pos, allocator); allocator.free(); } get_range_cost += (ObTimeUtility::current_time() - get_range_beg); } get_range_cost2 = get_range_cost/MAX_TEST_COUNT2 - base_get_range_cost2; _OB_LOG(INFO, "from_v2, cost time: %f us", (double)get_range_cost2 / MAX_TEST_COUNT); get_range_cost = 0; for (int64_t r = 1; r <= MAX_TEST_COUNT2; ++r) { get_range_beg = ObTimeUtility::current_time(); for (int64_t j = 0; j < MAX_TEST_COUNT; ++j) { INIT_NUMBER12; pos = 0; num1.format_v2(buf, MAX_BUF_SIZE, pos, -1); num1.from_v3(buf, pos, allocator); allocator.free(); } get_range_cost += (ObTimeUtility::current_time() - get_range_beg); } get_range_cost2 = get_range_cost/MAX_TEST_COUNT2 - base_get_range_cost2; _OB_LOG(INFO, "from_v3, cost time: %f us", (double)get_range_cost2 / MAX_TEST_COUNT); } } } TEST(ObNumber, format_number_from_cmp) { const int64_t MAX_TEST_COUNT = 1000; const int64_t MAX_BUF_SIZE = 256; char buf_v1[MAX_BUF_SIZE]; char buf_v2[MAX_BUF_SIZE]; int64_t pos_v1 = 0; int64_t pos_v2 = 0; char buf_alloc[MAX_BUF_SIZE]; int64_t pos = 0; int64_t get_range_beg = 0; int64_t get_range_cost = 0; int64_t begin_value = 0; int64_t end_value = 0; ObDataBuffer allocator(buf_alloc, MAX_BUF_SIZE); for (int64_t j = 0; j < 20; ++j) { begin_value = pow(10, j); end_value = pow(10, j + 1) - 1; _OB_LOG(INFO, "test numer %ld, [%ld, %ld]", j + 1, begin_value, end_value); number::ObNumber num1; number::ObNumber num2; number::ObNumber num3; int64_t add_step = (end_value - begin_value) / MAX_TEST_COUNT; if (add_step < 17) { add_step = 17; } for (int64_t i = begin_value, k = 0; k < MAX_TEST_COUNT && i < end_value - begin_value; i += add_step, ++k) { int64_t tmp_int = (i * (1 == i%2 ? 1 : -1)); ObFastFormatInt ffi(tmp_int); EXPECT_EQ(OB_SUCCESS, num1.from(tmp_int, allocator)); EXPECT_EQ(OB_SUCCESS, num2.from_v2(ffi.ptr(), ffi.length(), allocator)); EXPECT_EQ(OB_SUCCESS, num3.from_v3(ffi.ptr(), ffi.length(), allocator)); OB_LOG(INFO, "debug jianhua", K(j), K(i), K(k), "str", ffi.str(), K(num1), K(num2), K(num3)); ASSERT_EQ(0, num1.compare_v1(num2)); ASSERT_EQ(0, num1.compare_v1(num3)); allocator.free(); if (i > end_value) { i -= (end_value - begin_value); } } } for (int64_t j = 0; j < 20; ++j) { begin_value = pow(10, j); end_value = pow(10, j + 1) - 1; _OB_LOG(INFO, "test numer %ld, [%ld, %ld]", j + 1, begin_value, end_value); number::ObNumber num1; number::ObNumber num2; int64_t add_step = (end_value - begin_value) / MAX_TEST_COUNT; if (add_step < 17) { add_step = 17; } for (int64_t i = begin_value, k = 0; k < MAX_TEST_COUNT && i < end_value - begin_value; i += add_step, ++k) { ObFastFormatInt ffi(i * (1 == i%2 ? 1 : -1)); EXPECT_EQ(OB_SUCCESS, num1.from_v1(ffi.ptr(), ffi.length(), allocator)); EXPECT_EQ(OB_SUCCESS, num2.from_v2(ffi.ptr(), ffi.length(), allocator)); OB_LOG(INFO, "debug jianhua", K(j), K(i), K(k), "str", ffi.str(), K(num1), K(num2)); ASSERT_EQ(0, num1.compare_v1(num2)); allocator.free(); if (i > end_value) { i -= (end_value - begin_value); } } } for (int64_t j = 0; j < 20; ++j) { begin_value = pow(10, j); end_value = pow(10, j + 1) - 1; _OB_LOG(INFO, "test numer %ld, [%ld, %ld]", j + 1, begin_value, end_value); number::ObNumber num1; number::ObNumber num2; int64_t add_step = (end_value - begin_value) / MAX_TEST_COUNT; if (add_step < 17) { add_step = 17; } for (int64_t i = begin_value, k = 0; k < MAX_TEST_COUNT && i < end_value - begin_value; i += add_step, ++k) { for (int64_t m = 0; m < j; ++m) { ObFastFormatInt ffi(i * (1 == i%2 ? 1 : -1)); *const_cast(ffi.ptr() + m) = '.'; EXPECT_EQ(OB_SUCCESS, num1.from_v1(ffi.ptr(), ffi.length(), allocator)); EXPECT_EQ(OB_SUCCESS, num2.from_v2(ffi.ptr(), ffi.length(), allocator)); OB_LOG(INFO, "debug jianhua decimal", K(j), K(i), K(k), K(m), "str", ffi.str(), K(num1), K(num2)); ASSERT_EQ(0, num1.compare_v1(num2)); allocator.free(); } char tmp_buf[MAX_BUF_SIZE]; snprintf(tmp_buf, MAX_BUF_SIZE, "%.17lf", (i * (1 == i%2 ? 1 : -1)) * pow(10, 20)); ObString ffi(strlen(tmp_buf), tmp_buf); EXPECT_EQ(OB_SUCCESS, num1.from_v1(ffi.ptr(), ffi.length(), allocator)); EXPECT_EQ(OB_SUCCESS, num2.from_v2(ffi.ptr(), ffi.length(), allocator)); OB_LOG(INFO, "debug jianhua decimal", K(j), K(i), K(k), "str", tmp_buf); ASSERT_EQ(0, num1.compare_v1(num2)); allocator.free(); if (i > end_value) { i -= (end_value - begin_value); } } } //200 char const_number_str[] = "98765432109876543210987654321098765432109876543210987654321098765432109876543210987654321098765432109876543210987654321098765432109876543210987654321098765432109876543210987654321098765432109876543210"; ObString const_number_string(const_number_str); for (int64_t j = 1; j < 80; ++j) { number::ObNumber num1, num2, value_old,value_new; for (int64_t i = 0; i < 11; ++i) { _OB_LOG(INFO, "debug jianhua decimal, j=%ld, i=%ld, ffi=%.*s", j, i, static_cast(j), const_number_str + i); allocator.free(); num1.set_zero(); num2.set_zero(); ASSERT_EQ(num1.from_v1(const_number_str + i, j, allocator), num2.from_v2(const_number_str + i, j, allocator)); ASSERT_EQ(0, num1.compare_v1(num2)); allocator.free(); ASSERT_EQ(num1.from_v1(const_number_str + i, j, allocator), num2.from_v3(const_number_str + i, j, allocator)); ASSERT_EQ(0, num1.compare_v1(num2)); } } } TEST(ObNumber, format_number_from_perf) { const int64_t MAX_TEST_COUNT = 10000; const int64_t MAX_BUF_SIZE = 256; char buf[MAX_BUF_SIZE]; char buf_v2[MAX_BUF_SIZE]; char buf_alloc[MAX_BUF_SIZE]; int64_t pos = 0; int64_t get_range_beg = 0; int64_t get_range_cost = 0; int64_t base_get_range_cost = 0; int64_t begin_value = 0; int64_t end_value = 0; ObDataBuffer allocator(buf_alloc, MAX_BUF_SIZE); for (int64_t j = 0; j < 21; ++j) { begin_value = pow(10, j); end_value = pow(10, j + 1) - 1; _OB_LOG(INFO, "test numer %ld, [%ld, %ld]", j + 1, begin_value, end_value); get_range_beg = ObTimeUtility::current_time(); number::ObNumber num1; for (int64_t i = begin_value, k = 0; k < MAX_TEST_COUNT; ++i, ++k) { allocator.free(); if (i > end_value) { i -= (end_value - begin_value); } } base_get_range_cost = ObTimeUtility::current_time() - get_range_beg; get_range_beg = ObTimeUtility::current_time(); for (int64_t i = begin_value, k = 0; k < MAX_TEST_COUNT; ++i, ++k) { ObFastFormatInt ffi(i); num1.from_v1(ffi.ptr(), ffi.length(), allocator); allocator.free(); if (i > end_value) { i -= (end_value - begin_value); } } get_range_cost = ObTimeUtility::current_time() - get_range_beg - base_get_range_cost; _OB_LOG(INFO, "from, cost time: %f us", (double)get_range_cost / MAX_TEST_COUNT); get_range_beg = ObTimeUtility::current_time(); for (int64_t i = begin_value, k = 0; k < MAX_TEST_COUNT; ++i, ++k) { ObFastFormatInt ffi(i); num1.from_v2(ffi.ptr(), ffi.length(), allocator); allocator.free(); if (i > end_value) { i -= (end_value - begin_value); } } get_range_cost = ObTimeUtility::current_time() - get_range_beg - base_get_range_cost; _OB_LOG(INFO, "from_v2, cost time: %f us", (double)get_range_cost / MAX_TEST_COUNT); get_range_beg = ObTimeUtility::current_time(); for (int64_t i = begin_value, k = 0; k < MAX_TEST_COUNT; ++i, ++k) { ObFastFormatInt ffi(i); num1.from_v3(ffi.ptr(), ffi.length(), allocator); allocator.free(); if (i > end_value) { i -= (end_value - begin_value); } } get_range_cost = ObTimeUtility::current_time() - get_range_beg - base_get_range_cost; _OB_LOG(INFO, "from_v3, cost time: %f us", (double)get_range_cost / MAX_TEST_COUNT); } for (int64_t j = 0; j < 21; ++j) { begin_value = pow(10, j); end_value = pow(10, j + 1) - 1; _OB_LOG(INFO, "test numer decimal %ld, [%ld, %ld]", j + 1, begin_value, end_value); get_range_beg = ObTimeUtility::current_time(); number::ObNumber num1; for (int64_t i = begin_value, k = 0; k < MAX_TEST_COUNT; ++i, ++k) { allocator.free(); if (i > end_value) { i -= (end_value - begin_value); } } base_get_range_cost = ObTimeUtility::current_time() - get_range_beg; get_range_beg = ObTimeUtility::current_time(); for (int64_t i = begin_value, k = 0; k < MAX_TEST_COUNT; ++i, ++k) { ObFastFormatInt ffi(i); *const_cast(ffi.ptr() + ffi.length() / 2) = '.'; num1.from(ffi.ptr(), ffi.length(), allocator); allocator.free(); if (i > end_value) { i -= (end_value - begin_value); } } get_range_cost = ObTimeUtility::current_time() - get_range_beg - base_get_range_cost; _OB_LOG(INFO, "from, cost time: %f us", (double)get_range_cost / MAX_TEST_COUNT); get_range_beg = ObTimeUtility::current_time(); for (int64_t i = begin_value, k = 0; k < MAX_TEST_COUNT; ++i, ++k) { ObFastFormatInt ffi(i); *const_cast(ffi.ptr() + ffi.length() / 2) = '.'; num1.from_v2(ffi.ptr(), ffi.length(), allocator); allocator.free(); if (i > end_value) { i -= (end_value - begin_value); } } get_range_cost = ObTimeUtility::current_time() - get_range_beg - base_get_range_cost; _OB_LOG(INFO, "from_v2, cost time: %f us", (double)get_range_cost / MAX_TEST_COUNT); } } TEST(ObNumber, format_number_round_cmp) { const int64_t MAX_TEST_COUNT = 10000; const int64_t MAX_BUF_SIZE = 256; char buf_alloc[MAX_BUF_SIZE]; int64_t pos = 0; int64_t get_range_beg = 0; int64_t get_range_cost = 0; int64_t begin_value = 0; int64_t end_value = 0; ObDataBuffer allocator(buf_alloc, MAX_BUF_SIZE); for (int64_t j = 0; j < 20; ++j) { begin_value = pow(10, j); end_value = pow(10, j + 1) - 1; _OB_LOG(INFO, "test numer %ld, [%ld, %ld]", j + 1, begin_value, end_value); number::ObNumber num1; number::ObNumber num2; int64_t add_step = (end_value - begin_value) / MAX_TEST_COUNT; if (add_step < 17) { add_step = 17; } for (int64_t i = begin_value, k = 0; k < MAX_TEST_COUNT && i < end_value - begin_value; i += add_step, ++k) { num1.from(i * (1 == i%2 ? 1 : -1), allocator); num2.from(i * (1 == i%2 ? 1 : -1), allocator); _OB_LOG(INFO, "debug jianhua, j=%ld, i=%ld, k=%ld", j ,i, k); for (int64_t z = 0; z < j + 2; z++) { EXPECT_EQ(OB_SUCCESS, num1.round_v1(z)); EXPECT_EQ(OB_SUCCESS, num2.round_v2(z)); EXPECT_EQ(num1, num2); EXPECT_EQ(OB_SUCCESS, num2.round_v3(z)); EXPECT_EQ(num1, num2); } allocator.free(); if (i > end_value) { i -= (end_value - begin_value); } } } for (int64_t j = 0; j < 20; ++j) { begin_value = pow(10, j); end_value = pow(10, j + 1) - 1; _OB_LOG(INFO, "test numer %ld, [%ld, %ld]", j + 1, begin_value, end_value); number::ObNumber num1; number::ObNumber num2; int64_t add_step = (end_value - begin_value) / MAX_TEST_COUNT; if (add_step < 17) { add_step = 17; } for (int64_t i = begin_value, k = 0; k < MAX_TEST_COUNT && i < end_value - begin_value; i += add_step, ++k) { for (int64_t m = 0; m < j; ++m) { ObFastFormatInt ffi(i * (1 == i%2 ? 1 : -1)); *const_cast(ffi.ptr() + m) = '.'; { ObLogger::ObTraceLogGuard trace_log_guard; // prevent printing log num1.from_v1(ffi.ptr(), ffi.length(), allocator); num2.from_v1(ffi.ptr(), ffi.length(), allocator); } _OB_LOG(INFO, "debug jianhua decimal, j=%ld, i=%ld, k=%ld, ffi=%.*s", j, i, k, static_cast(ffi.length()), ffi.ptr()); for (int64_t z = -(j+2); z < j + 2; z++) { EXPECT_EQ(OB_SUCCESS, num1.round_v1(z)); EXPECT_EQ(OB_SUCCESS, num2.round_v2(z)); ASSERT_EQ(num1, num2); EXPECT_EQ(OB_SUCCESS, num2.round_v3(z)); ASSERT_EQ(num1, num2); } allocator.free(); } if (i > end_value) { i -= (end_value - begin_value); } } } for (int64_t j = 0; j < 20; ++j) { begin_value = pow(10, j); end_value = pow(10, j + 1) - 1; _OB_LOG(INFO, "test numer %ld, [%ld, %ld]", j + 1, begin_value, end_value); number::ObNumber num1; number::ObNumber num2; int64_t add_step = (end_value - begin_value) / MAX_TEST_COUNT; if (add_step < 17) { add_step = 17; } for (int64_t i = begin_value, k = 0; k < MAX_TEST_COUNT && i < end_value - begin_value; i += add_step, ++k) { char tmp_buf[MAX_BUF_SIZE]; snprintf(tmp_buf, MAX_BUF_SIZE, "%.17lf", (i * (1 == i%2 ? 1 : -1)) * pow(10, 20)); { ObLogger::ObTraceLogGuard trace_log_guard; num1.from(tmp_buf, allocator); num2.from(tmp_buf, allocator); } _OB_LOG(INFO, "debug jianhua decimal, j=%ld, i=%ld, k=%ld, str=%s", j, i, k, tmp_buf); for (int64_t z = -(j+2); z < j + 2; z++) { EXPECT_EQ(OB_SUCCESS, num1.round_v1(z)); EXPECT_EQ(OB_SUCCESS, num2.round_v2(z)); EXPECT_EQ(num1, num2); EXPECT_EQ(OB_SUCCESS, num2.round_v3(z)); EXPECT_EQ(num1, num2); } allocator.free(); if (i > end_value) { i -= (end_value - begin_value); } } } //200 char const_number_str[] = "98765432109876543210987654321098765432109876543210987654321098765432109876543210987654321098765432109876543210987654321098765432109876543210987654321098765432109876543210987654321098765432109876543210"; ObString const_number_string(const_number_str); for (int64_t j = 1; j < 80; ++j) { number::ObNumber num1, num2, value_old,value_new; for (int64_t i = 0; i < 11; ++i) { _OB_LOG(INFO, "debug jianhua decimal, j=%ld, i=%ld, ffi=%.*s", j, i, static_cast(j), const_number_str + i); allocator.free(); { ObLogger::ObTraceLogGuard trace_log_guard; num1.from_v1(const_number_str + i, j, allocator); num2.from_v1(const_number_str + i, j, allocator); } for (int64_t z = -(j+2); z < j + 2; z++) { num1.set_zero(); num2.set_zero(); EXPECT_EQ(num1.round_v1(z), num2.round_v2(z)); EXPECT_EQ(num1, num2); num1.set_zero(); num2.set_zero(); EXPECT_EQ(num1.round_v1(z), num2.round_v3(z)); EXPECT_EQ(num1, num2); } } } } TEST(ObNumber, DISABLED_format_number_round_perf) { const int64_t MAX_TEST_COUNT = 10000; const int64_t MAX_BUF_SIZE = 256; char buf_alloc[MAX_BUF_SIZE]; int64_t pos = 0; int64_t get_range_beg = 0; int64_t get_range_cost = 0; int64_t base_get_range_cost = 0; int64_t begin_value = 0; int64_t end_value = 0; ObDataBuffer allocator(buf_alloc, MAX_BUF_SIZE); for (int64_t j = 0; j < 20; ++j) { begin_value = pow(10, j); end_value = pow(10, j + 1) - 1; _OB_LOG(INFO, "test numer %ld, [%ld, %ld]", j + 1, begin_value, end_value); get_range_beg = ObTimeUtility::current_time(); number::ObNumber num1; for (int64_t i = begin_value, k = 0; k < MAX_TEST_COUNT; ++i, ++k) { num1.from(i, allocator); for (int64_t z = -(j+2); z < j + 2; z++) { //nothing } allocator.free(); if (i > end_value) { i -= (end_value - begin_value); } } base_get_range_cost = ObTimeUtility::current_time() - get_range_beg; get_range_beg = ObTimeUtility::current_time(); for (int64_t i = begin_value, k = 0; k < MAX_TEST_COUNT; ++i, ++k) { num1.from(i, allocator); for (int64_t z = -(j+2); z < j + 2; z++) { num1.round_v1(z); } allocator.free(); if (i > end_value) { i -= (end_value - begin_value); } } get_range_cost = ObTimeUtility::current_time() - get_range_beg - base_get_range_cost; _OB_LOG(INFO, "round_old, cost time: %f us", (double)get_range_cost / MAX_TEST_COUNT); get_range_beg = ObTimeUtility::current_time(); for (int64_t i = begin_value, k = 0; k < MAX_TEST_COUNT; ++i, ++k) { num1.from(i, allocator); pos = 0; for (int64_t z = -(j+2); z < j + 2; z++) { num1.round_v2(z); } allocator.free(); if (i > end_value) { i -= (end_value - begin_value); } } get_range_cost = ObTimeUtility::current_time() - get_range_beg - base_get_range_cost; _OB_LOG(INFO, "round_v2, cost time: %f us", (double)get_range_cost / MAX_TEST_COUNT); get_range_beg = ObTimeUtility::current_time(); for (int64_t i = begin_value, k = 0; k < MAX_TEST_COUNT; ++i, ++k) { num1.from(i, allocator); pos = 0; for (int64_t z = -(j+2); z < j + 2; z++) { num1.round_v3(z); } allocator.free(); if (i > end_value) { i -= (end_value - begin_value); } } get_range_cost = ObTimeUtility::current_time() - get_range_beg - base_get_range_cost; _OB_LOG(INFO, "round_v3, cost time: %f us", (double)get_range_cost / MAX_TEST_COUNT); } for (int64_t j = 0; j < 20; ++j) { begin_value = pow(10, j); end_value = pow(10, j + 1) - 1; _OB_LOG(INFO, "test numer decimal %ld, [%ld, %ld]", j + 1, begin_value, end_value); get_range_beg = ObTimeUtility::current_time(); number::ObNumber num1; for (int64_t i = begin_value, k = 0; k < MAX_TEST_COUNT; ++i, ++k) { ObFastFormatInt ffi(i); *const_cast(ffi.ptr() + ffi.length() / 2) = '.'; num1.from_v1(ffi.ptr(), ffi.length(), allocator); for (int64_t z = -(j+2); z < j + 2; z++) { // } allocator.free(); if (i > end_value) { i -= (end_value - begin_value); } } base_get_range_cost = ObTimeUtility::current_time() - get_range_beg; get_range_beg = ObTimeUtility::current_time(); for (int64_t i = begin_value, k = 0; k < MAX_TEST_COUNT; ++i, ++k) { ObFastFormatInt ffi(i); *const_cast(ffi.ptr() + ffi.length() / 2) = '.'; num1.from_v1(ffi.ptr(), ffi.length(), allocator); for (int64_t z = -(j+2); z < j + 2; z++) { num1.round_v1(z); } allocator.free(); if (i > end_value) { i -= (end_value - begin_value); } } get_range_cost = ObTimeUtility::current_time() - get_range_beg - base_get_range_cost; _OB_LOG(INFO, "rould_old, cost time: %f us", (double)get_range_cost / MAX_TEST_COUNT); get_range_beg = ObTimeUtility::current_time(); for (int64_t i = begin_value, k = 0; k < MAX_TEST_COUNT; ++i, ++k) { ObFastFormatInt ffi(i); *const_cast(ffi.ptr() + ffi.length() / 2) = '.'; num1.from_v1(ffi.ptr(), ffi.length(), allocator); for (int64_t z = -(j+2); z < j + 2; z++) { num1.round_v2(z); } allocator.free(); if (i > end_value) { i -= (end_value - begin_value); } } get_range_cost = ObTimeUtility::current_time() - get_range_beg - base_get_range_cost; _OB_LOG(INFO, "round_v2, cost time: %f us", (double)get_range_cost / MAX_TEST_COUNT); get_range_beg = ObTimeUtility::current_time(); for (int64_t i = begin_value, k = 0; k < MAX_TEST_COUNT; ++i, ++k) { ObFastFormatInt ffi(i); *const_cast(ffi.ptr() + ffi.length() / 2) = '.'; num1.from_v1(ffi.ptr(), ffi.length(), allocator); for (int64_t z = -(j+2); z < j + 2; z++) { num1.round_v3(z); } allocator.free(); if (i > end_value) { i -= (end_value - begin_value); } } get_range_cost = ObTimeUtility::current_time() - get_range_beg - base_get_range_cost; _OB_LOG(INFO, "round_v3, cost time: %f us", (double)get_range_cost / MAX_TEST_COUNT); } for (int64_t j = 0; j < 20; ++j) { begin_value = pow(10, j); end_value = pow(10, j + 1) - 1; _OB_LOG(INFO, "test numer decimal2 %ld, [%ld, %ld]", j + 1, begin_value, end_value); get_range_beg = ObTimeUtility::current_time(); number::ObNumber num1; for (int64_t i = begin_value, k = 0; k < MAX_TEST_COUNT; ++i, ++k) { char tmp_buf[MAX_BUF_SIZE]; snprintf(tmp_buf, MAX_BUF_SIZE, "%.17lf", (i * (1 == i%2 ? 1 : -1)) * pow(10, 20)); num1.from_v1(tmp_buf, strlen(tmp_buf), allocator); for (int64_t z = -(j+2); z < j + 2; z++) { // } allocator.free(); if (i > end_value) { i -= (end_value - begin_value); } } base_get_range_cost = ObTimeUtility::current_time() - get_range_beg; get_range_beg = ObTimeUtility::current_time(); for (int64_t i = begin_value, k = 0; k < MAX_TEST_COUNT; ++i, ++k) { char tmp_buf[MAX_BUF_SIZE]; snprintf(tmp_buf, MAX_BUF_SIZE, "%.17lf", (i * (1 == i%2 ? 1 : -1)) * pow(10, 20)); num1.from_v1(tmp_buf, strlen(tmp_buf), allocator); for (int64_t z = -(j+2); z < j + 2; z++) { num1.round_v1(z); } allocator.free(); if (i > end_value) { i -= (end_value - begin_value); } } get_range_cost = ObTimeUtility::current_time() - get_range_beg - base_get_range_cost; _OB_LOG(INFO, "rould_old, cost time: %f us", (double)get_range_cost / MAX_TEST_COUNT); get_range_beg = ObTimeUtility::current_time(); for (int64_t i = begin_value, k = 0; k < MAX_TEST_COUNT; ++i, ++k) { char tmp_buf[MAX_BUF_SIZE]; snprintf(tmp_buf, MAX_BUF_SIZE, "%.17lf", (i * (1 == i%2 ? 1 : -1)) * pow(10, 20)); num1.from_v1(tmp_buf, strlen(tmp_buf), allocator); for (int64_t z = -(j+2); z < j + 2; z++) { num1.round_v2(z); } allocator.free(); if (i > end_value) { i -= (end_value - begin_value); } } get_range_cost = ObTimeUtility::current_time() - get_range_beg - base_get_range_cost; _OB_LOG(INFO, "round_v2, cost time: %f us", (double)get_range_cost / MAX_TEST_COUNT); get_range_beg = ObTimeUtility::current_time(); for (int64_t i = begin_value, k = 0; k < MAX_TEST_COUNT; ++i, ++k) { char tmp_buf[MAX_BUF_SIZE]; snprintf(tmp_buf, MAX_BUF_SIZE, "%.17lf", (i * (1 == i%2 ? 1 : -1)) * pow(10, 20)); num1.from_v1(tmp_buf, strlen(tmp_buf), allocator); for (int64_t z = -(j+2); z < j + 2; z++) { num1.round_v3(z); } allocator.free(); if (i > end_value) { i -= (end_value - begin_value); } } get_range_cost = ObTimeUtility::current_time() - get_range_beg - base_get_range_cost; _OB_LOG(INFO, "round_v3, cost time: %f us", (double)get_range_cost / MAX_TEST_COUNT); } } TEST(ObNumber, compare_number_cmp) { const int64_t MAX_TEST_COUNT = 100; const int64_t MAX_BUF_SIZE = 256; const int64_t MAX_TEST_LOOP = 10; char buf_alloc[MAX_BUF_SIZE]; int64_t pos = 0; int64_t get_range_beg = 0; int64_t get_range_cost = 0; int64_t begin_value = 0; int64_t end_value = 0; ObDataBuffer allocator(buf_alloc, MAX_BUF_SIZE); const int64_t MAX_DOUBLE_STRICT_PRINT_SIZE2 = 512; char buf[MAX_DOUBLE_STRICT_PRINT_SIZE2] = {}; int64_t length = 0 ; int64_t arr[MAX_TEST_COUNT]; for (int i = 0; i < MAX_TEST_COUNT; i++) { arr[i] = rand(); } number::ObNumber num1; number::ObNumber num2; for (int l = 0; l < MAX_TEST_LOOP; l++) { for (int i = 0; i < MAX_TEST_COUNT; i++) { for (int j = 0; j < MAX_TEST_COUNT; j++) { EXPECT_EQ(OB_SUCCESS, num1.from(arr[i] * (1 == l%2 ? 1 : -1), allocator)); EXPECT_EQ(OB_SUCCESS, num2.from(arr[j] * (1 == l%2 ? 1 : -1), allocator)); EXPECT_EQ(num1.compare_v1(num2), num1.compare(num2)); allocator.free(); } } } double darr[MAX_TEST_COUNT]; for (int i = 0; i < MAX_TEST_COUNT; i++) { darr[i] = drand48() * 1000000000000000000; } for (int l = 0; l < MAX_TEST_LOOP; l++) { for (int i = 0; i < MAX_TEST_COUNT; i++) { for (int j = 0; j < MAX_TEST_COUNT; j++) { MEMSET(buf, 0, MAX_DOUBLE_STRICT_PRINT_SIZE2); length = ob_gcvt_strict(darr[i] * (1 == l%2 ? 1 : -1), OB_GCVT_ARG_DOUBLE, sizeof(buf) - 1, buf, NULL, TRUE, TRUE, FALSE); EXPECT_EQ(OB_SUCCESS, num1.from(buf, length, allocator)); ObString str(sizeof(buf), static_cast(length), buf); OB_LOG(DEBUG, "debug jianhua 1", K(darr[i]), K(str)); MEMSET(buf, 0, MAX_DOUBLE_STRICT_PRINT_SIZE2); length = ob_gcvt_strict(darr[j] * (1 == l%2 ? 1 : -1), OB_GCVT_ARG_DOUBLE, sizeof(buf) - 1, buf, NULL, TRUE, TRUE, FALSE); EXPECT_EQ(OB_SUCCESS, num2.from(buf, length, allocator)); ObString str2(sizeof(buf), static_cast(length), buf); OB_LOG(DEBUG, "debug jianhua 2", K(darr[i]), K(str2)); EXPECT_EQ(num1.compare_v1(num2), num1.compare(num2)); allocator.free(); } } } } #define ASSERT_FORMAT_STREQ(n1, n2) ASSERT_STREQ(format(n1, 45), format(n2, 45)) TEST(ObNumber, sin_cos_tan) { int ret = OB_SUCCESS; const int64_t MAX_TEST_COUNT = 100; const int64_t MAX_BUF_SIZE = 256; char const_buf_alloc[MAX_BUF_SIZE]; ObDataBuffer const_allocator(const_buf_alloc, MAX_BUF_SIZE); ObNumber const_point_five; ret = const_point_five.from("0.5", const_allocator); EXPECT_EQ(OB_SUCCESS, ret); char buf_alloc[MAX_BUF_SIZE]; char buf_alloc2[MAX_BUF_SIZE]; int64_t tmp = 0; ObDataBuffer allocator(buf_alloc, MAX_BUF_SIZE); enum TestType { TT_MIN = 0, TT_MAX, TT_RAND, TT_MAX_TYPE }; uint32_t data[TestType::TT_MAX_TYPE][MAX_TEST_COUNT]; for (int64_t i = 0; i < MAX_TEST_COUNT; ++i) { data[TestType::TT_MIN][i] = 3; data[TestType::TT_MAX][i] = ObNumber::BASE - 1; data[TestType::TT_RAND][i] = (rand() % (ObNumber::BASE - 3)) + 3; } uint32_t digits1[ObNumber::MAX_STORE_LEN]; uint32_t digits2[ObNumber::MAX_STORE_LEN]; for (int64_t k = 0; k < TestType::TT_MAX_TYPE; ++k) { for (int64_t i = 1; i <= ObNumber::MAX_STORE_LEN; ++i) { _OB_LOG(INFO, "test numer k=%ld, i=%ld", k, i); for (int64_t j = 0; j < MAX_TEST_COUNT; ++j) { INIT_NUMBER12; ObNumber value_v1, value_v2, value_v3; // sin/cos == tan EXPECT_EQ(OB_SUCCESS, num1.sin(value_v1, allocator)); EXPECT_EQ(OB_SUCCESS, num1.cos(value_v2, allocator)); EXPECT_EQ(OB_SUCCESS, value_v1.div_v3(value_v2, value_v3, allocator)); EXPECT_EQ(OB_SUCCESS, num1.tan(value_v1, allocator)); ASSERT_FORMAT_STREQ(value_v1, value_v3); allocator.free(); // sin^2 + cos^2 = 1 EXPECT_EQ(OB_SUCCESS, num1.sin(value_v1, allocator)); EXPECT_EQ(OB_SUCCESS, num1.cos(value_v2, allocator)); EXPECT_EQ(OB_SUCCESS, value_v1.mul_v3(value_v1, value_v1, allocator)); EXPECT_EQ(OB_SUCCESS, value_v2.mul_v3(value_v2, value_v2, allocator)); EXPECT_EQ(OB_SUCCESS, value_v1.add_v3(value_v2, value_v1, allocator)); EXPECT_EQ(OB_SUCCESS, value_v2.from(ObNumber::get_positive_one(), allocator)); ASSERT_FORMAT_STREQ(value_v1, value_v2); allocator.free(); // sin(-x) = -sin(x) EXPECT_EQ(OB_SUCCESS, num1.sin(value_v1, allocator)); EXPECT_EQ(OB_SUCCESS, num1.negate(value_v2, allocator)); EXPECT_EQ(OB_SUCCESS, value_v2.sin(value_v2, allocator)); EXPECT_EQ(OB_SUCCESS, value_v2.negate(value_v2, allocator)); ASSERT_FORMAT_STREQ(value_v1, value_v2); allocator.free(); // cos(-x) = cos(x) EXPECT_EQ(OB_SUCCESS, num1.cos(value_v1, allocator)); EXPECT_EQ(OB_SUCCESS, num1.negate(value_v2, allocator)); EXPECT_EQ(OB_SUCCESS, value_v2.cos(value_v2, allocator)); ASSERT_FORMAT_STREQ(value_v1, value_v2); allocator.free(); // tan(-x) = tan(x) EXPECT_EQ(OB_SUCCESS, num1.tan(value_v1, allocator)); EXPECT_EQ(OB_SUCCESS, num1.negate(value_v2, allocator)); EXPECT_EQ(OB_SUCCESS, value_v2.tan(value_v2, allocator)); EXPECT_EQ(OB_SUCCESS, value_v2.negate(value_v2, allocator)); ASSERT_FORMAT_STREQ(value_v1, value_v2); allocator.free(); } } } } TEST(ObNumber, number_round_remainder) { const int64_t MAX_TEST_COUNT = 10000; const int64_t MAX_BUF_SIZE = 256; char buf_alloc[MAX_BUF_SIZE]; ObDataBuffer allocator(buf_alloc, MAX_BUF_SIZE); int64_t tmp = 0; const int64_t MAX_DOUBLE_STRICT_PRINT_SIZE2 = 512; char buf[MAX_DOUBLE_STRICT_PRINT_SIZE2] = {}; int64_t length = 0 ; enum TestType { TT_MIN = 0, TT_MAX, TT_RAND, TT_MAX_TYPE }; uint32_t data[TestType::TT_MAX_TYPE][MAX_TEST_COUNT]; for (int i = 0; i < MAX_TEST_COUNT; ++i) { data[TestType::TT_MIN][i] = rand() % 100; data[TestType::TT_MAX][i] = rand() % 100 + ObNumber::BASE - 120; data[TestType::TT_RAND][i] = rand() % ObNumber::BASE; } uint32_t digits1[ObNumber::MAX_STORE_LEN]; uint32_t digits2[ObNumber::MAX_STORE_LEN]; for (int64_t k = 0; k < TestType::TT_MAX_TYPE; ++k) { for (int64_t i = 1; i <= ObNumber::MAX_STORE_LEN; ++i) { OB_LOG(INFO, "debug jianhua 1", K(k), K(i)); for (int64_t j = 0; j < MAX_TEST_COUNT; ++j) { INIT_NUMBER12; ObNumber value; ASSERT_EQ(OB_SUCCESS, num1.round_remainder(num2, value, allocator)); ASSERT_EQ(OB_SUCCESS, value.add(value, value, allocator)); ASSERT_LE(value.abs_compare(num2), 0); allocator.free(); } } } } TEST(ObNumber, round_even_number) { const int64_t MAX_TEST_COUNT = 10000; const int64_t MAX_BUF_SIZE = 256; const int64_t MAX_TEST_LOOP = 10; char buf_alloc[MAX_BUF_SIZE]; ObDataBuffer allocator(buf_alloc, MAX_BUF_SIZE); const int64_t MAX_DOUBLE_STRICT_PRINT_SIZE2 = 512; char buf[MAX_DOUBLE_STRICT_PRINT_SIZE2] = {}; int64_t length = 0 ; number::ObNumber num1; number::ObNumber num2; double darr[MAX_TEST_COUNT]; for (int i = 0; i < MAX_TEST_COUNT; i++) { darr[i] = drand48() * pow(10.0,i % 16); } for (int l = 0; l < MAX_TEST_LOOP; l++) { for (int i = 0; i < MAX_TEST_COUNT; i++) { MEMSET(buf, 0, MAX_DOUBLE_STRICT_PRINT_SIZE2); double tmp_darr = darr[i] * (1 == l%2 ? 1 : -1); length = ob_gcvt_strict(tmp_darr, OB_GCVT_ARG_DOUBLE, sizeof(buf) - 1, buf, NULL, TRUE, TRUE, FALSE); EXPECT_EQ(OB_SUCCESS, num1.from(buf, length, allocator)); // int err = 0; // char *endptr = NULL; ObString str(sizeof(buf), static_cast(length), buf); // double tmp_darr = ObCharset::strntod(str.ptr(), str.length(), &endptr, &err); OB_LOG(DEBUG, "debug shanting 1", K(tmp_darr), K(num1), K(str), K(length)); EXPECT_EQ(OB_SUCCESS, num1.round_even_number()); double num = tmp_darr > 0 ? tmp_darr : - tmp_darr; int64_t integer = (int64_t)num; double rem = num - integer; int64_t res = (int64_t)integer; if (rem > 0.5 + DBL_EPSILON || (rem < 0.5 + DBL_EPSILON && rem > 0.5 - DBL_EPSILON && 1 == (integer % 2))) { res++; } res *= (tmp_darr > 0 ? 1 : -1); OB_LOG(DEBUG, "debug shanting 2", K(tmp_darr), K(res), K(num1), K(str)); EXPECT_EQ(OB_SUCCESS, num2.from(res, allocator)); ASSERT_EQ(0,num1.compare(num2)); allocator.free(); } } } TEST(ObNumber, DISABLED_compare_number_perf) { const int64_t MAX_TEST_COUNT = 200; const int64_t MAX_BUF_SIZE = 256; const int64_t MAX_TEST_LOOP = 10; char buf_alloc[MAX_BUF_SIZE]; int64_t pos = 0; int64_t get_range_beg = 0; int64_t get_range_cost = 0; int64_t base_get_range_cost = 0; int64_t begin_value = 0; int64_t end_value = 0; int64_t sum = 0; ObDataBuffer allocator(buf_alloc, MAX_BUF_SIZE); const int64_t MAX_DOUBLE_STRICT_PRINT_SIZE2 = 512; char buf[MAX_DOUBLE_STRICT_PRINT_SIZE2] = {}; int64_t length = 0; int64_t arr[MAX_TEST_COUNT]; for (int i = 0; i < MAX_TEST_COUNT; i++) { arr[i] = rand(); } get_range_beg = ObTimeUtility::current_time(); number::ObNumber num1; number::ObNumber num2; for (int l = 0; l < MAX_TEST_LOOP; l++) { for (int i = 0; i < MAX_TEST_COUNT; i++) { for (int j = 0; j < MAX_TEST_COUNT; j++) { EXPECT_EQ(OB_SUCCESS, num1.from(arr[i] * (1 == l%2 ? 1 : -1), allocator)); EXPECT_EQ(OB_SUCCESS, num2.from(arr[j] * (1 == l%2 ? 1 : -1), allocator)); sum += j; allocator.free(); } } } base_get_range_cost = ObTimeUtility::current_time() - get_range_beg; _OB_LOG(INFO, "compare base, cost time: %f us, sum=%ld", (double)base_get_range_cost / (MAX_TEST_COUNT * MAX_TEST_COUNT), sum); get_range_beg = ObTimeUtility::current_time(); for (int l = 0; l < MAX_TEST_LOOP; l++) { for (int i = 0; i < MAX_TEST_COUNT; i++) { for (int j = 0; j < MAX_TEST_COUNT; j++) { EXPECT_EQ(OB_SUCCESS, num1.from(arr[i] * (1 == l%2 ? 1 : -1), allocator)); EXPECT_EQ(OB_SUCCESS, num2.from(arr[j] * (1 == l%2 ? 1 : -1), allocator)); sum += num1.compare_v1(num2); allocator.free(); } } } get_range_cost = ObTimeUtility::current_time() - get_range_beg - base_get_range_cost; _OB_LOG(INFO, "compare int, cost time: %f us, sum=%ld", (double)get_range_cost / (MAX_TEST_COUNT * MAX_TEST_COUNT), sum); sum = 0; get_range_beg = ObTimeUtility::current_time(); for (int l = 0; l < MAX_TEST_LOOP; l++) { for (int i = 0; i < MAX_TEST_COUNT; i++) { for (int j = 0; j < MAX_TEST_COUNT; j++) { EXPECT_EQ(OB_SUCCESS, num1.from(arr[i] * (1 == l%2 ? 1 : -1), allocator)); EXPECT_EQ(OB_SUCCESS, num2.from(arr[j] * (1 == l%2 ? 1 : -1), allocator)); sum += num1.compare(num2); allocator.free(); } } } get_range_cost = ObTimeUtility::current_time() - get_range_beg - base_get_range_cost; _OB_LOG(INFO, "compare_v2 int, cost time: %f us, sum=%ld", (double)get_range_cost / (MAX_TEST_COUNT * MAX_TEST_COUNT), sum); double darr[MAX_TEST_COUNT]; for (int i = 0; i < MAX_TEST_COUNT; i++) { darr[i] = drand48() * 1000000000000000000; } sum = 0; get_range_beg = ObTimeUtility::current_time(); for (int l = 0; l < MAX_TEST_LOOP; l++) { for (int i = 0; i < MAX_TEST_COUNT; i++) { for (int j = 0; j < MAX_TEST_COUNT; j++) { MEMSET(buf, 0, MAX_DOUBLE_STRICT_PRINT_SIZE2); length = ob_gcvt_strict(darr[i] * (1 == l%2 ? 1 : -1), OB_GCVT_ARG_DOUBLE, sizeof(buf) - 1, buf, NULL, TRUE, TRUE, FALSE); EXPECT_EQ(OB_SUCCESS, num1.from(buf, length, allocator)); MEMSET(buf, 0, MAX_DOUBLE_STRICT_PRINT_SIZE2); length = ob_gcvt_strict(darr[j] * (1 == l%2 ? 1 : -1), OB_GCVT_ARG_DOUBLE, sizeof(buf) - 1, buf, NULL, TRUE, TRUE, FALSE); EXPECT_EQ(OB_SUCCESS, num2.from(buf, length, allocator)); sum += j; allocator.free(); } } } base_get_range_cost = ObTimeUtility::current_time() - get_range_beg; _OB_LOG(INFO, "compare base, cost time: %f us, sum=%ld", (double)base_get_range_cost / (MAX_TEST_COUNT * MAX_TEST_COUNT), sum); sum = 0; get_range_beg = ObTimeUtility::current_time(); for (int l = 0; l < MAX_TEST_LOOP; l++) { for (int i = 0; i < MAX_TEST_COUNT; i++) { for (int j = 0; j < MAX_TEST_COUNT; j++) { MEMSET(buf, 0, MAX_DOUBLE_STRICT_PRINT_SIZE2); length = ob_gcvt_strict(darr[i] * (1 == l%2 ? 1 : -1), OB_GCVT_ARG_DOUBLE, sizeof(buf) - 1, buf, NULL, TRUE, TRUE, FALSE); EXPECT_EQ(OB_SUCCESS, num1.from(buf, length, allocator)); MEMSET(buf, 0, MAX_DOUBLE_STRICT_PRINT_SIZE2); length = ob_gcvt_strict(darr[j] * (1 == l%2 ? 1 : -1), OB_GCVT_ARG_DOUBLE, sizeof(buf) - 1, buf, NULL, TRUE, TRUE, FALSE); EXPECT_EQ(OB_SUCCESS, num2.from(buf, length, allocator)); sum += num1.compare_v1(num2); allocator.free(); } } } get_range_cost = ObTimeUtility::current_time() - get_range_beg - base_get_range_cost; _OB_LOG(INFO, "compare double, cost time: %f us, sum=%ld", (double)get_range_cost / (MAX_TEST_COUNT * MAX_TEST_COUNT), sum); sum = 0; get_range_beg = ObTimeUtility::current_time(); for (int l = 0; l < MAX_TEST_LOOP; l++) { for (int i = 0; i < MAX_TEST_COUNT; i++) { for (int j = 0; j < MAX_TEST_COUNT; j++) { MEMSET(buf, 0, MAX_DOUBLE_STRICT_PRINT_SIZE2); length = ob_gcvt_strict(darr[i] * (1 == l%2 ? 1 : -1), OB_GCVT_ARG_DOUBLE, sizeof(buf) - 1, buf, NULL, TRUE, TRUE, FALSE); EXPECT_EQ(OB_SUCCESS, num1.from(buf, length, allocator)); MEMSET(buf, 0, MAX_DOUBLE_STRICT_PRINT_SIZE2); length = ob_gcvt_strict(darr[j] * (1 == l%2 ? 1 : -1), OB_GCVT_ARG_DOUBLE, sizeof(buf) - 1, buf, NULL, TRUE, TRUE, FALSE); EXPECT_EQ(OB_SUCCESS, num2.from(buf, length, allocator)); sum += num1.compare(num2); allocator.free(); } } } get_range_cost = ObTimeUtility::current_time() - get_range_beg - base_get_range_cost; _OB_LOG(INFO, "compare_v2 double, cost time: %f us, sum=%ld", (double)get_range_cost / (MAX_TEST_COUNT * MAX_TEST_COUNT), sum); } TEST(ObNumber, encode_decode) { const int64_t MAX_TEST_COUNT = 100; const int64_t MAX_BUF_SIZE = 256; const int64_t MAX_TEST_LOOP = 10; char buf_alloc[MAX_BUF_SIZE]; int64_t pos = 0; int64_t get_range_beg = 0; int64_t get_range_cost = 0; int64_t begin_value = 0; int64_t end_value = 0; ObDataBuffer allocator(buf_alloc, MAX_BUF_SIZE); const int64_t MAX_DOUBLE_STRICT_PRINT_SIZE2 = 512; char buf[MAX_DOUBLE_STRICT_PRINT_SIZE2] = {}; int64_t length = 0 ; int64_t arr[MAX_TEST_COUNT]; for (int i = 0; i < MAX_TEST_COUNT; i++) { arr[i] = rand(); } number::ObNumber num1; number::ObNumber num2; for (int l = 0; l < MAX_TEST_LOOP; l++) { for (int i = 0; i < MAX_TEST_COUNT; i++) { for (int j = 0; j < MAX_TEST_COUNT; j++) { EXPECT_EQ(OB_SUCCESS, num1.from(arr[i] * (1 == l%2 ? 1 : -1), allocator)); pos = 0 ; EXPECT_EQ(OB_SUCCESS, num1.encode(buf, MAX_DOUBLE_STRICT_PRINT_SIZE2, pos)); pos = 0 ; EXPECT_EQ(OB_SUCCESS, num2.decode(buf, MAX_DOUBLE_STRICT_PRINT_SIZE2, pos)); ASSERT_EQ(0, num1.compare_v1(num2)); allocator.free(); } } } double darr[MAX_TEST_COUNT]; for (int i = 0; i < MAX_TEST_COUNT; i++) { darr[i] = drand48() * 1000000000000000000; } for (int l = 0; l < MAX_TEST_LOOP; l++) { for (int i = 0; i < MAX_TEST_COUNT; i++) { for (int j = 0; j < MAX_TEST_COUNT; j++) { MEMSET(buf, 0, MAX_DOUBLE_STRICT_PRINT_SIZE2); length = ob_gcvt_strict(darr[i] * (1 == l%2 ? 1 : -1), OB_GCVT_ARG_DOUBLE, sizeof(buf) - 1, buf, NULL, TRUE, TRUE, FALSE); EXPECT_EQ(OB_SUCCESS, num1.from(buf, length, allocator)); pos = 0 ; EXPECT_EQ(OB_SUCCESS, num1.encode(buf, MAX_DOUBLE_STRICT_PRINT_SIZE2, pos)); pos = 0 ; EXPECT_EQ(OB_SUCCESS, num2.decode(buf, MAX_DOUBLE_STRICT_PRINT_SIZE2, pos)); ASSERT_EQ(0, num1.compare_v1(num2)); allocator.free(); } } } } TEST(ObNumber, int64_conversion) { const int64_t MAX_BUF_SIZE = 256; char buf_alloc[MAX_BUF_SIZE]; ObDataBuffer allocator(buf_alloc, MAX_BUF_SIZE); int64_t from_int = 0; int64_t to_int = 0; number::ObNumber num; // int64_t boundary test [-9,223,372,036,854,775,808, 9,223,372,036,854,775,807] allocator.free(); from_int = 9223372036854775806; ASSERT_EQ(OB_SUCCESS, num.from("9223372036854775806", allocator)); ASSERT_TRUE(num.is_int64()); ASSERT_EQ(OB_SUCCESS, num.cast_to_int64(to_int)); ASSERT_EQ(from_int, to_int); allocator.free(); from_int = 9223372036854775807; ASSERT_EQ(OB_SUCCESS, num.from("9223372036854775807", allocator)); ASSERT_TRUE(num.is_int64()); ASSERT_EQ(OB_SUCCESS, num.cast_to_int64(to_int)); ASSERT_EQ(from_int, to_int); allocator.free(); ASSERT_EQ(OB_SUCCESS, num.from("9223372036854775808", allocator)); ASSERT_FALSE(num.is_int64()); ASSERT_EQ(OB_INTEGER_PRECISION_OVERFLOW, num.cast_to_int64(to_int)); allocator.free(); from_int = -9223372036854775807; ASSERT_EQ(OB_SUCCESS, num.from("-9223372036854775807", allocator)); ASSERT_TRUE(num.is_int64()); ASSERT_EQ(OB_SUCCESS, num.cast_to_int64(to_int)); ASSERT_EQ(from_int, to_int); allocator.free(); from_int = INT64_MIN; ASSERT_EQ(OB_SUCCESS, num.from("-9223372036854775808", allocator)); ASSERT_TRUE(num.is_int64()); ASSERT_EQ(OB_SUCCESS, num.cast_to_int64(to_int)); ASSERT_EQ(from_int, to_int); allocator.free(); ASSERT_EQ(OB_SUCCESS, num.from("-9223372036854775809", allocator)); ASSERT_FALSE(num.is_int64()); ASSERT_EQ(OB_INTEGER_PRECISION_OVERFLOW, num.cast_to_int64(to_int)); // 0 value test for (from_int = -1; from_int <= 1; from_int++) { allocator.free(); ASSERT_EQ(OB_SUCCESS, num.from(from_int, allocator)); ASSERT_TRUE(num.is_int64()); ASSERT_EQ(OB_SUCCESS, num.cast_to_int64(to_int)); ASSERT_EQ(from_int, to_int); } // digital array boundary allocator.free(); from_int = 999999999; ASSERT_EQ(OB_SUCCESS, num.from(from_int, allocator)); ASSERT_TRUE(num.is_int64()); ASSERT_EQ(OB_SUCCESS, num.cast_to_int64(to_int)); ASSERT_EQ(from_int, to_int); allocator.free(); from_int = -999999999; ASSERT_EQ(OB_SUCCESS, num.from(from_int, allocator)); ASSERT_TRUE(num.is_int64()); ASSERT_EQ(OB_SUCCESS, num.cast_to_int64(to_int)); ASSERT_EQ(from_int, to_int); allocator.free(); from_int = 1000000000; ASSERT_EQ(OB_SUCCESS, num.from(from_int, allocator)); ASSERT_TRUE(num.is_int64()); ASSERT_EQ(OB_SUCCESS, num.cast_to_int64(to_int)); ASSERT_EQ(from_int, to_int); allocator.free(); from_int = -1000000000; ASSERT_EQ(OB_SUCCESS, num.from(from_int, allocator)); ASSERT_TRUE(num.is_int64()); ASSERT_EQ(OB_SUCCESS, num.cast_to_int64(to_int)); ASSERT_EQ(from_int, to_int); allocator.free(); from_int = 999999999999999999; ASSERT_EQ(OB_SUCCESS, num.from(from_int, allocator)); ASSERT_TRUE(num.is_int64()); ASSERT_EQ(OB_SUCCESS, num.cast_to_int64(to_int)); ASSERT_EQ(from_int, to_int); allocator.free(); from_int = -999999999999999999; ASSERT_EQ(OB_SUCCESS, num.from(from_int, allocator)); ASSERT_TRUE(num.is_int64()); ASSERT_EQ(OB_SUCCESS, num.cast_to_int64(to_int)); ASSERT_EQ(from_int, to_int); allocator.free(); from_int = 1000000000000000000; ASSERT_EQ(OB_SUCCESS, num.from(from_int, allocator)); ASSERT_TRUE(num.is_int64()); ASSERT_EQ(OB_SUCCESS, num.cast_to_int64(to_int)); ASSERT_EQ(from_int, to_int); allocator.free(); from_int = -1000000000000000000; ASSERT_EQ(OB_SUCCESS, num.from(from_int, allocator)); ASSERT_TRUE(num.is_int64()); ASSERT_EQ(OB_SUCCESS, num.cast_to_int64(to_int)); ASSERT_EQ(from_int, to_int); // decimal test allocator.free(); ASSERT_EQ(OB_SUCCESS, num.from("0.0", allocator)); ASSERT_TRUE(num.is_int64()); ASSERT_EQ(OB_SUCCESS, num.cast_to_int64(to_int)); ASSERT_EQ(0, to_int); allocator.free(); ASSERT_EQ(OB_SUCCESS, num.from("0.001", allocator)); ASSERT_FALSE(num.is_int64()); ASSERT_EQ(OB_INTEGER_PRECISION_OVERFLOW, num.cast_to_int64(to_int)); // overflow test // middle digit greater then 223372036 allocator.free(); ASSERT_EQ(OB_SUCCESS, num.from("-9223372037000000000", allocator)); ASSERT_FALSE(num.is_int64()); ASSERT_EQ(OB_INTEGER_PRECISION_OVERFLOW, num.cast_to_int64(to_int)); // max digit greater then 9 allocator.free(); ASSERT_EQ(OB_SUCCESS, num.from("10223372035000000000", allocator)); ASSERT_FALSE(num.is_int64()); ASSERT_EQ(OB_INTEGER_PRECISION_OVERFLOW, num.cast_to_int64(to_int)); } TEST(ObNumber, arithmetic_cmp) { const int64_t MAX_TEST_COUNT = 100; const int64_t MAX_TEST2_COUNT = 1000; const int64_t MAX_BUF_SIZE = 256; char buf_alloc[MAX_BUF_SIZE]; char buf_alloc2[MAX_BUF_SIZE]; int64_t pos = 0; int64_t get_range_beg = 0; int64_t get_range_cost = 0; int64_t begin_value = 0; int64_t end_value = 0; ObDataBuffer allocator(buf_alloc, MAX_BUF_SIZE); ObDataBuffer allocator2(buf_alloc2, MAX_BUF_SIZE); const int64_t MAX_DOUBLE_STRICT_PRINT_SIZE2 = 512; char buf[MAX_DOUBLE_STRICT_PRINT_SIZE2] = {}; int64_t length = 0 ; int64_t arr[MAX_TEST2_COUNT]; for (int i = 0; i < MAX_TEST2_COUNT; i++) { arr[i] = rand(); } double darr[MAX_TEST2_COUNT]; for (int i = 0; i < MAX_TEST2_COUNT; i++) { darr[i] = drand48() * 1000000000000000000; } allocator.free(); number::ObNumber num1; int64_t tmp_int64 = 0; ASSERT_EQ(OB_SUCCESS, num1.from("9223372036854775807", allocator)); ASSERT_EQ(OB_SUCCESS, num1.from("-9223372036854775808", allocator)); ASSERT_EQ(OB_SUCCESS, num1.from("9223372036854775808", allocator)); ASSERT_EQ(OB_SUCCESS, num1.from("9223372036854775808", allocator)); allocator.free(); ASSERT_EQ(OB_INTEGER_PRECISION_OVERFLOW, num1.cast_to_int64(tmp_int64)); ASSERT_EQ(OB_SUCCESS, num1.from("-9223372036854775809", allocator)); ASSERT_EQ(OB_INTEGER_PRECISION_OVERFLOW, num1.cast_to_int64(tmp_int64)); ASSERT_EQ(OB_SUCCESS, num1.from("9999999999000000000", allocator)); ASSERT_EQ(OB_INTEGER_PRECISION_OVERFLOW, num1.cast_to_int64(tmp_int64)); ASSERT_EQ(OB_SUCCESS, num1.from("9999999999999999999", allocator)); ASSERT_EQ(OB_INTEGER_PRECISION_OVERFLOW, num1.cast_to_int64(tmp_int64)); ASSERT_EQ(OB_SUCCESS, num1.from("-9999999999000000000", allocator)); ASSERT_EQ(OB_INTEGER_PRECISION_OVERFLOW, num1.cast_to_int64(tmp_int64)); ASSERT_EQ(OB_SUCCESS, num1.from("-9999999999999999999", allocator)); ASSERT_EQ(OB_INTEGER_PRECISION_OVERFLOW, num1.cast_to_int64(tmp_int64)); ASSERT_EQ(OB_SUCCESS, num1.from("1111111111111111111111111", allocator)); ASSERT_EQ(OB_INTEGER_PRECISION_OVERFLOW, num1.cast_to_int64(tmp_int64)); ASSERT_EQ(OB_SUCCESS, num1.from("-1111111111111111111111111", allocator)); ASSERT_EQ(OB_INTEGER_PRECISION_OVERFLOW, num1.cast_to_int64(tmp_int64)); ASSERT_EQ(OB_SUCCESS, num1.from("1111111000000000000000000", allocator)); ASSERT_EQ(OB_INTEGER_PRECISION_OVERFLOW, num1.cast_to_int64(tmp_int64)); ASSERT_EQ(OB_SUCCESS, num1.from("-1111111000000000000000000", allocator)); ASSERT_EQ(OB_INTEGER_PRECISION_OVERFLOW, num1.cast_to_int64(tmp_int64)); for (int64_t j = 1; j < 20; ++j) { begin_value = pow(10, j); end_value = pow(10, j + 1) - 1; number::ObNumber num1; number::ObNumber num2; int64_t add_step = (end_value - begin_value) / MAX_TEST_COUNT; if (add_step < 17) { add_step = 17; } _OB_LOG(INFO, "test numer %ld, [%ld, %ld], add_step=%ld", j + 1, begin_value, end_value, add_step); for (int64_t i = begin_value, k = 0; k < MAX_TEST_COUNT && i < end_value - begin_value; i += add_step, ++k) { int64_t tmp_i = i * (1 == i%2 ? 1 : -1); num1.from(tmp_i, allocator); _OB_LOG(INFO, "debug jianhua, j=%ld, i=%ld, k=%ld", j ,i, k); int64_t tmp_v1 = 0; if (tmp_i <= INT64_MAX && tmp_i >= INT64_MIN) { EXPECT_EQ(OB_SUCCESS, num1.cast_to_int64(tmp_v1)); ASSERT_EQ(tmp_i, tmp_v1); } for (int d = 0; d < MAX_TEST2_COUNT; d++) { const int64_t test_rd = (arr[d] * (1 == d%2 ? 1 : -1) % begin_value) + tmp_i; num2.from(test_rd, allocator2); ObNumber value_old,value_new,value_new2; EXPECT_EQ(OB_SUCCESS, num1.add_v1(num2, value_old, allocator2)); EXPECT_EQ(OB_SUCCESS, num1.add_v2(num2, value_new, allocator2)); EXPECT_EQ(OB_SUCCESS, num1.add_v3(num2, value_new2, allocator2)); ASSERT_EQ(0, value_new.compare(value_old)); ASSERT_EQ(0, value_new2.compare(value_old)); allocator2.free(); num2.from(test_rd, allocator2); EXPECT_EQ(OB_SUCCESS, num1.sub_v1(num2, value_old, allocator2)); EXPECT_EQ(OB_SUCCESS, num1.sub_v2(num2, value_new, allocator2)); EXPECT_EQ(OB_SUCCESS, num1.sub_v3(num2, value_new2, allocator2)); ASSERT_EQ(0, value_new.compare(value_old)); ASSERT_EQ(0, value_new2.compare(value_old)); allocator2.free(); num2.from(test_rd, allocator2); EXPECT_EQ(OB_SUCCESS, num1.mul_v1(num2, value_old, allocator2)); EXPECT_EQ(OB_SUCCESS, num1.mul_v2(num2, value_new, allocator2)); EXPECT_EQ(OB_SUCCESS, num1.mul_v3(num2, value_new2, allocator2)); ASSERT_EQ(0, value_new.compare(value_old)); ASSERT_EQ(0, value_new2.compare(value_old)); allocator2.free(); num2.from(test_rd, allocator2); if (!num2.is_zero()) { //EXPECT_EQ(OB_SUCCESS, num1.div_v1(num2, value_old, allocator2)); EXPECT_EQ(OB_SUCCESS, num1.div_v2(num2, value_new, allocator2)); EXPECT_EQ(OB_SUCCESS, num1.div_v3(num2, value_new2, allocator2)); //ASSERT_EQ(0, value_new.compare(value_old)); //ASSERT_EQ(0, value_new2.compare(value_old)); ASSERT_EQ(0, value_new2.compare(value_new)); } allocator2.free(); num2.from(test_rd, allocator2); if (!num2.is_zero()) { //EXPECT_EQ(OB_SUCCESS, num1.rem_v1(num2, value_old, allocator2)); EXPECT_EQ(OB_SUCCESS, num1.rem_v2(num2, value_old, allocator2)); EXPECT_EQ(OB_SUCCESS, num1.rem_v2(num2, value_new, allocator2)); ASSERT_EQ(0, value_new.compare(value_old)); } allocator2.free(); } if (i > end_value) { i -= (end_value - begin_value); } allocator.free(); } } for (int64_t j = 1; j < 20; ++j) { begin_value = pow(10, j); end_value = pow(10, j + 1) - 1; number::ObNumber num1; number::ObNumber num2; int64_t add_step = (end_value - begin_value) / MAX_TEST_COUNT; if (add_step < 17) { add_step = 17; } _OB_LOG(INFO, "test numer %ld, [%ld, %ld], add_step=%ld", j + 1, begin_value, end_value, add_step); for (int64_t i = begin_value, k = 0; k < MAX_TEST_COUNT && i < end_value - begin_value; i += add_step, ++k) { for (int64_t m = 0; m < j; ++m) { ObFastFormatInt ffi(i * (1 == i%2 ? 1 : -1)); *const_cast(ffi.ptr() + m) = '.'; { ObLogger::ObTraceLogGuard trace_log_guard; num1.from_v1(ffi.ptr(), ffi.length(), allocator); } _OB_LOG(INFO, "debug jianhua decimal, j=%ld, i=%ld, k=%ld, ffi=%.*s", j, i, k, static_cast(ffi.length()), ffi.ptr()); for (int d = 0; d < MAX_TEST2_COUNT; d++) { MEMSET(buf, 0, MAX_DOUBLE_STRICT_PRINT_SIZE2); double test_rd = darr[d] * (1 == d%2 ? 1 : -1) / begin_value + i * (1 == i%2 ? 1 : -1); length = ob_gcvt_strict(test_rd, OB_GCVT_ARG_DOUBLE, sizeof(buf) - 1, buf, NULL, TRUE, TRUE, FALSE); ObNumber value_old,value_new,value_new2; EXPECT_EQ(OB_SUCCESS, num2.from(buf, length, allocator2)); EXPECT_EQ(OB_SUCCESS, num1.add_v1(num2, value_old, allocator2)); EXPECT_EQ(OB_SUCCESS, num1.add_v2(num2, value_new, allocator2)); EXPECT_EQ(OB_SUCCESS, num1.add_v3(num2, value_new2, allocator2)); ASSERT_EQ(0, value_new.compare(value_old)); ASSERT_EQ(0, value_new2.compare(value_old)); allocator2.free(); EXPECT_EQ(OB_SUCCESS, num2.from(buf, length, allocator2)); EXPECT_EQ(OB_SUCCESS, num1.sub_v1(num2, value_old, allocator2)); EXPECT_EQ(OB_SUCCESS, num1.sub_v2(num2, value_new, allocator2)); EXPECT_EQ(OB_SUCCESS, num1.sub_v3(num2, value_new2, allocator2)); ASSERT_EQ(0, value_new.compare(value_old)); ASSERT_EQ(0, value_new2.compare(value_old)); allocator2.free(); EXPECT_EQ(OB_SUCCESS, num2.from(buf, length, allocator2)); EXPECT_EQ(OB_SUCCESS, num1.mul_v1(num2, value_old, allocator2)); EXPECT_EQ(OB_SUCCESS, num1.mul_v2(num2, value_new, allocator2)); EXPECT_EQ(OB_SUCCESS, num1.mul_v3(num2, value_new2, allocator2)); ASSERT_EQ(0, value_new.compare(value_old)); ASSERT_EQ(0, value_new2.compare(value_old)); allocator2.free(); EXPECT_EQ(OB_SUCCESS, num2.from(buf, length, allocator2)); if (!num2.is_zero()) { //EXPECT_EQ(OB_SUCCESS, num1.div_v1(num2, value_old, allocator2)); EXPECT_EQ(OB_SUCCESS, num1.div_v2(num2, value_new, allocator2)); EXPECT_EQ(OB_SUCCESS, num1.div_v3(num2, value_new2, allocator2)); //ASSERT_EQ(0, value_new.compare(value_old)); //ASSERT_EQ(0, value_new2.compare(value_old)); ASSERT_EQ(0, value_new2.compare(value_new)); } allocator2.free(); } allocator.free(); } if (i > end_value) { i -= (end_value - begin_value); } } } //200 char const_number_str[] = "98765432109876543210987654321098765432109876543210987654321098765432109876543210987654321098765432109876543210987654321098765432109876543210987654321098765432109876543210987654321098765432109876543210"; ObString const_number_string(const_number_str); for (int64_t j = 1; j < 80; ++j) { number::ObNumber num1, num2, value_old,value_new; for (int64_t i = 0; i < 11; ++i) { _OB_LOG(INFO, "debug jianhua decimal, j=%ld, i=%ld, ffi=%.*s", j, i, static_cast(j), const_number_str + i); for (int64_t m = 0; m < 11; ++m) { allocator.free(); { ObLogger::ObTraceLogGuard trace_log_guard; num1.from_v1(const_number_str + i, j, allocator); num2.from_v1(const_number_str + m, j, allocator); } value_old.set_zero(); value_new.set_zero(); EXPECT_EQ(num1.add_v1(num2, value_old, allocator2), num1.add_v2(num2, value_new, allocator2)); ASSERT_EQ(0, value_new.compare(value_old)); allocator2.free(); value_old.set_zero(); value_new.set_zero(); EXPECT_EQ(num1.add_v1(num2, value_old, allocator2), num1.add_v3(num2, value_new, allocator2, false)); ASSERT_EQ(0, value_new.compare(value_old)); allocator2.free(); value_old.set_zero(); value_new.set_zero(); EXPECT_EQ(num1.add_v1(num2, value_old, allocator2), num1.add_v3(num2, value_new, allocator2)); ASSERT_EQ(0, value_new.compare(value_old)); allocator2.free(); value_old.set_zero(); value_new.set_zero(); EXPECT_EQ(num1.sub_v1(num2, value_old, allocator2), num1.sub_v2(num2, value_new, allocator2)); ASSERT_EQ(0, value_new.compare(value_old)); allocator2.free(); value_old.set_zero(); value_new.set_zero(); EXPECT_EQ(num1.sub_v1(num2, value_old, allocator2), num1.sub_v3(num2, value_new, allocator2, false)); ASSERT_EQ(0, value_new.compare(value_old)); allocator2.free(); value_old.set_zero(); value_new.set_zero(); EXPECT_EQ(num1.sub_v1(num2, value_old, allocator2), num1.sub_v3(num2, value_new, allocator2)); ASSERT_EQ(0, value_new.compare(value_old)); allocator2.free(); value_old.set_zero(); value_new.set_zero(); EXPECT_EQ(num1.mul_v1(num2, value_old, allocator2), num1.mul_v2(num2, value_new, allocator2)); ASSERT_EQ(0, value_new.compare(value_old)); allocator2.free(); value_old.set_zero(); value_new.set_zero(); EXPECT_EQ(num1.mul_v1(num2, value_old, allocator2), num1.mul_v3(num2, value_new, allocator2, false)); ASSERT_EQ(0, value_new.compare(value_old)); allocator2.free(); value_old.set_zero(); value_new.set_zero(); EXPECT_EQ(num1.mul_v1(num2, value_old, allocator2), num1.mul_v3(num2, value_new, allocator2)); ASSERT_EQ(0, value_new.compare(value_old)); allocator2.free(); if (!num2.is_zero()) { value_old.set_zero(); value_new.set_zero(); //EXPECT_EQ(num1.div_v1(num2, value_old, allocator2), num1.div_v2(num2, value_new, allocator2)); EXPECT_EQ(num1.div_v3(num2, value_old, allocator2), num1.div_v2(num2, value_new, allocator2)); ASSERT_EQ(0, value_new.compare(value_old)); //value_old.set_zero(); //value_new.set_zero(); //EXPECT_EQ(num1.div_v1(num2, value_old, allocator2), num1.div_v3(num2, value_new, allocator2)); //ASSERT_EQ(0, value_new.compare(value_old)); } allocator2.free(); } } } } TEST(ObNumber, arithmetic_cmp_v2) { const int64_t MAX_TEST_COUNT = 10000; const int64_t MAX_BUF_SIZE = 256; char buf_alloc[MAX_BUF_SIZE]; char buf_alloc2[MAX_BUF_SIZE]; int64_t tmp = 0; int64_t pos = 0; int64_t get_range_beg = 0; int64_t get_range_cost = 0; int64_t begin_value = 0; int64_t end_value = 0; ObDataBuffer allocator(buf_alloc, MAX_BUF_SIZE); const int64_t MAX_DOUBLE_STRICT_PRINT_SIZE2 = 512; char buf[MAX_DOUBLE_STRICT_PRINT_SIZE2] = {}; int64_t length = 0 ; enum TestType { TT_MIN = 0, TT_MAX, TT_RAND, TT_MAX_TYPE }; uint32_t data[TestType::TT_MAX_TYPE][MAX_TEST_COUNT]; for (int64_t i = 0; i < MAX_TEST_COUNT; ++i) { data[TestType::TT_MIN][i] = rand() % 100; data[TestType::TT_MAX][i] = rand() % 100 + ObNumber::BASE - 120; data[TestType::TT_RAND][i] = rand() % ObNumber::BASE; } uint32_t digits1[ObNumber::MAX_STORE_LEN]; uint32_t digits2[ObNumber::MAX_STORE_LEN]; for (int64_t k = 0; k < TestType::TT_MAX_TYPE; ++k) { for (int64_t i = 1; i <= ObNumber::MAX_STORE_LEN; ++i) { _OB_LOG(INFO, "test numer k=%ld, i=%ld", k, i); for (int64_t j = 0; j < MAX_TEST_COUNT; ++j) { INIT_NUMBER12; // OB_LOG(INFO, "--", K(j), K(num1), K(num2)); ObNumber value_v1,value_v2,value_v3; EXPECT_EQ(OB_SUCCESS, num1.add_v1(num2, value_v1, allocator)); EXPECT_EQ(OB_SUCCESS, num1.add_v2(num2, value_v2, allocator)); EXPECT_EQ(OB_SUCCESS, num1.add_v3(num2, value_v3, allocator)); ASSERT_EQ(0, value_v1.compare(value_v2)); ASSERT_EQ(0, value_v1.compare(value_v3)); allocator.free(); EXPECT_EQ(OB_SUCCESS, num1.sub_v1(num2, value_v1, allocator)); EXPECT_EQ(OB_SUCCESS, num1.sub_v2(num2, value_v2, allocator)); EXPECT_EQ(OB_SUCCESS, num1.sub_v3(num2, value_v3, allocator)); ASSERT_EQ(0, value_v1.compare(value_v2)); ASSERT_EQ(0, value_v1.compare(value_v3)); allocator.free(); EXPECT_EQ(OB_SUCCESS, num1.mul_v1(num2, value_v1, allocator)); EXPECT_EQ(OB_SUCCESS, num1.mul_v2(num2, value_v2, allocator)); EXPECT_EQ(OB_SUCCESS, num1.mul_v3(num2, value_v3, allocator)); ASSERT_EQ(0, value_v1.compare(value_v2)); ASSERT_EQ(0, value_v1.compare(value_v3)); allocator.free(); //EXPECT_EQ(OB_SUCCESS, num1.div_v1(num2, value_v1, allocator)); EXPECT_EQ(OB_SUCCESS, num1.div_v2(num2, value_v2, allocator)); EXPECT_EQ(OB_SUCCESS, num1.div_v3(num2, value_v3, allocator)); ASSERT_EQ(0, value_v3.compare(value_v2)); // ASSERT_EQ(0, value_v1.compare(value_v2)); // ASSERT_EQ(0, value_v1.compare(value_v3)); allocator.free(); } } } } TEST(ObNumber, DISABLED_arithmetic_perf_v2) { const int64_t MAX_TEST_COUNT = 10000; const int64_t MAX_TEST_COUNT2 = 20; const int64_t MAX_BUF_SIZE = 256; char buf_alloc[MAX_BUF_SIZE]; char buf_alloc2[MAX_BUF_SIZE]; int64_t tmp = 0; int64_t pos = 0; int64_t get_range_beg = 0; int64_t base_get_range_cost = 0; int64_t base_get_range_cost2 = 0; int64_t get_range_cost = 0; int64_t get_range_cost2 = 0; int64_t begin_value = 0; int64_t end_value = 0; ObDataBuffer allocator(buf_alloc, MAX_BUF_SIZE); const int64_t MAX_DOUBLE_STRICT_PRINT_SIZE2 = 512; char buf[MAX_DOUBLE_STRICT_PRINT_SIZE2] = {}; int64_t length = 0 ; enum TestType { TT_MIN = 0, TT_MAX, TT_RAND, TT_MAX_TYPE }; uint32_t data[TestType::TT_MAX_TYPE][MAX_TEST_COUNT]; for (int64_t i = 0; i < MAX_TEST_COUNT; ++i) { data[TestType::TT_MIN][i] = rand() % 100; data[TestType::TT_MAX][i] = rand() % 100 + ObNumber::BASE - 120; data[TestType::TT_RAND][i] = rand() % ObNumber::BASE; } uint32_t digits1[ObNumber::MAX_STORE_LEN]; uint32_t digits2[ObNumber::MAX_STORE_LEN]; for (int64_t k = 0; k < TestType::TT_MAX_TYPE; ++k) { for (int64_t i = 1; i <= ObNumber::MAX_STORE_LEN; ++i) { _OB_LOG(INFO, "test numer k=%ld, i=%ld", k, i); ObNumber value_v1; base_get_range_cost = 0; for (int64_t r = 1; r <= MAX_TEST_COUNT2; ++r) { get_range_beg = ObTimeUtility::current_time(); for (int64_t j = 0; j < MAX_TEST_COUNT; ++j) { INIT_NUMBER12; // EXPECT_EQ(OB_SUCCESS, num1.add_v1(num2, value_v1, allocator)); allocator.free(); } base_get_range_cost += (ObTimeUtility::current_time() - get_range_beg); } base_get_range_cost2 = base_get_range_cost / MAX_TEST_COUNT2; get_range_cost = 0; for (int64_t r = 1; r <= MAX_TEST_COUNT2; ++r) { get_range_beg = ObTimeUtility::current_time(); for (int64_t j = 0; j < MAX_TEST_COUNT; ++j) { INIT_NUMBER12; EXPECT_EQ(OB_SUCCESS, num1.add_v1(num2, value_v1, allocator)); allocator.free(); } get_range_cost += (ObTimeUtility::current_time() - get_range_beg); } get_range_cost2 = get_range_cost/MAX_TEST_COUNT2 - base_get_range_cost2; _OB_LOG(INFO, "add v1, cost time: %f us", (double)get_range_cost2 / MAX_TEST_COUNT); get_range_cost = 0; for (int64_t r = 1; r <= MAX_TEST_COUNT2; ++r) { get_range_beg = ObTimeUtility::current_time(); for (int64_t j = 0; j < MAX_TEST_COUNT; ++j) { INIT_NUMBER12; EXPECT_EQ(OB_SUCCESS, num1.add_v2(num2, value_v1, allocator)); allocator.free(); } get_range_cost += (ObTimeUtility::current_time() - get_range_beg); } get_range_cost2 = get_range_cost/MAX_TEST_COUNT2 - base_get_range_cost2; _OB_LOG(INFO, "add v2, cost time: %f us", (double)get_range_cost2 / MAX_TEST_COUNT); get_range_cost = 0; for (int64_t r = 1; r <= MAX_TEST_COUNT2; ++r) { get_range_beg = ObTimeUtility::current_time(); for (int64_t j = 0; j < MAX_TEST_COUNT; ++j) { INIT_NUMBER12; EXPECT_EQ(OB_SUCCESS, num1.add_v3(num2, value_v1, allocator)); allocator.free(); } get_range_cost += (ObTimeUtility::current_time() - get_range_beg); } get_range_cost2 = get_range_cost/MAX_TEST_COUNT2 - base_get_range_cost2; _OB_LOG(INFO, "add v3, cost time: %f us", (double)get_range_cost2 / MAX_TEST_COUNT); get_range_cost = 0; for (int64_t r = 1; r <= MAX_TEST_COUNT2; ++r) { get_range_beg = ObTimeUtility::current_time(); for (int64_t j = 0; j < MAX_TEST_COUNT; ++j) { INIT_NUMBER12; EXPECT_EQ(OB_SUCCESS, num1.sub_v1(num2, value_v1, allocator)); allocator.free(); } get_range_cost += (ObTimeUtility::current_time() - get_range_beg); } get_range_cost2 = get_range_cost/MAX_TEST_COUNT2 - base_get_range_cost2; _OB_LOG(INFO, "sub v1, cost time: %f us", (double)get_range_cost2 / MAX_TEST_COUNT); get_range_cost = 0; for (int64_t r = 1; r <= MAX_TEST_COUNT2; ++r) { get_range_beg = ObTimeUtility::current_time(); for (int64_t j = 0; j < MAX_TEST_COUNT; ++j) { INIT_NUMBER12; EXPECT_EQ(OB_SUCCESS, num1.sub_v2(num2, value_v1, allocator)); allocator.free(); } get_range_cost += (ObTimeUtility::current_time() - get_range_beg); } get_range_cost2 = get_range_cost/MAX_TEST_COUNT2 - base_get_range_cost2; _OB_LOG(INFO, "sub v2, cost time: %f us", (double)get_range_cost2 / MAX_TEST_COUNT); get_range_cost = 0; for (int64_t r = 1; r <= MAX_TEST_COUNT2; ++r) { get_range_beg = ObTimeUtility::current_time(); for (int64_t j = 0; j < MAX_TEST_COUNT; ++j) { INIT_NUMBER12; EXPECT_EQ(OB_SUCCESS, num1.sub_v3(num2, value_v1, allocator)); allocator.free(); } get_range_cost += (ObTimeUtility::current_time() - get_range_beg); } get_range_cost2 = get_range_cost/MAX_TEST_COUNT2 - base_get_range_cost2; _OB_LOG(INFO, "sub v3, cost time: %f us", (double)get_range_cost2 / MAX_TEST_COUNT); get_range_cost = 0; for (int64_t r = 1; r <= MAX_TEST_COUNT2; ++r) { get_range_beg = ObTimeUtility::current_time(); for (int64_t j = 0; j < MAX_TEST_COUNT; ++j) { INIT_NUMBER12; EXPECT_EQ(OB_SUCCESS, num1.mul_v1(num2, value_v1, allocator)); allocator.free(); } get_range_cost += (ObTimeUtility::current_time() - get_range_beg); } get_range_cost2 = get_range_cost/MAX_TEST_COUNT2 - base_get_range_cost2; _OB_LOG(INFO, "mul v1, cost time: %f us", (double)get_range_cost2 / MAX_TEST_COUNT); get_range_cost = 0; for (int64_t r = 1; r <= MAX_TEST_COUNT2; ++r) { get_range_beg = ObTimeUtility::current_time(); for (int64_t j = 0; j < MAX_TEST_COUNT; ++j) { INIT_NUMBER12; EXPECT_EQ(OB_SUCCESS, num1.mul_v2(num2, value_v1, allocator)); allocator.free(); } get_range_cost += (ObTimeUtility::current_time() - get_range_beg); } get_range_cost2 = get_range_cost/MAX_TEST_COUNT2 - base_get_range_cost2; _OB_LOG(INFO, "mul v2, cost time: %f us", (double)get_range_cost2 / MAX_TEST_COUNT); get_range_cost = 0; for (int64_t r = 1; r <= MAX_TEST_COUNT2; ++r) { get_range_beg = ObTimeUtility::current_time(); for (int64_t j = 0; j < MAX_TEST_COUNT; ++j) { INIT_NUMBER12; EXPECT_EQ(OB_SUCCESS, num1.mul_v3(num2, value_v1, allocator)); allocator.free(); } get_range_cost += (ObTimeUtility::current_time() - get_range_beg); } get_range_cost2 = get_range_cost/MAX_TEST_COUNT2 - base_get_range_cost2; _OB_LOG(INFO, "mul v3, cost time: %f us", (double)get_range_cost2 / MAX_TEST_COUNT); //get_range_cost = 0; //for (int64_t r = 1; r <= MAX_TEST_COUNT2; ++r) { // get_range_beg = ObTimeUtility::current_time(); // for (int64_t j = 0; j < MAX_TEST_COUNT; ++j) { // INIT_NUMBER12; // EXPECT_EQ(OB_SUCCESS, num1.div_v1(num2, value_v1, allocator)); // allocator.free(); // } // get_range_cost += (ObTimeUtility::current_time() - get_range_beg); //} //get_range_cost2 = get_range_cost/MAX_TEST_COUNT2 - base_get_range_cost2; //_OB_LOG(INFO, "div v1, cost time: %f us", (double)get_range_cost2 / MAX_TEST_COUNT); get_range_cost = 0; for (int64_t r = 1; r <= MAX_TEST_COUNT2; ++r) { get_range_beg = ObTimeUtility::current_time(); for (int64_t j = 0; j < MAX_TEST_COUNT; ++j) { INIT_NUMBER12; EXPECT_EQ(OB_SUCCESS, num1.div_v2(num2, value_v1, allocator)); allocator.free(); } get_range_cost += (ObTimeUtility::current_time() - get_range_beg); } get_range_cost2 = get_range_cost/MAX_TEST_COUNT2 - base_get_range_cost2; _OB_LOG(INFO, "div v2, cost time: %f us", (double)get_range_cost2 / MAX_TEST_COUNT); get_range_cost = 0; for (int64_t r = 1; r <= MAX_TEST_COUNT2; ++r) { get_range_beg = ObTimeUtility::current_time(); for (int64_t j = 0; j < MAX_TEST_COUNT; ++j) { INIT_NUMBER12; EXPECT_EQ(OB_SUCCESS, num1.div_v3(num2, value_v1, allocator)); allocator.free(); } get_range_cost += (ObTimeUtility::current_time() - get_range_beg); } get_range_cost2 = get_range_cost/MAX_TEST_COUNT2 - base_get_range_cost2; _OB_LOG(INFO, "div v3, cost time: %f us", (double)get_range_cost2 / MAX_TEST_COUNT); get_range_cost = 0; for (int64_t r = 1; r <= MAX_TEST_COUNT2; ++r) { get_range_beg = ObTimeUtility::current_time(); for (int64_t j = 0; j < MAX_TEST_COUNT; ++j) { INIT_NUMBER12; if (!num2.is_zero()) { // EXPECT_EQ(OB_SUCCESS, num1.rem_v1(num2, value_v1, allocator)); EXPECT_EQ(OB_SUCCESS, num1.rem_v2(num2, value_v1, allocator)); } allocator.free(); } get_range_cost += (ObTimeUtility::current_time() - get_range_beg); } get_range_cost2 = get_range_cost/MAX_TEST_COUNT2 - base_get_range_cost2; _OB_LOG(INFO, "rem v1, cost time: %f us", (double)get_range_cost2 / MAX_TEST_COUNT); get_range_cost = 0; for (int64_t r = 1; r <= MAX_TEST_COUNT2; ++r) { get_range_beg = ObTimeUtility::current_time(); for (int64_t j = 0; j < MAX_TEST_COUNT; ++j) { INIT_NUMBER12; if (!num2.is_zero()) { EXPECT_EQ(OB_SUCCESS, num1.rem_v2(num2, value_v1, allocator)); } allocator.free(); } get_range_cost += (ObTimeUtility::current_time() - get_range_beg); } get_range_cost2 = get_range_cost/MAX_TEST_COUNT2 - base_get_range_cost2; _OB_LOG(INFO, "rem v3, cost time: %f us", (double)get_range_cost2 / MAX_TEST_COUNT); } } } int main(int argc, char **argv) { OB_LOGGER.set_file_name("test_number_v2.log", true); //oceanbase::common::ObLogger::get_logger().set_log_level("DEBUG"); ::testing::InitGoogleTest(&argc,argv); return RUN_ALL_TESTS(); }