com.fasterxml.jackson.core.io.NumberInput Maven / Gradle / Ivy
package com.fasterxml.jackson.core.io;
import ch.randelshofer.fastdoubleparser.JavaDoubleParser;
import ch.randelshofer.fastdoubleparser.JavaFloatParser;
import java.math.BigDecimal;
import java.math.BigInteger;
import java.util.regex.Pattern;
/**
* Helper class for efficient parsing of various JSON numbers.
*
* NOTE! Does NOT validate against maximum length limits: caller must
* do that if and as necessary.
*/
public final class NumberInput
{
/**
* Formerly used constant for a value that was problematic on certain
* pre-1.8 JDKs.
*
* @deprecated Since 2.14 -- do not use
*/
@Deprecated // since 2.14
public final static String NASTY_SMALL_DOUBLE = "2.2250738585072012e-308";
/**
* Constants needed for parsing longs from basic int parsing methods
*/
final static long L_BILLION = 1000000000;
final static String MIN_LONG_STR_NO_SIGN = String.valueOf(Long.MIN_VALUE).substring(1);
final static String MAX_LONG_STR = String.valueOf(Long.MAX_VALUE);
/**
* Regexp used to pre-validate "Stringified Numbers": slightly looser than
* JSON Number definition (allows leading zeroes, positive sign).
*
* @since 2.17
*/
private final static Pattern PATTERN_FLOAT = Pattern.compile(
"[+-]?[0-9]*[\\.]?[0-9]+([eE][+-]?[0-9]+)?");
/**
* Secondary regexp used along with {@code PATTERN_FLOAT} to cover
* case where number ends with dot, like {@code "+12."}
*
* @since 2.17.2
*/
private final static Pattern PATTERN_FLOAT_TRAILING_DOT = Pattern.compile(
"[+-]?[0-9]+[\\.]");
/**
* Fast method for parsing unsigned integers that are known to fit into
* regular 32-bit signed int type. This means that length is
* between 1 and 9 digits (inclusive) and there is no sign character.
*
* Note: public to let unit tests call it; not meant to be used by any
* code outside this package.
*
* @param ch Buffer that contains integer value to decode
* @param off Offset of the first digit character in buffer
* @param len Length of the number to decode (in characters)
*
* @return Decoded {@code int} value
*/
public static int parseInt(char[] ch, int off, int len)
{
if (len > 0 && ch[off] == '+') {
off++;
len--;
}
int num = ch[off + len - 1] - '0';
switch(len) {
case 9:
num += (ch[off++] - '0') * 100000000;
case 8:
num += (ch[off++] - '0') * 10000000;
case 7:
num += (ch[off++] - '0') * 1000000;
case 6:
num += (ch[off++] - '0') * 100000;
case 5:
num += (ch[off++] - '0') * 10000;
case 4:
num += (ch[off++] - '0') * 1000;
case 3:
num += (ch[off++] - '0') * 100;
case 2:
num += (ch[off] - '0') * 10;
}
return num;
}
/**
* Helper method to (more) efficiently parse integer numbers from
* String values. Input String must be simple Java integer value.
* No range checks are made to verify that the value fits in 32-bit Java {@code int}:
* caller is expected to only calls this in cases where this can be guaranteed
* (basically: number of digits does not exceed 9)
*
* NOTE: semantics differ significantly from {@link #parseInt(char[], int, int)}.
*
* @param s String that contains integer value to decode
*
* @return Decoded {@code int} value
*/
public static int parseInt(String s)
{
/* Ok: let's keep strategy simple: ignoring optional minus sign,
* we'll accept 1 - 9 digits and parse things efficiently;
* otherwise just defer to JDK parse functionality.
*/
char c = s.charAt(0);
int len = s.length();
boolean neg = (c == '-');
int offset = 1;
// must have 1 - 9 digits after optional sign:
// negative?
if (neg) {
if (len == 1 || len > 10) {
return Integer.parseInt(s);
}
c = s.charAt(offset++);
} else {
if (len > 9) {
return Integer.parseInt(s);
}
}
if (c > '9' || c < '0') {
return Integer.parseInt(s);
}
int num = c - '0';
if (offset < len) {
c = s.charAt(offset++);
if (c > '9' || c < '0') {
return Integer.parseInt(s);
}
num = (num * 10) + (c - '0');
if (offset < len) {
c = s.charAt(offset++);
if (c > '9' || c < '0') {
return Integer.parseInt(s);
}
num = (num * 10) + (c - '0');
// Let's just loop if we have more than 3 digits:
if (offset < len) {
do {
c = s.charAt(offset++);
if (c > '9' || c < '0') {
return Integer.parseInt(s);
}
num = (num * 10) + (c - '0');
} while (offset < len);
}
}
}
return neg ? -num : num;
}
public static long parseLong(char[] ch, int off, int len)
{
// Note: caller must ensure length is [10, 18]
int len1 = len-9;
long val = parseInt(ch, off, len1) * L_BILLION;
return val + parseInt(ch, off+len1, 9);
}
/**
* Parses an unsigned long made up of exactly 19 digits.
*
* It is the callers responsibility to make sure the input is exactly 19 digits.
* and fits into a 64bit long by calling {@link #inLongRange(char[], int, int, boolean)}
* first.
*
* Note that input String must NOT contain leading minus sign (even
* if {@code negative} is set to true).
*
* @param ch Buffer that contains integer value to decode
* @param off Offset of the first digit character in buffer
* @param negative Whether original number had a minus sign
* @return Decoded {@code long} value
*
* @since 2.15.0
*/
public static long parseLong19(char[] ch, int off, boolean negative)
{
// Note: caller must ensure length is 19
long num = 0L;
for (int i = 0; i < 19; i++) {
char c = ch[off + i];
num = (num * 10) + (c - '0');
}
return negative ? -num : num;
}
/**
* Similar to {@link #parseInt(String)} but for {@code long} values.
*
* @param s String that contains {@code long} value to decode
*
* @return Decoded {@code long} value
*/
public static long parseLong(String s)
{
// Ok, now; as the very first thing, let's just optimize case of "fake longs";
// that is, if we know they must be ints, call int parsing
int length = s.length();
if (length <= 9) {
return parseInt(s);
}
// !!! TODO: implement efficient 2-int parsing...
return Long.parseLong(s);
}
/**
* Helper method for determining if given String representation of
* an integral number would fit in 64-bit Java long or not.
* Note that input String must NOT contain leading minus sign (even
* if 'negative' is set to true).
*
* @param ch Buffer that contains long value to check
* @param off Offset of the first digit character in buffer
* @param len Length of the number to decode (in characters)
* @param negative Whether original number had a minus sign (which is
* NOT passed to this method) or not
*
* @return {@code True} if specified String representation is within Java
* {@code long} range; {@code false} if not.
*/
public static boolean inLongRange(char[] ch, int off, int len,
boolean negative)
{
String cmpStr = negative ? MIN_LONG_STR_NO_SIGN : MAX_LONG_STR;
int cmpLen = cmpStr.length();
if (len < cmpLen) return true;
if (len > cmpLen) return false;
for (int i = 0; i < cmpLen; ++i) {
int diff = ch[off+i] - cmpStr.charAt(i);
if (diff != 0) {
return (diff < 0);
}
}
return true;
}
/**
* Similar to {@link #inLongRange(char[],int,int,boolean)}, but
* with String argument
*
* @param s String that contains {@code long} value to check
* @param negative Whether original number had a minus sign (which is
* NOT passed to this method) or not
*
* @return {@code True} if specified String representation is within Java
* {@code long} range; {@code false} if not.
*/
public static boolean inLongRange(String s, boolean negative)
{
String cmp = negative ? MIN_LONG_STR_NO_SIGN : MAX_LONG_STR;
int cmpLen = cmp.length();
int alen = s.length();
if (alen < cmpLen) return true;
if (alen > cmpLen) return false;
// could perhaps just use String.compareTo()?
for (int i = 0; i < cmpLen; ++i) {
int diff = s.charAt(i) - cmp.charAt(i);
if (diff != 0) {
return (diff < 0);
}
}
return true;
}
public static int parseAsInt(String s, int def)
{
if (s == null) {
return def;
}
s = s.trim();
int len = s.length();
if (len == 0) {
return def;
}
// One more thing: use integer parsing for 'simple'
int i = 0;
// skip leading sign, if any
final char sign = s.charAt(0);
if (sign == '+') { // for plus, actually physically remove
s = s.substring(1);
len = s.length();
} else if (sign == '-') { // minus, just skip for checks, must retain
i = 1;
}
for (; i < len; ++i) {
char c = s.charAt(i);
// if other symbols, parse as Double, coerce
if (c > '9' || c < '0') {
try {
//useFastParser=true is used because there is a lot less risk that small changes in result will have an affect
//and performance benefit is useful
return (int) parseDouble(s, true);
} catch (NumberFormatException e) {
return def;
}
}
}
try {
return Integer.parseInt(s);
} catch (NumberFormatException e) { }
return def;
}
public static long parseAsLong(String s, long def)
{
if (s == null) {
return def;
}
s = s.trim();
int len = s.length();
if (len == 0) {
return def;
}
// One more thing: use long parsing for 'simple'
int i = 0;
// skip leading sign, if any
final char sign = s.charAt(0);
if (sign == '+') { // for plus, actually physically remove
s = s.substring(1);
len = s.length();
} else if (sign == '-') { // minus, just skip for checks, must retain
i = 1;
}
for (; i < len; ++i) {
char c = s.charAt(i);
// if other symbols, parse as Double, coerce
if (c > '9' || c < '0') {
try {
//useFastParser=true is used because there is a lot less risk that small changes in result will have an affect
//and performance benefit is useful
return (long) parseDouble(s, true);
} catch (NumberFormatException e) {
return def;
}
}
}
try {
return Long.parseLong(s);
} catch (NumberFormatException e) { }
return def;
}
/**
* @param s a string representing a number to parse
* @param def the default to return if `s` is not a parseable number
* @return closest matching double (or `def` if there is an issue with `s`) where useFastParser=false
* @see #parseAsDouble(String, double, boolean)
*/
public static double parseAsDouble(final String s, final double def)
{
return parseAsDouble(s, def, false);
}
/**
* @param s a string representing a number to parse
* @param def the default to return if `s` is not a parseable number
* @param useFastParser whether to use {@code FastDoubleParser}
* @return closest matching double (or `def` if there is an issue with `s`)
* @since 2.14
*/
public static double parseAsDouble(String s, final double def, final boolean useFastParser)
{
if (s == null) { return def; }
s = s.trim();
if (s.isEmpty()) {
return def;
}
try {
return parseDouble(s, useFastParser);
} catch (NumberFormatException e) { }
return def;
}
/**
* @param s a string representing a number to parse
* @return closest matching double
* @throws NumberFormatException if string cannot be represented by a double where useFastParser=false
* @see #parseDouble(String, boolean)
*
* @deprecated Since 2.17 use {@link #parseDouble(String, boolean)} instead
*/
@Deprecated // since 2.17
public static double parseDouble(final String s) throws NumberFormatException {
return parseDouble(s, false);
}
/**
* @param s a string representing a number to parse
* @param useFastParser whether to use {@code FastDoubleParser}
* @return closest matching double
* @throws NumberFormatException if string cannot be represented by a double
* @since v2.14
*/
public static double parseDouble(final String s, final boolean useFastParser) throws NumberFormatException {
return useFastParser ? JavaDoubleParser.parseDouble(s) : Double.parseDouble(s);
}
/**
* @param s a string representing a number to parse
* @return closest matching float
* @throws NumberFormatException if string cannot be represented by a float where useFastParser=false
* @see #parseFloat(String, boolean)
* @since v2.14
*
* @deprecated Since 2.17 use {@link #parseFloat(String, boolean)} instead
*/
@Deprecated // since 2.17
public static float parseFloat(final String s) throws NumberFormatException {
return parseFloat(s, false);
}
/**
* @param s a string representing a number to parse
* @param useFastParser whether to use {@code FastDoubleParser}
* @return closest matching float
* @throws NumberFormatException if string cannot be represented by a float
* @since v2.14
*/
public static float parseFloat(final String s, final boolean useFastParser) throws NumberFormatException {
if (useFastParser) {
return JavaFloatParser.parseFloat(s);
}
return Float.parseFloat(s);
}
/**
* @param s a string representing a number to parse
* @return a BigDecimal
* @throws NumberFormatException if the char array cannot be represented by a BigDecimal
*
* @deprecated Since 2.17 use {@link #parseBigDecimal(String, boolean)} instead
*/
@Deprecated // since 2.17
public static BigDecimal parseBigDecimal(final String s) throws NumberFormatException {
return parseBigDecimal(s, false);
}
/**
* @param s a string representing a number to parse
* @param useFastParser whether to use custom fast parser (true) or JDK default (false) parser
* @return a BigDecimal
* @throws NumberFormatException if the char array cannot be represented by a BigDecimal
* @since v2.15
*/
public static BigDecimal parseBigDecimal(final String s, final boolean useFastParser) throws NumberFormatException {
if (useFastParser) {
return BigDecimalParser.parseWithFastParser(s);
}
return BigDecimalParser.parse(s);
}
/**
* @param ch a char array with text that makes up a number
* @param off the offset to apply when parsing the number in the char array
* @param len the length of the number in the char array
* @return a BigDecimal
* @throws NumberFormatException if the char array cannot be represented by a BigDecimal
*
* @deprecated Since 2.17 use {@link #parseBigDecimal(char[], int, int, boolean)} instead
*/
@Deprecated // since 2.17
public static BigDecimal parseBigDecimal(final char[] ch, final int off, final int len) throws NumberFormatException {
return BigDecimalParser.parse(ch, off, len);
}
/**
* @param ch a char array with text that makes up a number
* @param off the offset to apply when parsing the number in the char array
* @param len the length of the number in the char array
* @param useFastParser whether to use custom fast parser (true) or JDK default (false) parser
* @return a BigDecimal
* @throws NumberFormatException if the char array cannot be represented by a BigDecimal
* @since v2.15
*/
public static BigDecimal parseBigDecimal(final char[] ch, final int off, final int len,
final boolean useFastParser)
throws NumberFormatException
{
if (useFastParser) {
return BigDecimalParser.parseWithFastParser(ch, off, len);
}
return BigDecimalParser.parse(ch, off, len);
}
/**
* @param ch a char array with text that makes up a number
* @return a BigDecimal
* @throws NumberFormatException if the char array cannot be represented by a BigDecimal
*
* @deprecated Since 2.17 use {@link #parseBigDecimal(char[], boolean)} instead
*/
@Deprecated // since 2.17
public static BigDecimal parseBigDecimal(final char[] ch) throws NumberFormatException {
return BigDecimalParser.parse(ch);
}
/**
* @param ch a char array with text that makes up a number
* @param useFastParser whether to use custom fast parser (true) or JDK default (false) parser
* @return a BigDecimal
* @throws NumberFormatException if the char array cannot be represented by a BigDecimal
* @since v2.15
*/
public static BigDecimal parseBigDecimal(final char[] ch, final boolean useFastParser) throws NumberFormatException {
return useFastParser ?
BigDecimalParser.parseWithFastParser(ch, 0, ch.length) :
BigDecimalParser.parse(ch);
}
/**
* @param s a string representing a number to parse
* @return a BigInteger
* @throws NumberFormatException if string cannot be represented by a BigInteger
* @since v2.14
*
* @deprecated Since 2.17 use {@link #parseBigInteger(String, boolean)} instead
*/
@Deprecated // since 2.17
public static BigInteger parseBigInteger(final String s) throws NumberFormatException {
return parseBigInteger(s, false);
}
/**
* @param s a string representing a number to parse
* @param useFastParser whether to use custom fast parser (true) or JDK default (false) parser
* @return a BigInteger
* @throws NumberFormatException if string cannot be represented by a BigInteger
* @since v2.15
*/
public static BigInteger parseBigInteger(final String s, final boolean useFastParser) throws NumberFormatException {
if (useFastParser) {
return BigIntegerParser.parseWithFastParser(s);
}
return new BigInteger(s);
}
/**
* @param s a string representing a number to parse
* @param radix for parse
* @param useFastParser whether to use custom fast parser (true) or JDK default (false) parser
* @return a BigInteger
* @throws NumberFormatException if string cannot be represented by a BigInteger
* @since v2.15
*/
public static BigInteger parseBigIntegerWithRadix(final String s, final int radix,
final boolean useFastParser) throws NumberFormatException {
if (useFastParser) {
return BigIntegerParser.parseWithFastParser(s, radix);
}
return new BigInteger(s, radix);
}
/**
* Method called to check whether given pattern looks like a valid Java
* Number (which is bit looser definition than valid JSON Number).
* Used as pre-parsing check when parsing "Stringified numbers".
*
* The differences to stricter JSON Number are:
*
* - Positive sign is allowed
*
* - Leading zeroes are allowed
*
*
*
* Note: no trimming ({@code String.trim()}) nor null checks are performed
* on String passed.
*
* Note: this method returning {@code true} DOES NOT GUARANTEE String is valid
* number but just that it looks close enough.
*
* @param s String to validate
*
* @return True if String looks like valid Java number; false otherwise.
*
* @since 2.17
*/
public static boolean looksLikeValidNumber(final String s) {
// While PATTERN_FLOAT handles most cases we can optimize some simple ones:
if (s == null || s.isEmpty()) {
return false;
}
if (s.length() == 1) {
char c = s.charAt(0);
return (c <= '9') && (c >= '0');
}
return PATTERN_FLOAT.matcher(s).matches()
|| PATTERN_FLOAT_TRAILING_DOT.matcher(s).matches();
}
}