FastDecimalFormatter.java
/* Copyright 2022-2025 Thales Alenia Space
* Licensed to CS GROUP (CS) under one or more
* contributor license agreements. See the NOTICE file distributed with
* this work for additional information regarding copyright ownership.
* CS licenses this file to You under the Apache License, Version 2.0
* (the "License"); you may not use this file except in compliance with
* the License. You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package org.orekit.utils.formatting;
import org.hipparchus.util.FastMath;
import org.orekit.errors.OrekitException;
import org.orekit.errors.OrekitMessages;
import java.io.IOException;
/** Formatter for double numbers with low overhead.
* <p>
* This class is intended to be used when formatting large amounts of data with
* fixed formats like, for example, large ephemeris or measurement files.
* </p>
* <p>
* Building the formatter is done once, and the formatter
* {@link #appendTo(Appendable, double)} or {@link #toString(double)} methods can
* be called hundreds of thousands of times, without incurring the overhead that
* would occur with {@code String.format()}. Some tests showed this formatter is
* about 5 times faster than {@code String.format()} with
* {@code %{width}.{%precision}f} format.
* </p>
* <p>
* Instances of this class are immutable
* </p>
* @author Luc Maisonobe
* @since 13.0.3
*/
public class FastDecimalFormatter extends FastDoubleFormatter {
/** Scaling array for fractional part. */
private static final long[] SCALING = new long[19];
static {
SCALING[0] = 1L;
for (int i = 1; i < SCALING.length; ++i) {
SCALING[i] = 10L * SCALING[i - 1];
}
}
/** Precision. */
private final int precision;
/** Scaling. */
private final long scaling;
/** Formatter for integer part. */
private final FastLongFormatter beforeFormatter;
/** Formatter for fractional part. */
private final FastLongFormatter afterFormatter;
/** Simple constructor.
* <p>
* This constructor is equivalent to {@link java.util.Formatter Formatter}
* float format {@code %{width}.{precision}f}
* </p>
* @param width number of characters to output
* @param precision number of decimal precision
*/
public FastDecimalFormatter(final int width, final int precision) {
super(width);
this.precision = precision;
if (width <= 0 || width > SCALING.length || precision < 0 || precision > width) {
throw new OrekitException(OrekitMessages.INVALID_FORMAT, width, precision);
}
this.scaling = SCALING[precision];
this.beforeFormatter = precision == 0 ?
new FastLongFormatter(width, false) :
new FastLongFormatter(width - precision - 1, false);
this.afterFormatter = new FastLongFormatter(precision, true);
}
/** Get the precision.
* @return precision
*/
public int getPrecision() {
return precision;
}
/** {@inheritDoc} */
@Override
protected void appendRegularValueTo(final Appendable appendable, final double value) throws IOException {
final double abs = FastMath.abs(value);
double before = FastMath.floor(abs);
long after = FastMath.round((abs - before) * scaling);
if (after >= scaling) {
// we have to round up to the next integer
before += 1;
after = 0L;
}
// convert to string
final double sign = FastMath.copySign(1.0, value);
if (sign < 0 && before == 0.0) {
// special case for negative values between -0.0 and -1.0
for (int i = 0; i < beforeFormatter.getWidth() - 2; ++i) {
appendable.append(' ');
}
appendable.append("-0");
} else {
// regular case
beforeFormatter.appendTo(appendable, FastMath.round(sign * before));
}
// fractional part
if (scaling > 1) {
appendable.append('.');
afterFormatter.appendTo(appendable, after);
}
}
}