mirror of
https://github.com/fmtlib/fmt.git
synced 2025-01-15 19:51:19 +00:00
Optimize floating point formatting
This commit is contained in:
parent
3c13a88b14
commit
bf19051a9f
@ -1031,7 +1031,7 @@ template <typename Char> void copy2(Char* dst, const char* src) {
|
|||||||
*dst++ = static_cast<Char>(*src++);
|
*dst++ = static_cast<Char>(*src++);
|
||||||
*dst = static_cast<Char>(*src);
|
*dst = static_cast<Char>(*src);
|
||||||
}
|
}
|
||||||
inline void copy2(char* dst, const char* src) { memcpy(dst, src, 2); }
|
FMT_INLINE void copy2(char* dst, const char* src) { memcpy(dst, src, 2); }
|
||||||
|
|
||||||
template <typename Iterator> struct format_decimal_result {
|
template <typename Iterator> struct format_decimal_result {
|
||||||
Iterator begin;
|
Iterator begin;
|
||||||
@ -1764,6 +1764,9 @@ inline Char* write_significand(Char* out, UInt significand,
|
|||||||
if (!decimal_point)
|
if (!decimal_point)
|
||||||
return format_decimal(out, significand, significand_size).end;
|
return format_decimal(out, significand, significand_size).end;
|
||||||
auto end = format_decimal(out + 1, significand, significand_size).end;
|
auto end = format_decimal(out + 1, significand, significand_size).end;
|
||||||
|
if (integral_size == 1)
|
||||||
|
out[0] = out[1];
|
||||||
|
else
|
||||||
std::copy_n(out + 1, integral_size, out);
|
std::copy_n(out + 1, integral_size, out);
|
||||||
out[integral_size] = decimal_point;
|
out[integral_size] = decimal_point;
|
||||||
return end;
|
return end;
|
||||||
@ -1799,8 +1802,8 @@ OutputIt write_float(OutputIt out, const DecimalFP& fp,
|
|||||||
auto significand = fp.significand;
|
auto significand = fp.significand;
|
||||||
int significand_size = get_significand_size(fp);
|
int significand_size = get_significand_size(fp);
|
||||||
static const Char zero = static_cast<Char>('0');
|
static const Char zero = static_cast<Char>('0');
|
||||||
char sign = data::signs[fspecs.sign];
|
auto sign = fspecs.sign;
|
||||||
int size = significand_size + (fspecs.sign ? 1 : 0);
|
int size = significand_size + (sign ? 1 : 0);
|
||||||
using iterator = remove_reference_t<decltype(reserve(out, 0))>;
|
using iterator = remove_reference_t<decltype(reserve(out, 0))>;
|
||||||
|
|
||||||
int output_exp = fp.exponent + significand_size - 1;
|
int output_exp = fp.exponent + significand_size - 1;
|
||||||
@ -1816,21 +1819,19 @@ OutputIt write_float(OutputIt out, const DecimalFP& fp,
|
|||||||
if (use_exp_format()) {
|
if (use_exp_format()) {
|
||||||
int num_zeros = 0;
|
int num_zeros = 0;
|
||||||
if (fspecs.showpoint) {
|
if (fspecs.showpoint) {
|
||||||
num_zeros = fspecs.precision - significand_size;
|
num_zeros = (std::max)(fspecs.precision - significand_size, 0);
|
||||||
if (num_zeros < 0) num_zeros = 0;
|
size += num_zeros;
|
||||||
} else if (significand_size == 1) {
|
} else if (significand_size == 1) {
|
||||||
decimal_point = Char();
|
decimal_point = Char();
|
||||||
}
|
}
|
||||||
|
auto abs_output_exp = output_exp >= 0 ? output_exp : -output_exp;
|
||||||
auto abs_output_exp = static_cast<unsigned>(output_exp);
|
|
||||||
if (output_exp < 0) abs_output_exp = 0 - abs_output_exp;
|
|
||||||
int exp_digits = 2;
|
int exp_digits = 2;
|
||||||
if (abs_output_exp >= 100) exp_digits = abs_output_exp >= 1000 ? 4 : 3;
|
if (abs_output_exp >= 100) exp_digits = abs_output_exp >= 1000 ? 4 : 3;
|
||||||
|
|
||||||
size += (decimal_point ? 1 : 0) + num_zeros + 2 + exp_digits;
|
size += (decimal_point ? 1 : 0) + 2 + exp_digits;
|
||||||
char exp_char = fspecs.upper ? 'E' : 'e';
|
char exp_char = fspecs.upper ? 'E' : 'e';
|
||||||
auto write = [=](iterator it) {
|
auto write = [=](iterator it) {
|
||||||
if (sign) *it++ = static_cast<Char>(sign);
|
if (sign) *it++ = static_cast<Char>(data::signs[sign]);
|
||||||
// Insert a decimal point after the first digit and add an exponent.
|
// Insert a decimal point after the first digit and add an exponent.
|
||||||
it = write_significand(it, significand, significand_size, 1,
|
it = write_significand(it, significand, significand_size, 1,
|
||||||
decimal_point);
|
decimal_point);
|
||||||
@ -1859,7 +1860,7 @@ OutputIt write_float(OutputIt out, const DecimalFP& fp,
|
|||||||
}
|
}
|
||||||
return write_padded<align::right>(
|
return write_padded<align::right>(
|
||||||
out, specs, to_unsigned(size), [&](iterator it) {
|
out, specs, to_unsigned(size), [&](iterator it) {
|
||||||
if (sign) *it++ = static_cast<Char>(sign);
|
if (sign) *it++ = static_cast<Char>(data::signs[sign]);
|
||||||
it = write_significand<Char>(it, significand, significand_size);
|
it = write_significand<Char>(it, significand, significand_size);
|
||||||
it = std::fill_n(it, fp.exponent, zero);
|
it = std::fill_n(it, fp.exponent, zero);
|
||||||
if (!fspecs.showpoint) return it;
|
if (!fspecs.showpoint) return it;
|
||||||
@ -1872,7 +1873,7 @@ OutputIt write_float(OutputIt out, const DecimalFP& fp,
|
|||||||
size += 1 + (num_zeros > 0 ? num_zeros : 0);
|
size += 1 + (num_zeros > 0 ? num_zeros : 0);
|
||||||
return write_padded<align::right>(
|
return write_padded<align::right>(
|
||||||
out, specs, to_unsigned(size), [&](iterator it) {
|
out, specs, to_unsigned(size), [&](iterator it) {
|
||||||
if (sign) *it++ = static_cast<Char>(sign);
|
if (sign) *it++ = static_cast<Char>(data::signs[sign]);
|
||||||
it = write_significand(it, significand, significand_size, exp,
|
it = write_significand(it, significand, significand_size, exp,
|
||||||
decimal_point);
|
decimal_point);
|
||||||
return num_zeros > 0 ? std::fill_n(it, num_zeros, zero) : it;
|
return num_zeros > 0 ? std::fill_n(it, num_zeros, zero) : it;
|
||||||
@ -1887,7 +1888,7 @@ OutputIt write_float(OutputIt out, const DecimalFP& fp,
|
|||||||
size += 2 + num_zeros;
|
size += 2 + num_zeros;
|
||||||
return write_padded<align::right>(
|
return write_padded<align::right>(
|
||||||
out, specs, to_unsigned(size), [&](iterator it) {
|
out, specs, to_unsigned(size), [&](iterator it) {
|
||||||
if (sign) *it++ = static_cast<Char>(sign);
|
if (sign) *it++ = static_cast<Char>(data::signs[sign]);
|
||||||
*it++ = zero;
|
*it++ = zero;
|
||||||
if (num_zeros == 0 && significand_size == 0 && !fspecs.showpoint)
|
if (num_zeros == 0 && significand_size == 0 && !fspecs.showpoint)
|
||||||
return it;
|
return it;
|
||||||
|
Loading…
Reference in New Issue
Block a user