hoosiertransfer-mod/src/main/java/net/jafama/NumbersUtils.java

2659 lines
89 KiB
Java

/*
* Copyright 2012-2015 Jeff Hain
*
* Licensed 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 net.jafama;
/**
* Class containing various basic utility methods to deal with numbers.
* This class is meant to be light (no big look-up tables or such).
*
* Check methods return boolean if success,
* for it allows to use them in assertions.
*
* toString methods use capital letters, unlike JDK's toStrings, for it is more
* readable (especially, "l" and "1" can easily be confused with one another).
*
* Some methods have an int version additionally to the long version,
* even though long version could be used instead, for performance reasons,
* either for the methods themselves (if they do computations with ints
* instead of longs), or to be used in an int use case (like methods
* checking whether or not a signed int can fit in such number of bits).
*/
public final class NumbersUtils {
//--------------------------------------------------------------------------
// MEMBERS
//--------------------------------------------------------------------------
/**
* Double.MIN_NORMAL since Java 6.
*/
public static final double DOUBLE_MIN_NORMAL = Double.longBitsToDouble(0x0010000000000000L); // 2.2250738585072014E-308
/**
* Float.MIN_NORMAL since Java 6.
*/
public static final float FLOAT_MIN_NORMAL = Float.intBitsToFloat(0x00800000); // 1.17549435E-38f
private static final int MIN_DOUBLE_EXPONENT = -1074;
private static final int MAX_DOUBLE_EXPONENT = 1023;
/**
* All possible upper case chars for representing a number as a String.
*/
private final static char[] CHAR_BY_DIGIT;
static {
final char minDecimal = '0';
final char maxDecimal = '9';
final int n1 = maxDecimal - minDecimal + 1;
final char minLetter = 'A';
final char maxLetter = 'Z';
final int n2 = maxLetter - minLetter + 1;
CHAR_BY_DIGIT = new char[n1+n2];
int i=0;
for (char c=minDecimal;c<=maxDecimal;c++) {
CHAR_BY_DIGIT[i++] = c;
}
for (char c=minLetter;c<=maxLetter;c++) {
CHAR_BY_DIGIT[i++] = c;
}
}
/**
* For power-of-two radixes only.
*/
private static final int[] DIV_SHIFT_BY_RADIX;
static {
DIV_SHIFT_BY_RADIX = new int[32+1];
int shift=1;
for (int radix=2;radix<=32;radix*=2) {
DIV_SHIFT_BY_RADIX[radix] = shift++;
}
}
private final static int[] MAX_NBR_OF_NEG_INT_DIGITS_BY_RADIX = new int[Character.MAX_RADIX+1];
private final static int[] MAX_NBR_OF_NEG_LONG_DIGITS_BY_RADIX = new int[Character.MAX_RADIX+1];
static {
for (int radix=Character.MIN_RADIX;radix<=Character.MAX_RADIX;radix++) {
/*
* Brutal but works.
* -1 for the sign.
*/
MAX_NBR_OF_NEG_INT_DIGITS_BY_RADIX[radix] = Integer.toString(Integer.MIN_VALUE, radix).length()-1;
MAX_NBR_OF_NEG_LONG_DIGITS_BY_RADIX[radix] = Long.toString(Long.MIN_VALUE, radix).length()-1;
}
}
static final double NO_CSN_MIN_BOUND_INCL = 1e-3;
static final double NO_CSN_MAX_BOUND_EXCL = 1e7;
private static final double PIO2_HI = Double.longBitsToDouble(0x3FF921FB54400000L); // 1.57079632673412561417e+00 first 33 bits of pi/2
private static final double PIO2_LO = Double.longBitsToDouble(0x3DD0B4611A626331L); // 6.07710050650619224932e-11 pi/2 - PIO2_HI
private static final double PI_HI = 2*PIO2_HI;
private static final double PI_LO = 2*PIO2_LO;
private static final double TWOPI_HI = 4*PIO2_HI;
private static final double TWOPI_LO = 4*PIO2_LO;
//--------------------------------------------------------------------------
// PUBLIC METHODS
//--------------------------------------------------------------------------
/**
* @return True if the specified values are equal or both NaN, false otherwise.
*/
public static boolean equal(float a, float b) {
// Only does one test if a == b.
return (a == b) ? true : ((a != a) && (b != b));
}
/**
* @return True if the specified values are equal or both NaN, false otherwise.
*/
public static boolean equal(double a, double b) {
// Only does one test if a == b.
return (a == b) ? true : ((a != a) && (b != b));
}
/**
* @return True if the specified value is a mathematical integer,
* false otherwise (which includes NaN and +-Infinity).
*/
public static boolean isMathematicalInteger(float value) {
// Doing magnitude test first, for cast
// might be very slow for huge values.
// It also helps be faster for huge values,
// for which the test with cast always fail.
value = Math.abs(value);
return ((value >= (float)(1<<23)
&& (value != Float.POSITIVE_INFINITY)))
|| (value == (float)(int)value);
}
/**
* @return True if the specified value is a mathematical integer,
* false otherwise (which includes NaN and +-Infinity).
*/
public static boolean isMathematicalInteger(double value) {
// Doing magnitude test first, for cast
// might be very slow for huge values.
// It also helps be faster for huge values,
// for which the test with cast always fail.
value = Math.abs(value);
return ((value >= (double)(1L<<52))
&& (value != Double.POSITIVE_INFINITY))
|| (value == (double)(long)value);
}
/**
* @param value A float value.
* @return True if the specified value is equidistant from two adjacent
* mathematical integers, false otherwise (which includes NaN
* and +-Infinity).
*/
public static boolean isEquidistant(float value) {
if (false) {
// Also works, but slower.
final int bits = Float.floatToRawIntBits(value);
final int exponent = ((bits>>23)&0xFF)-127;
final int nbrOfPostCommaBits = 23 - exponent;
if ((nbrOfPostCommaBits <= 0) || (nbrOfPostCommaBits >= 25)) {
// No mantissa bit after comma, or all mantissa bits
// (including implicit 1) are at least one bit away from it.
//System.out.println("can't be");
return false;
}
final int mantissa = 0x00800000|(bits&0x007FFFFF);
final int postCommaMask = ~((-1)<<nbrOfPostCommaBits);
// True if in post-comma bits the only 1-bit is the one for 0.5.
return ((mantissa & postCommaMask) == (1<<(nbrOfPostCommaBits-1)));
}
final float valueAbs = Math.abs(value);
if (!(valueAbs < (float)(1<<23))) {
// NaN or too large to have a chance
return false;
}
final float twice = valueAbs+valueAbs;
// Test on twice first, for it's the most likely to fail.
return (twice == (float)(int)twice)
&& (value != (float)(int)value);
}
/**
* @param value A double value.
* @return True if the specified value is equidistant from two adjacent
* mathematical integers, false otherwise (which includes NaN
* and +-Infinity).
*/
public static boolean isEquidistant(double value) {
if (false) {
// Also works, but slower.
final long bits = Double.doubleToRawLongBits(value);
final int exponent = (((int)(bits>>52))&0x7FF)-1023;
final int nbrOfPostCommaBits = 52 - exponent;
if ((nbrOfPostCommaBits <= 0) || (nbrOfPostCommaBits >= 54)) {
// No mantissa bit after comma, or all mantissa bits
// (including implicit 1) are at least one bit away from it.
return false;
}
final long mantissa = 0x0010000000000000L|(bits&0x000FFFFFFFFFFFFFL);
final long postCommaMask = ~((-1L)<<nbrOfPostCommaBits);
// True if in post-comma bits the only 1-bit is the one for 0.5.
return ((mantissa & postCommaMask) == (1L<<(nbrOfPostCommaBits-1)));
}
final double valueAbs = Math.abs(value);
if (!(valueAbs < (double)(1L<<52))) {
return false;
}
final double twice = valueAbs+valueAbs;
// Test on twice first, for it's the most likely to fail.
return (twice == (double)(long)twice)
&& (value != (double)(long)value);
}
/**
* @param value A float value.
* @return True if the specified value is NaN or +-Infinity, false otherwise.
*/
public static boolean isNaNOrInfinite(float value) {
// value-value is not equal to 0.0f (and is NaN) <-> value is NaN or +-Infinity
return !(value-value == 0.0f);
}
/**
* @param value A double value.
* @return True if the specified value is NaN or +-Infinity, false otherwise.
*/
public static boolean isNaNOrInfinite(double value) {
// value-value is not equal to 0.0 (and is NaN) <-> value is NaN or +-Infinity
return !(value-value == 0.0);
}
/**
* @param value A float value.
* @return -1 if sign bit is 1, 1 if sign bit is 0.
*/
public static int signFromBit(float value) {
return ((Float.floatToRawIntBits(value)>>30)|1);
}
/**
* @param value A double value.
* @return -1 if sign bit is 1, 1 if sign bit is 0.
*/
public static long signFromBit(double value) {
// Returning a long, to avoid useless cast into int.
return ((Double.doubleToRawLongBits(value)>>62)|1);
}
/*
* min/max ranges
*/
/**
* @return True if the specified value is in the specified range (inclusive), false otherwise.
*/
public static boolean isInRange(int min, int max, int a) {
return (min <= a) && (a <= max);
}
/**
* @return True if the specified value is in the specified range (inclusive), false otherwise.
*/
public static boolean isInRange(long min, long max, long a) {
return (min <= a) && (a <= max);
}
/**
* Returns false if any value is NaN.
*
* @return True if the specified value is in the specified range (inclusive), false otherwise.
*/
public static boolean isInRange(float min, float max, float a) {
return (min <= a) && (a <= max);
}
/**
* Returns false if any value is NaN.
*
* @return True if the specified value is in the specified range (inclusive), false otherwise.
*/
public static boolean isInRange(double min, double max, double a) {
return (min <= a) && (a <= max);
}
/*
*
*/
/**
* @return True if does not throw.
* @throws IllegalArgumentException if the specified value is not in the specified range (inclusive).
*/
public static boolean checkIsInRange(int min, int max, int a) {
if (!isInRange(min, max, a)) {
throw new IllegalArgumentException(a+" not in ["+min+","+max+"]");
}
return true;
}
/**
* @return True if does not throw.
* @throws IllegalArgumentException if the specified value is not in the specified range (inclusive).
*/
public static boolean checkIsInRange(long min, long max, long a) {
if (!isInRange(min, max, a)) {
throw new IllegalArgumentException(a+" not in ["+min+","+max+"]");
}
return true;
}
/**
* @return True if does not throw.
* @throws IllegalArgumentException if the specified value is not in the specified range (inclusive)
* or any parameter is NaN.
*/
public static boolean checkIsInRange(float min, float max, float a) {
if (!isInRange(min, max, a)) {
throw new IllegalArgumentException(a+" not in ["+min+","+max+"]");
}
return true;
}
/**
* @return True if does not throw.
* @throws IllegalArgumentException if the specified value is not in the specified range (inclusive)
* or any parameter is NaN.
*/
public static boolean checkIsInRange(double min, double max, double a) {
if (!isInRange(min, max, a)) {
throw new IllegalArgumentException(a+" not in ["+min+","+max+"]");
}
return true;
}
/*
*
*/
/**
* @param min A value.
* @param max A value.
* @param a A value.
* @return min if a <= min, else max if a >= max, else a.
*/
public static int toRange(int min, int max, int a) {
if (a <= min) {
return min;
} else if (a >= max) {
return max;
} else {
return a;
}
}
/**
* @param min A value.
* @param max A value.
* @param a A value.
* @return min if a <= min, else max if a >= max, else a.
*/
public static long toRange(long min, long max, long a) {
if (a <= min) {
return min;
} else if (a >= max) {
return max;
} else {
return a;
}
}
/**
* @param min A value.
* @param max A value.
* @param a A value.
* @return min if a <= min, else max if a >= max, else a.
*/
public static float toRange(float min, float max, float a) {
if (a <= min) {
return min;
} else if (a >= max) {
return max;
} else {
return a;
}
}
/**
* @param min A value.
* @param max A value.
* @param a A value.
* @return min if a <= min, else max if a >= max, else a.
*/
public static double toRange(double min, double max, double a) {
if (a <= min) {
return min;
} else if (a >= max) {
return max;
} else {
return a;
}
}
/*
* bitwise ranges
*/
/**
* @param bitSize A number of bits, in [1,32].
* @return True if the specified value fits as a signed integer
* over the specified number of bits, false otherwise.
* @throws IllegalArgumentException if the specified number of bits is not in [1,32].
*/
public static boolean isInRangeSigned(int a, int bitSize) {
checkBitSizeForSignedInt(bitSize);
return (minSignedIntForBitSize_noCheck(bitSize) <= a) && (a <= maxSignedIntForBitSize_noCheck(bitSize));
}
/**
* @param bitSize A number of bits, in [1,64].
* @return True if the specified value fits as a signed integer
* over the specified number of bits, false otherwise.
* @throws IllegalArgumentException if the specified number of bits is not in [1,64].
*/
public static boolean isInRangeSigned(long a, int bitSize) {
checkBitSizeForSignedLong(bitSize);
return (minSignedLongForBitSize_noCheck(bitSize) <= a) && (a <= maxSignedLongForBitSize_noCheck(bitSize));
}
/**
* @param bitSize A number of bits, in [1,31].
* @return True if the specified value fits as an unsigned integer
* over the specified number of bits, false otherwise.
* @throws IllegalArgumentException if the specified number of bits is not in [1,31].
*/
public static boolean isInRangeUnsigned(int a, int bitSize) {
return isInRange(0, maxUnsignedIntForBitSize(bitSize), a);
}
/**
* @param bitSize A number of bits, in [1,63].
* @return True if the specified value fits as an unsigned integer
* over the specified number of bits, false otherwise.
* @throws IllegalArgumentException if the specified number of bits is not in [1,63].
*/
public static boolean isInRangeUnsigned(long a, int bitSize) {
return isInRange(0, maxUnsignedLongForBitSize(bitSize), a);
}
/*
*
*/
/**
* @param bitSize A number of bits, in [1,32].
* @return True if does not throw.
* @throws IllegalArgumentException if the specified value does not fit
* as a signed integer over the specified number of bits.
*/
public static boolean checkIsInRangeSigned(int a, int bitSize) {
if (!isInRangeSigned(a, bitSize)) {
throw new IllegalArgumentException(a+" does not fit as a signed value over "+bitSize+" bits");
}
return true;
}
/**
* @param bitSize A number of bits, in [1,64].
* @return True if does not throw.
* @throws IllegalArgumentException if the specified value does not fit
* as a signed integer over the specified number of bits.
*/
public static boolean checkIsInRangeSigned(long a, int bitSize) {
if (!isInRangeSigned(a, bitSize)) {
throw new IllegalArgumentException(a+" does not fit as a signed value over "+bitSize+" bits");
}
return true;
}
/**
* @param bitSize A number of bits, in [1,31].
* @return True if does not throw.
* @throws IllegalArgumentException if the specified value does not fit
* as an unsigned integer over the specified number of bits.
*/
public static boolean checkIsInRangeUnsigned(int a, int bitSize) {
if (!isInRangeUnsigned(a, bitSize)) {
throw new IllegalArgumentException(a+" does not fit as an unsigned value over "+bitSize+" bits");
}
return true;
}
/**
* @param bitSize A number of bits, in [1,63].
* @return True if does not throw.
* @throws IllegalArgumentException if the specified value does not fit
* as an unsigned integer over the specified number of bits.
*/
public static boolean checkIsInRangeUnsigned(long a, int bitSize) {
if (!isInRangeUnsigned(a, bitSize)) {
throw new IllegalArgumentException(a+" does not fit as an unsigned value over "+bitSize+" bits");
}
return true;
}
/*
* masks (int)
*/
/**
* @param bitSize A number of bits, in [0,32].
* @return Mask with the specified number of left bits set with 0,
* and other bits set with 1.
*/
public static int intMaskMSBits0(int bitSize) {
checkIsInRange(0, 32, bitSize);
// Shifting in two times, for >>> doesn't work for full bit size (<< as well).
final int halfish = (bitSize>>1);
return ((-1)>>>halfish)>>>(bitSize-halfish);
}
/**
* @param bitSize A number of bits, in [0,32].
* @return Mask with the specified number of left bits set with 1,
* and other bits set with 0.
*/
public static int intMaskMSBits1(int bitSize) {
return ~intMaskMSBits0(bitSize);
}
/**
* @param bitSize A number of bits, in [0,32].
* @return Mask with the specified number of right bits set with 0,
* and other bits set with 1.
*/
public static int intMaskLSBits0(int bitSize) {
return ~intMaskMSBits0(32-bitSize);
}
/**
* @param bitSize A number of bits, in [0,32].
* @return Mask with the specified number of right bits set with 1,
* and other bits set with 0.
*/
public static int intMaskLSBits1(int bitSize) {
return intMaskMSBits0(32-bitSize);
}
/*
* masks (long)
*/
/**
* @param bitSize A number of bits, in [0,64].
* @return Mask with the specified number of left bits set with 0,
* and other bits set with 1.
*/
public static long longMaskMSBits0(int bitSize) {
checkIsInRange(0, 64, bitSize);
// Shifting in two times, for >>> doesn't work for full bit size (<< as well).
final int halfish = (bitSize>>1);
return ((-1L)>>>halfish)>>>(bitSize-halfish);
}
/**
* @param bitSize A number of bits, in [0,64].
* @return Mask with the specified number of left bits set with 1,
* and other bits set with 0.
*/
public static long longMaskMSBits1(int bitSize) {
return ~longMaskMSBits0(bitSize);
}
/**
* @param bitSize A number of bits, in [0,64].
* @return Mask with the specified number of right bits set with 0,
* and other bits set with 1.
*/
public static long longMaskLSBits0(int bitSize) {
return ~longMaskMSBits0(64-bitSize);
}
/**
* @param bitSize A number of bits, in [0,64].
* @return Mask with the specified number of right bits set with 1,
* and other bits set with 0.
*/
public static long longMaskLSBits1(int bitSize) {
return longMaskMSBits0(64-bitSize);
}
/*
* signed/unsigned
*/
/**
* @return Unsigned value corresponding to bits of the specified byte.
*/
public static short byteAsUnsigned(byte value) {
return (short)(((short)value) & 0xFF);
}
/**
* @return Unsigned value corresponding to bits of the specified short.
*/
public static int shortAsUnsigned(short value) {
return ((int)value) & 0xFFFF;
}
/**
* @return Unsigned value corresponding to bits of the specified int.
*/
public static long intAsUnsigned(int value) {
return ((long)value) & 0xFFFFFFFF;
}
/*
* bitwise ranges
*/
/**
* @return True if a signed int value can be read over the specified number of bits,
* i.e. if it is in [1,32], false otherwise.
*/
public static boolean isValidBitSizeForSignedInt(int bitSize) {
return (bitSize > 0) && (bitSize <= 32);
}
/**
* @return True if a signed long value can be read over the specified number of bits,
* i.e. if it is in [1,64], false otherwise.
*/
public static boolean isValidBitSizeForSignedLong(int bitSize) {
return (bitSize > 0) && (bitSize <= 64);
}
/**
* @return True if an unsigned int value can be read over the specified number of bits,
* i.e. if it is in [1,31], false otherwise.
*/
public static boolean isValidBitSizeForUnsignedInt(int bitSize) {
return (bitSize > 0) && (bitSize < 32);
}
/**
* @return True if an unsigned long value can be read over the specified number of bits,
* i.e. if it is in [1,63], false otherwise.
*/
public static boolean isValidBitSizeForUnsignedLong(int bitSize) {
return (bitSize > 0) && (bitSize < 64);
}
/*
*
*/
/**
* @return True if does not throw.
* @throws IllegalArgumentException if a signed int value can't be read over the
* specified number of bits, i.e. if it is not in [1,32].
*/
public static boolean checkBitSizeForSignedInt(int bitSize) {
if (!isValidBitSizeForSignedInt(bitSize)) {
throw new IllegalArgumentException("bit size ["+bitSize+"] must be in [1,32] for signed int values");
}
return true;
}
/**
* @return True if does not throw.
* @throws IllegalArgumentException if a signed long value can't be read over the
* specified number of bits, i.e. if it is not in [1,64].
*/
public static boolean checkBitSizeForSignedLong(int bitSize) {
if (!isValidBitSizeForSignedLong(bitSize)) {
throw new IllegalArgumentException("bit size ["+bitSize+"] must be in [1,64] for signed long values");
}
return true;
}
/**
* @return True if does not throw.
* @throws IllegalArgumentException if an unsigned int value can't be read over the
* specified number of bits, i.e. if it is not in [1,31].
*/
public static boolean checkBitSizeForUnsignedInt(int bitSize) {
if (!isValidBitSizeForUnsignedInt(bitSize)) {
throw new IllegalArgumentException("bit size ["+bitSize+"] must be in [1,31] for unsigned int values");
}
return true;
}
/**
* @return True if does not throw.
* @throws IllegalArgumentException if an unsigned long value can't be read over the
* specified number of bits, i.e. if it is not in [1,63].
*/
public static boolean checkBitSizeForUnsignedLong(int bitSize) {
if (!isValidBitSizeForUnsignedLong(bitSize)) {
throw new IllegalArgumentException("bit size ["+bitSize+"] must be in [1,63] for unsigned long values");
}
return true;
}
/*
*
*/
/**
* @param bitSize A number of bits in [1,32].
* @return The min signed int value that can be stored over the specified number of bits.
* @throws IllegalArgumentException if the specified number of bits is out of range.
*/
public static int minSignedIntForBitSize(int bitSize) {
checkBitSizeForSignedInt(bitSize);
return minSignedIntForBitSize_noCheck(bitSize);
}
/**
* @param bitSize A number of bits in [1,64].
* @return The min signed long value that can be stored over the specified number of bits.
* @throws IllegalArgumentException if the specified number of bits is out of range.
*/
public static long minSignedLongForBitSize(int bitSize) {
checkBitSizeForSignedLong(bitSize);
return minSignedLongForBitSize_noCheck(bitSize);
}
/**
* @param bitSize A number of bits in [1,32].
* @return The max signed int value that can be stored over the specified number of bits.
* @throws IllegalArgumentException if the specified number of bits is out of range.
*/
public static int maxSignedIntForBitSize(int bitSize) {
checkBitSizeForSignedInt(bitSize);
return maxSignedIntForBitSize_noCheck(bitSize);
}
/**
* @param bitSize A number of bits in [1,64].
* @return The max signed long value that can be stored over the specified number of bits.
* @throws IllegalArgumentException if the specified number of bits is out of range.
*/
public static long maxSignedLongForBitSize(int bitSize) {
checkBitSizeForSignedLong(bitSize);
return maxSignedLongForBitSize_noCheck(bitSize);
}
/**
* @param bitSize A number of bits in [1,31].
* @return The max unsigned int value that can be stored over the specified number of bits.
* @throws IllegalArgumentException if the specified number of bits is out of range.
*/
public static int maxUnsignedIntForBitSize(int bitSize) {
checkBitSizeForUnsignedInt(bitSize);
// i.e. (1<<bitSize)-1
return (Integer.MAX_VALUE>>(31-bitSize));
}
/**
* @param bitSize A number of bits in [1,63].
* @return The max unsigned long value that can be stored over the specified number of bits.
* @throws IllegalArgumentException if the specified number of bits is out of range.
*/
public static long maxUnsignedLongForBitSize(int bitSize) {
checkBitSizeForUnsignedLong(bitSize);
// i.e. (1L<<bitSize)-1
return (Long.MAX_VALUE>>(63-bitSize));
}
/*
*
*/
/**
* @return The number of bits required to store the specified value as a signed integer,
* i.e. a result in [1,32].
*/
public static int bitSizeForSignedValue(int value) {
if (value > 0) {
return 33-Integer.numberOfLeadingZeros(value);
} else if (value == 0) {
return 1;
} else {
// Works for Integer.MIN_VALUE as well.
return 33-Integer.numberOfLeadingZeros(-value-1);
}
}
/**
* @return The number of bits required to store the specified value as a signed integer,
* i.e. a result in [1,64].
*/
public static int bitSizeForSignedValue(long value) {
if (value > 0) {
return 65-Long.numberOfLeadingZeros(value);
} else if (value == 0) {
return 1;
} else {
// Works for Long.MIN_VALUE as well.
return 65-Long.numberOfLeadingZeros(-value-1);
}
}
/**
* @param value An integer value in [0,Integer.MAX_VALUE].
* @return The number of bits required to store the specified value as an unsigned integer,
* i.e. a result in [1,31].
* @throws IllegalArgumentException if the specified value is < 0.
*/
public static int bitSizeForUnsignedValue(int value) {
if (value > 0) {
return 32-Integer.numberOfLeadingZeros(value);
} else {
if (value == 0) {
return 1;
} else {
throw new IllegalArgumentException("unsigned value ["+value+"] must be >= 0");
}
}
}
/**
* @param value An integer value in [0,Long.MAX_VALUE].
* @return The number of bits required to store the specified value as an unsigned integer,
* i.e. a result in [1,63].
* @throws IllegalArgumentException if the specified value is < 0.
*/
public static int bitSizeForUnsignedValue(long value) {
if (value > 0) {
return 64-Long.numberOfLeadingZeros(value);
} else {
if (value == 0) {
return 1;
} else {
throw new IllegalArgumentException("unsigned value ["+value+"] must be >= 0");
}
}
}
/*
* integer functions
*/
/**
* @return 1 if the specified value is > 0, 0 if it is 0, -1 otherwise.
*/
public static int signum(int a) {
return (a < 0) ? -1 : ((a == 0) ? 0 : 1);
}
/**
* @return 1 if the specified value is > 0, 0 if it is 0, -1 otherwise.
*/
public static int signum(long a) {
return (a < 0) ? -1 : ((a == 0) ? 0 : 1);
}
/**
* @return True if the specified value is even, false otherwise.
*/
public static boolean isEven(int a) {
return ((a&1) == 0);
}
/**
* @return True if the specified value is even, false otherwise.
*/
public static boolean isEven(long a) {
// faster to work on ints
return isEven((int)a);
}
/**
* @return True if the specified value is odd, false otherwise.
*/
public static boolean isOdd(int a) {
return ((a&1) != 0);
}
/**
* @return True if the specified value is odd, false otherwise.
*/
public static boolean isOdd(long a) {
// faster to work on ints
return isOdd((int)a);
}
/**
* @return True if the specified values are both even or both odd, false otherwise.
*/
public static boolean haveSameEvenness(int a, int b) {
return (((a^b)&1) == 0);
}
/**
* @return True if the specified values are both even or both odd, false otherwise.
*/
public static boolean haveSameEvenness(long a, long b) {
// faster to work on ints
return haveSameEvenness((int)a, (int)b);
}
/**
* @return True if the specified values are both >= 0 or both < 0, false otherwise.
*/
public static boolean haveSameSign(int a, int b) {
return ((a^b) >= 0);
}
/**
* @return True if the specified values are both >= 0 or both < 0, false otherwise.
*/
public static boolean haveSameSign(long a, long b) {
return ((a^b) >= 0);
}
/**
* @return True if the specified value is a power of two,
* i.e. a value of the form 2^k, with k >= 0.
*/
public static boolean isPowerOfTwo(int a) {
if (a <= 0) {
return false;
}
if (false) {
// also works
return (a & -a) == a;
}
return (a & (a-1)) == 0;
}
/**
* @return True if the specified value is a power of two,
* i.e. a value of the form 2^k, with k >= 0.
*/
public static boolean isPowerOfTwo(long a) {
if (a <= 0) {
return false;
}
if (false) {
// also works
return (a & -a) == a;
}
return (a & (a-1)) == 0;
}
/**
* @return True if the specified value is a signed power of two,
* i.e. a value of the form +-2^k, with k >= 0.
*/
public static boolean isSignedPowerOfTwo(int a) {
if (a > 0) {
return (a & (a-1)) == 0;
} else {
if (a == -a) {
// a is 0 or Integer.MIN_VALUE
return (a != 0);
}
return ((-a) & (-a-1)) == 0;
}
}
/**
* @return True if the specified value is a signed power of two,
* i.e. a value of the form +-2^k, with k >= 0.
*/
public static boolean isSignedPowerOfTwo(long a) {
if (a > 0) {
return (a & (a-1)) == 0;
} else {
if (a == -a) {
// a is 0 or Long.MIN_VALUE
return (a != 0);
}
return ((-a) & (-a-1)) == 0;
}
}
/**
* @param a A value in [1,Integer.MAX_VALUE].
* @return The highest power of two <= a.
*/
public static int floorPowerOfTwo(int a) {
if (a <= 0) {
throw new IllegalArgumentException("a ["+a+"] must be > 0");
}
return Integer.highestOneBit(a);
}
/**
* @param a A value in [1,Long.MAX_VALUE].
* @return The highest power of two <= a.
*/
public static long floorPowerOfTwo(long a) {
if (a <= 0) {
throw new IllegalArgumentException("a ["+a+"] must be > 0");
}
// Faster than copying int method
// (less computations on long).
return 1L << (63 - Long.numberOfLeadingZeros(a));
}
/**
* @param a A value in [0,2^30].
* @return The lowest power of two >= a.
*/
public static int ceilingPowerOfTwo(int a) {
checkIsInRange(0, (1<<30), a);
return (a >= 2) ? Integer.highestOneBit((a-1)<<1) : 1;
}
/**
* @param a A value in [0,2^62].
* @return The lowest power of two >= a.
*/
public static long ceilingPowerOfTwo(long a) {
checkIsInRange(0L, (1L<<62), a);
// Faster than copying int method
// (less computations on long).
return 1L << (64 - Long.numberOfLeadingZeros(a - 1));
}
/**
* @return Mean without overflow, rounded to the lowest value (i.e. mathematical floor((a+b)/2), using floating point division).
*/
public static int meanLow(int a, int b) {
return (a & b) + ((a ^ b) >> 1);
}
/**
* @return Mean without overflow, rounded to the lowest value (i.e. mathematical floor((a+b)/2), using floating point division).
*/
public static long meanLow(long a, long b) {
return (a & b) + ((a ^ b) >> 1);
}
/**
* @return Mean without overflow, rounded to the value of smallest magnitude (i.e. mathematical (a+b)/2, using integer division).
*/
public static int meanSml(int a, int b) {
int result = meanLow(a,b);
if (!haveSameEvenness(a, b)) {
// inexact
if (((a&b) < 0) || (((a|b) < 0) && (a+b < 0))) {
// both < 0, or only one is < 0 and it has the largest magnitude
result++;
}
}
return result;
}
/**
* @return Mean without overflow, rounded to the value of smallest magnitude (i.e. mathematical (a+b)/2, using integer division).
*/
public static long meanSml(long a, long b) {
long result = meanLow(a,b);
if (!haveSameEvenness(a, b)) {
// inexact
if (((a&b) < 0) || (((a|b) < 0) && (a+b < 0))) {
// both < 0, or only one is < 0 and it has the largest magnitude
result++;
}
}
return result;
}
/**
* Useful because a positive int value could not represent half the width
* of full int range width, which is mathematically Integer.MAX_VALUE+1.
*
* @return Minus half the range width (inclusive, and rounded to the value of smaller magnitude)
* between the specified bounds.
* @throws IllegalArgumentException if min > max.
*/
public static int negHalfWidth(int min, int max) {
if (min > max) {
throw new IllegalArgumentException("min ["+min+"] must be <= max ["+max+"]");
}
int mean = meanLow(min, max);
return min - mean - ((min^max)&1);
}
/**
* Useful because a positive long value could not represent half the width
* of full long range width, which is mathematically Long.MAX_VALUE+1.
*
* @return Minus half the range width (inclusive, and rounded to the value of smaller magnitude)
* between the specified bounds.
* @throws IllegalArgumentException if min > max.
*/
public static long negHalfWidth(long min, long max) {
if (min > max) {
throw new IllegalArgumentException("min ["+min+"] must be <= max ["+max+"]");
}
long mean = meanLow(min, max);
return min - mean - ((min^max)&1);
}
/**
* This treatment being designed for optimization, the fact that spot
* is a signed power of two is not checked.
*
* @param value A value.
* @param spot A signed power of two (i.e. a value of the form +-2^k, k >= 0).
* @return value % spot, i.e. a value in ]-|spot|,|spot|[.
*/
public static int moduloSignedPowerOfTwo(int value, int spot) {
if (spot == Integer.MIN_VALUE) {
return (value != Integer.MIN_VALUE) ? value : 0;
} else {
int s = (value>>31);
return ((((value+s) ^ s) & (abs(spot)-1)) + s) ^ s;
}
}
/**
* This treatment being designed for optimization, the fact that spot
* is a signed power of two is not checked.
*
* @param value A value.
* @param spot A signed power of two (i.e. a value of the form +-2^k, k >= 0).
* @return value % spot, i.e. a value in ]-|spot|,|spot|[.
*/
public static long moduloSignedPowerOfTwo(long value, long spot) {
if (spot == Long.MIN_VALUE) {
return (value != Long.MIN_VALUE) ? value : 0;
} else {
long s = (value>>63);
return ((((value+s) ^ s) & (abs(spot)-1)) + s) ^ s;
}
}
/**
* @param value An integer value > 0.
* @return The integer part of the logarithm, in base 2, of the specified value,
* i.e. a result in [0,30]
* @throws IllegalArgumentException if the specified value is <= 0.
*/
public static int log2(int value) {
if (value <= 0) {
throw new IllegalArgumentException("value ["+value+"] must be > 0");
}
return 31-Integer.numberOfLeadingZeros(value);
}
/**
* @param value An integer value > 0.
* @return The integer part of the logarithm, in base 2, of the specified value,
* i.e. a result in [0,62]
* @throws IllegalArgumentException if the specified value is <= 0.
*/
public static int log2(long value) {
if (value <= 0) {
throw new IllegalArgumentException("value ["+value+"] must be > 0");
}
return 63-Long.numberOfLeadingZeros(value);
}
/**
* Possibly faster than java.lang.Math.abs(int).
*
* @return The absolute value, except if value is Integer.MIN_VALUE, for which it returns Integer.MIN_VALUE.
*/
public static int abs(int a) {
return (a^(a>>31))-(a>>31);
}
/**
* Possibly faster than java.lang.Math.abs(long).
*
* @return The absolute value, except if value is Long.MIN_VALUE, for which it returns Long.MIN_VALUE.
*/
public static long abs(long a) {
return (a^(a>>63))-(a>>63);
}
/**
* @return The negative of the absolute value (always exact).
*/
public static int absNeg(int a) {
return (a>>31)-(a^(a>>31));
}
/**
* @return The negative of the absolute value (always exact).
*/
public static long absNeg(long a) {
return (a>>63)-(a^(a>>63));
}
/**
* If the specified value is in int range, the returned value is identical.
*
* @return An int hash of the specified value.
*/
public static int intHash(long a) {
if (false) {
// also works
int hash = ((int)(a>>32)) ^ ((int)a);
if (a < 0) {
hash = -hash-1;
}
return hash;
}
int hash = ((int)(a>>32)) + ((int)a);
if (a < 0) {
hash++;
}
return hash;
}
/**
* @param a An int value.
* @return The specified value as byte.
* @throws ArithmeticException if the specified value is not in [Byte.MIN_VALUE,Byte.MAX_VALUE] range.
*/
public static byte asByte(int a) {
if (a != (byte)a) {
throw new ArithmeticException("overflow: "+a);
}
return (byte)a;
}
/**
* @param a A long value.
* @return The specified value as int.
* @throws ArithmeticException if the specified value is not in [Integer.MIN_VALUE,Integer.MAX_VALUE] range.
*/
public static int asInt(long a) {
if (a != (int)a) {
throw new ArithmeticException("overflow: "+a);
}
return (int)a;
}
/**
* @param a A long value.
* @return The closest int value in [Integer.MIN_VALUE,Integer.MAX_VALUE] range.
*/
public static int toInt(long a) {
if (a != (int)a) {
return (a < 0) ? Integer.MIN_VALUE : Integer.MAX_VALUE;
}
return (int)a;
}
/**
* @param a An int value.
* @param b An int value.
* @return The mathematical result of a+b.
* @throws ArithmeticException if the mathematical result of a+b is not in [Integer.MIN_VALUE,Integer.MAX_VALUE] range.
*/
public static int plusExact(int a, int b) {
final int sum = a + b;
// HD 2-12 Overflow iff both arguments
// have the opposite sign of the result.
if (((a ^ sum) & (b ^ sum)) < 0) {
throw new ArithmeticException("overflow: "+a+"+"+b);
}
return sum;
}
/**
* @param a A long value.
* @param b A long value.
* @return The mathematical result of a+b.
* @throws ArithmeticException if the mathematical result of a+b is not in [Long.MIN_VALUE,Long.MAX_VALUE] range.
*/
public static long plusExact(long a, long b) {
final long sum = a + b;
// HD 2-12 Overflow iff both arguments
// have the opposite sign of the result.
if (((a ^ sum) & (b ^ sum)) < 0) {
throw new ArithmeticException("overflow: "+a+"+"+b);
}
return sum;
}
/**
* @param a An int value.
* @param b An int value.
* @return The int value of [Integer.MIN_VALUE,Integer.MAX_VALUE] range which is the closest to mathematical result of a+b.
*/
public static int plusBounded(int a, int b) {
return toInt(((long)a) + ((long)b));
}
/**
* @param a A long value.
* @param b A long value.
* @return The long value of [Long.MIN_VALUE,Long.MAX_VALUE] range which is the closest to mathematical result of a+b.
*/
public static long plusBounded(long a, long b) {
final long sum = a + b;
if (((a ^ sum) & (b ^ sum)) < 0) {
return (sum >= 0) ? Long.MIN_VALUE : Long.MAX_VALUE;
}
return sum;
}
/**
* @param a An int value.
* @param b An int value.
* @return The mathematical result of a-b.
* @throws ArithmeticException if the mathematical result of a-b is not in [Integer.MIN_VALUE,Integer.MAX_VALUE] range.
*/
public static int minusExact(int a, int b) {
final int diff = a - b;
// HD 2-12 Overflow iff the arguments have different signs and
// the sign of the result is different than the sign of "a".
if (((a ^ b) & (a ^ diff)) < 0) {
throw new ArithmeticException("integer overflow");
}
return diff;
}
/**
* @param a A long value.
* @param b A long value.
* @return The mathematical result of a-b.
* @throws ArithmeticException if the mathematical result of a-b is not in [Long.MIN_VALUE,Long.MAX_VALUE] range.
*/
public static long minusExact(long a, long b) {
final long diff = a - b;
// HD 2-12 Overflow iff the arguments have different signs and
// the sign of the result is different than the sign of "a".
if (((a ^ b) & (a ^ diff)) < 0) {
throw new ArithmeticException("integer overflow");
}
return diff;
}
/**
* @param a An int value.
* @param b An int value.
* @return The int value of [Integer.MIN_VALUE,Integer.MAX_VALUE] range which is the closest to mathematical result of a-b.
*/
public static int minusBounded(int a, int b) {
return toInt(((long)a) - ((long)b));
}
/**
* @param a A long value.
* @param b A long value.
* @return The long value of [Long.MIN_VALUE,Long.MAX_VALUE] range which is the closest to mathematical result of a-b.
*/
public static long minusBounded(long a, long b) {
final long diff = a - b;
if (((a ^ b) & (a ^ diff)) < 0) {
return (diff >= 0) ? Long.MIN_VALUE : Long.MAX_VALUE;
}
return diff;
}
/**
* @param a An int value.
* @param b An int value.
* @return The mathematical result of a*b.
* @throws ArithmeticException if the mathematical result of a*b is not in [Integer.MIN_VALUE,Integer.MAX_VALUE] range.
*/
public static int timesExact(int a, int b) {
final long prod = a * (long)b;
if (prod != (int)prod) {
throw new ArithmeticException("overflow: "+a+"*"+b);
}
return (int)prod;
}
/**
* @param a A long value.
* @param b A long value.
* @return The mathematical result of a*b.
* @throws ArithmeticException if the mathematical result of a*b is not in [Long.MIN_VALUE,Long.MAX_VALUE] range.
*/
public static long timesExact(long a, long b) {
final long prod = a * b;
final long absA = abs(a);
final long absB = abs(b);
if (((absA|absB)>>>31) != 0) {
// Some bits greater than 2^31 that might cause overflow
// Check the result using the divide operator
// and check for the special case of Long.MIN_VALUE * -1
if (((b != 0) && (prod/b != a)) ||
((a == Long.MIN_VALUE) && (b == -1))) {
throw new ArithmeticException("overflow: "+a+"*"+b);
}
}
return prod;
}
/**
* @param a An int value.
* @param b An int value.
* @return The int value of [Integer.MIN_VALUE,Integer.MAX_VALUE] range which is the closest to mathematical result of a*b.
*/
public static int timesBounded(int a, int b) {
return (int)(a * (double)b);
}
/**
* @param a A long value.
* @param b A long value.
* @return The long value of [Long.MIN_VALUE,Long.MAX_VALUE] range which is the closest to mathematical result of a*b.
*/
public static long timesBounded(long a, long b) {
final long prod = a * b;
final long absA = abs(a);
final long absB = abs(b);
if (((absA|absB)>>>31) != 0) {
// Some bits greater than 2^31 that might cause overflow
// Check the result using the divide operator
// and check for the special case of Long.MIN_VALUE * -1
if (((b != 0) && (prod/b != a)) ||
((a == Long.MIN_VALUE) && (b == -1))) {
return ((a^b) >= 0) ? Long.MAX_VALUE : Long.MIN_VALUE;
}
}
return prod;
}
/*
* powers
*/
/**
* Returns the exact result, provided it's in double range,
* i.e. if power is in [-1074,1023].
*
* @param power An int power.
* @return 2^power as a double, or +-Infinity in case of overflow.
*/
public static double twoPow(int power) {
if (power <= -MAX_DOUBLE_EXPONENT) { // Not normal.
if (power >= MIN_DOUBLE_EXPONENT) { // Subnormal.
return Double.longBitsToDouble(0x0008000000000000L>>(-(power+MAX_DOUBLE_EXPONENT)));
} else { // Underflow.
return 0.0;
}
} else if (power > MAX_DOUBLE_EXPONENT) { // Overflow.
return Double.POSITIVE_INFINITY;
} else { // Normal.
return Double.longBitsToDouble(((long)(power+MAX_DOUBLE_EXPONENT))<<52);
}
}
/**
* @param power An int power.
* @return 2^power as an int.
* @throws ArithmeticException if the mathematical result
* is not in int range, i.e. if power is not in [0,30].
*/
public static int twoPowAsIntExact(int power) {
if ((power < 0) || (power > 30)) {
throw new ArithmeticException("integer overflow");
}
return 1 << power;
}
/**
* @param power An int power.
* @return 2^power as an int, or the closest power of two in int range
* in case of overflow, i.e. if power is not in [0,30].
*/
public static int twoPowAsIntBounded(int power) {
power = toRange(0, 30, power);
return 1 << power;
}
/**
* @param power An int power.
* @return 2^power as a long.
* @throws ArithmeticException if the mathematical result
* is not in long range, i.e. if power is not in [0,62].
*/
public static long twoPowAsLongExact(int power) {
if ((power < 0) || (power > 62)) {
throw new ArithmeticException("long overflow");
}
return 1L << power;
}
/**
* @param power An int power.
* @return 2^power as a long, or the closest power of two in long range
* in case of overflow, i.e. if power is not in [0,62].
*/
public static long twoPowAsLongBounded(int power) {
power = toRange(0, 62, power);
return 1L << power;
}
/**
* @param a A value.
* @return a*a.
*/
public static int pow2(int a) {
return a*a;
}
/**
* @param a A value.
* @return a*a.
*/
public static long pow2(long a) {
return a*a;
}
/**
* @param a A value.
* @return a*a.
*/
public static float pow2(float a) {
return a*a;
}
/**
* Strict version.
*
* @param a A value.
* @return a*a.
*/
public static strictfp float pow2_strict(float a) {
return a*a;
}
/**
* @param a A value.
* @return a*a.
*/
public static double pow2(double a) {
return a*a;
}
/**
* Strict version.
*
* @param a A value.
* @return a*a.
*/
public static strictfp double pow2_strict(double a) {
return a*a;
}
/**
* @param a A value.
* @return a*a*a.
*/
public static int pow3(int a) {
return a*a*a;
}
/**
* @param a A value.
* @return a*a*a.
*/
public static long pow3(long a) {
return a*a*a;
}
/**
* @param a A value.
* @return a*a*a.
*/
public static float pow3(float a) {
return a*a*a;
}
/**
* Strict version.
*
* @param a A value.
* @return a*a*a.
*/
public static strictfp float pow3_strict(float a) {
return a*a*a;
}
/**
* @param a A value.
* @return a*a*a.
*/
public static double pow3(double a) {
return a*a*a;
}
/**
* Strict version.
*
* @param a A value.
* @return a*a*a.
*/
public static strictfp double pow3_strict(double a) {
return a*a*a;
}
/*
* Accurate +-m*PI/n.
*/
/**
* @param angRad An angle, in radians.
* @return angRad + 2*PI, accurately computed.
*/
public static double plus2PI(double angRad) {
if (angRad > -Math.PI) {
// LO then HI, for better accuracy (if starting near 0).
return (angRad + TWOPI_LO) + TWOPI_HI;
} else {
// HI then LO, for better accuracy (if ending near 0).
return (angRad + TWOPI_HI) + TWOPI_LO;
}
}
/**
* Strict version.
*
* @param angRad An angle, in radians.
* @return angRad + 2*PI, accurately computed.
*/
public static strictfp double plus2PI_strict(double angRad) {
if (angRad > -Math.PI) {
// LO then HI, for better accuracy (if starting near 0).
return (angRad + TWOPI_LO) + TWOPI_HI;
} else {
// HI then LO, for better accuracy (if ending near 0).
return (angRad + TWOPI_HI) + TWOPI_LO;
}
}
/**
* @param angRad An angle, in radians.
* @return angRad - 2*PI, accurately computed.
*/
public static double minus2PI(double angRad) {
if (angRad < Math.PI) {
// LO then HI, for better accuracy (if starting near 0).
return (angRad - TWOPI_LO) - TWOPI_HI;
} else {
// HI then LO, for better accuracy (if ending near 0).
return (angRad - TWOPI_HI) - TWOPI_LO;
}
}
/**
* Strict version.
*
* @param angRad An angle, in radians.
* @return angRad - 2*PI, accurately computed.
*/
public static strictfp double minus2PI_strict(double angRad) {
if (angRad < Math.PI) {
// LO then HI, for better accuracy (if starting near 0).
return (angRad - TWOPI_LO) - TWOPI_HI;
} else {
// HI then LO, for better accuracy (if ending near 0).
return (angRad - TWOPI_HI) - TWOPI_LO;
}
}
/**
* @param angRad An angle, in radians.
* @return angRad + PI, accurately computed.
*/
public static double plusPI(double angRad) {
if (angRad > -Math.PI/2) {
// LO then HI, for better accuracy (if starting near 0).
return (angRad + PI_LO) + PI_HI;
} else {
// HI then LO, for better accuracy (if ending near 0).
return (angRad + PI_HI) + PI_LO;
}
}
/**
* Strict version.
*
* @param angRad An angle, in radians.
* @return angRad + PI, accurately computed.
*/
public static strictfp double plusPI_strict(double angRad) {
if (angRad > -Math.PI/2) {
// LO then HI, for better accuracy (if starting near 0).
return (angRad + PI_LO) + PI_HI;
} else {
// HI then LO, for better accuracy (if ending near 0).
return (angRad + PI_HI) + PI_LO;
}
}
/**
* @param angRad An angle, in radians.
* @return angRad - PI, accurately computed.
*/
public static double minusPI(double angRad) {
if (angRad < Math.PI/2) {
// LO then HI, for better accuracy (if starting near 0).
return (angRad - PI_LO) - PI_HI;
} else {
// HI then LO, for better accuracy (if ending near 0).
return (angRad - PI_HI) - PI_LO;
}
}
/**
* Strict version.
*
* @param angRad An angle, in radians.
* @return angRad - PI, accurately computed.
*/
public static strictfp double minusPI_strict(double angRad) {
if (angRad < Math.PI/2) {
// LO then HI, for better accuracy (if starting near 0).
return (angRad - PI_LO) - PI_HI;
} else {
// HI then LO, for better accuracy (if ending near 0).
return (angRad - PI_HI) - PI_LO;
}
}
/**
* @param angRad An angle, in radians.
* @return angRad + PI/2, accurately computed.
*/
public static double plusPIO2(double angRad) {
if (angRad > -Math.PI/4) {
// LO then HI, for better accuracy (if starting near 0).
return (angRad + PIO2_LO) + PIO2_HI;
} else {
// HI then LO, for better accuracy (if ending near 0).
return (angRad + PIO2_HI) + PIO2_LO;
}
}
/**
* Strict version.
*
* @param angRad An angle, in radians.
* @return angRad + PI/2, accurately computed.
*/
public static strictfp double plusPIO2_strict(double angRad) {
if (angRad > -Math.PI/4) {
// LO then HI, for better accuracy (if starting near 0).
return (angRad + PIO2_LO) + PIO2_HI;
} else {
// HI then LO, for better accuracy (if ending near 0).
return (angRad + PIO2_HI) + PIO2_LO;
}
}
/**
* @param angRad An angle, in radians.
* @return angRad - PI/2, accurately computed.
*/
public static double minusPIO2(double angRad) {
if (angRad < Math.PI/4) {
// LO then HI, for better accuracy (if starting near 0).
return (angRad - PIO2_LO) - PIO2_HI;
} else {
// HI then LO, for better accuracy (if ending near 0).
return (angRad - PIO2_HI) - PIO2_LO;
}
}
/**
* Strict version.
*
* @param angRad An angle, in radians.
* @return angRad - PI/2, accurately computed.
*/
public static strictfp double minusPIO2_strict(double angRad) {
if (angRad < Math.PI/4) {
// LO then HI, for better accuracy (if starting near 0).
return (angRad - PIO2_LO) - PIO2_HI;
} else {
// HI then LO, for better accuracy (if ending near 0).
return (angRad - PIO2_HI) - PIO2_LO;
}
}
/*
* toString (radix)
*/
/**
* @param radix Radix to be checked.
* @return True if does not throw.
* @throws IllegalArgumentException if the specified radix is not in [2,36].
*/
public static boolean checkRadix(int radix) {
if (!isInRange(Character.MIN_RADIX, Character.MAX_RADIX, radix)) {
throw new IllegalArgumentException("radix ["+radix+"] must be in ["+Character.MIN_RADIX+","+Character.MAX_RADIX+"]");
}
return true;
}
/**
* @param radix A radix in [2,36].
* @return Number of characters (minus sign included)
* to represent the specified value in the specified radix.
*/
public static int computeNbrOfChars(int value, int radix) {
if (value < 0) {
// 1 for sign
return 1 + computeNbrOfDigits_negValue(value, radix);
} else {
return computeNbrOfDigits_negValue(-value, radix);
}
}
/**
* @param radix A radix in [2,36].
* @return Number of characters (minus sign included)
* to represent the specified value in the specified radix.
*/
public static int computeNbrOfChars(long value, int radix) {
if (value < 0) {
// 1 for sign
return 1 + computeNbrOfDigits_negValue(value, radix);
} else {
return computeNbrOfDigits_negValue(-value, radix);
}
}
/**
* @param radix A radix in [2,36].
* @param paddingUpTo Number of digits (sign excluded) up to which left-padding with zeros is done.
* @return Number of characters (minus sign included)
* to represent the specified value in the specified radix.
*/
public static int computeNbrOfChars(int value, int radix, int paddingUpTo) {
if (value < 0) {
// 1 for sign
return 1 + Math.max(paddingUpTo, computeNbrOfDigits_negValue(value, radix));
} else {
return Math.max(paddingUpTo, computeNbrOfDigits_negValue(-value, radix));
}
}
/**
* @param radix A radix in [2,36].
* @param paddingUpTo Number of digits (sign excluded) up to which left-padding with zeros is done.
* @return Number of characters (minus sign included)
* to represent the specified value in the specified radix.
*/
public static int computeNbrOfChars(long value, int radix, int paddingUpTo) {
if (value < 0) {
// 1 for sign
return 1 + Math.max(paddingUpTo, computeNbrOfDigits_negValue(value, radix));
} else {
return Math.max(paddingUpTo, computeNbrOfDigits_negValue(-value, radix));
}
}
/**
* @param radix A radix in [2,36].
* @return Number of digits of the specified value in the specified radix.
*/
public static int computeNbrOfDigits(int value, int radix) {
return computeNbrOfDigits_negValue(-abs(value), radix);
}
/**
* @param radix A radix in [2,36].
* @return Number of digits of the specified value in the specified radix.
*/
public static int computeNbrOfDigits(long value, int radix) {
return computeNbrOfDigits_negValue(-abs(value), radix);
}
/**
* @param radix A radix in [2,36].
* @param paddingUpTo Number of digits (sign excluded) up to which left-padding with zeros is done.
* @return Number of digits of the specified value in the specified radix,
* including the specified padding.
*/
public static int computeNbrOfDigits(int value, int radix, int paddingUpTo) {
return Math.max(paddingUpTo,computeNbrOfDigits(value, radix));
}
/**
* @param radix A radix in [2,36].
* @param paddingUpTo Number of digits (sign excluded) up to which left-padding with zeros is done.
* @return Number of digits of the specified value in the specified radix,
* including the specified padding.
*/
public static int computeNbrOfDigits(long value, int radix, int paddingUpTo) {
return Math.max(paddingUpTo,computeNbrOfDigits(value, radix));
}
/**
* This method just delegates to Integer.toString(int),
* but is defined here to complete the API.
*
* @return String representation of the specified value in base 10.
*/
public static String toString(int value) {
return Integer.toString(value);
}
/**
* This method just delegates to Long.toString(long),
* but is defined here to complete the API.
*
* @return String representation of the specified value in base 10.
*/
public static String toString(long value) {
return Long.toString(value);
}
/**
* @param radix A radix in [2,36].
* @return String representation of the specified value in the specified radix.
* @throws IllegalArgumentException if the specified radix is out of range.
*/
public static String toString(int value, int radix) {
return toString(value, radix, 0);
}
/**
* @param radix A radix in [2,36].
* @return String representation of the specified value in the specified radix.
* @throws IllegalArgumentException if the specified radix is out of range.
*/
public static String toString(long value, int radix) {
return toString(value, radix, 0);
}
/**
* @param radix A radix in [2,36].
* @param paddingUpTo Number of digits (sign excluded) up to which left-padding with zeros is done.
* @return String representation of the specified value in the specified radix.
* @throws IllegalArgumentException if the specified radix is out of range.
*/
public static String toString(int value, int radix, int paddingUpTo) {
// Only one test if radix+paddingUpTo != 10.
if ((radix+paddingUpTo == 10) && (paddingUpTo == 0)) {
// Using JDK's optimized algorithm.
return Integer.toString(value);
}
int negValue;
final int signSize;
final boolean negative = (value < 0);
if (negative) {
negValue = value;
signSize = 1;
} else {
negValue = -value;
signSize = 0;
}
// Faster if we just use max possible number of characters (33),
// but we prefer to take care of garbage's memory footprint.
// Checks radix.
final int nbrOfChars = signSize + Math.max(paddingUpTo, computeNbrOfDigits_negValue(negValue, radix));
final char[] chars = new char[nbrOfChars];
int charPos = nbrOfChars;
final boolean radixIsPowerOfTwo = ((radix & (radix-1)) == 0);
// Not allowing Integer.MIN_VALUE so it can be negated.
if (radixIsPowerOfTwo && (negValue != Integer.MIN_VALUE)) {
final int mask = radix-1;
final int divShift = DIV_SHIFT_BY_RADIX[radix];
while (negValue <= -radix) {
chars[--charPos] = CHAR_BY_DIGIT[(int)((-negValue) & mask)];
negValue = -((-negValue) >> divShift);
}
} else {
while (negValue <= -radix) {
chars[--charPos] = CHAR_BY_DIGIT[(int)(-(negValue % radix))];
negValue /= radix;
}
}
chars[--charPos] = CHAR_BY_DIGIT[(int)(-negValue)];
while (charPos > signSize) {
chars[--charPos] = '0';
}
if (negative) {
chars[0] = '-';
}
return new String(chars);
}
/**
* @param radix A radix in [2,36].
* @param paddingUpTo Number of digits (sign excluded) up to which left-padding with zeros is done.
* @return String representation of the specified value in the specified radix.
* @throws IllegalArgumentException if the specified radix is out of range.
*/
public static String toString(long value, int radix, int paddingUpTo) {
// Only one test if radix+paddingUpTo != 10.
if ((radix+paddingUpTo == 10) && (paddingUpTo == 0)) {
// Using JDK's optimized algorithm.
return Long.toString(value);
}
long negValue;
final int signSize;
final boolean negative = (value < 0);
if (negative) {
negValue = value;
signSize = 1;
} else {
negValue = -value;
signSize = 0;
}
// Checks radix.
final int nbrOfChars = signSize + Math.max(paddingUpTo, computeNbrOfDigits_negValue(negValue, radix));
final char[] chars = new char[nbrOfChars];
int charPos = nbrOfChars;
final boolean radixIsPowerOfTwo = ((radix & (radix-1)) == 0);
// Not allowing Long.MIN_VALUE so it can be negated.
if (radixIsPowerOfTwo && (negValue != Long.MIN_VALUE)) {
final int mask = radix-1;
final int divShift = DIV_SHIFT_BY_RADIX[radix];
while (negValue <= -radix) {
chars[--charPos] = CHAR_BY_DIGIT[(int)((-negValue) & mask)];
negValue = -((-negValue) >> divShift);
}
} else {
while (negValue <= -radix) {
chars[--charPos] = CHAR_BY_DIGIT[(int)(-(negValue % radix))];
negValue /= radix;
}
}
chars[--charPos] = CHAR_BY_DIGIT[(int)(-negValue)];
while (charPos > signSize) {
chars[--charPos] = '0';
}
if (negative) {
chars[0] = '-';
}
return new String(chars);
}
/*
* toString (bits)
*/
/**
* @param firstBitPos First bit position (inclusive).
* @param lastBitPosExcl Last bit position (exclusive).
* @return True if does not throw.
* @throws IllegalArgumentException if the specified bit range does not fit in a byte.
*/
public static boolean checkBitPositionsByte(int firstBitPos, int lastBitPosExcl) {
return checkBitPositions(firstBitPos, lastBitPosExcl, 8);
}
/**
* @param firstBitPos First bit position (inclusive).
* @param lastBitPosExcl Last bit position (exclusive).
* @return True if does not throw.
* @throws IllegalArgumentException if the specified bit range does not fit in a short.
*/
public static boolean checkBitPositionsShort(int firstBitPos, int lastBitPosExcl) {
return checkBitPositions(firstBitPos, lastBitPosExcl, 16);
}
/**
* @param firstBitPos First bit position (inclusive).
* @param lastBitPosExcl Last bit position (exclusive).
* @return True if does not throw.
* @throws IllegalArgumentException if the specified bit range does not fit in an int.
*/
public static boolean checkBitPositionsInt(int firstBitPos, int lastBitPosExcl) {
return checkBitPositions(firstBitPos, lastBitPosExcl, 32);
}
/**
* @param firstBitPos First bit position (inclusive).
* @param lastBitPosExcl Last bit position (exclusive).
* @return True if does not throw.
* @throws IllegalArgumentException if the specified bit range does not fit in a long.
*/
public static boolean checkBitPositionsLong(int firstBitPos, int lastBitPosExcl) {
return checkBitPositions(firstBitPos, lastBitPosExcl, 64);
}
/**
* @return String representation of specified bits, in big endian.
*/
public static String toStringBits(byte bits) {
final char[] chars = new char[8];
int bitIndex = 8;
while (--bitIndex >= 0) {
chars[7-bitIndex] = (char)('0'+((bits>>bitIndex)&1));
}
return new String(chars);
}
/**
* @return String representation of specified bits, in big endian.
*/
public static String toStringBits(short bits) {
final char[] chars = new char[16];
int bitIndex = 16;
while (--bitIndex >= 0) {
chars[15-bitIndex] = (char)('0'+((bits>>bitIndex)&1));
}
return new String(chars);
}
/**
* @return String representation of specified bits, in big endian.
*/
public static String toStringBits(int bits) {
final char[] chars = new char[32];
int bitIndex = 32;
while (--bitIndex >= 0) {
chars[31-bitIndex] = (char)('0'+((bits>>bitIndex)&1));
}
return new String(chars);
}
/**
* @return String representation of specified bits, in big endian.
*/
public static String toStringBits(long bits) {
final char[] chars = new char[64];
int bitIndex = 64;
while (--bitIndex >= 0) {
chars[63-bitIndex] = (char)('0'+((bits>>bitIndex)&1));
}
return new String(chars);
}
/**
* @param firstBitPos First bit position (inclusive).
* @param lastBitPosExcl Last bit position (exclusive).
* @param bigEndian True for bits to be added in big endian order (MSBit to LSBit)
* false for little endian order.
* @param padding True if underscores must be added instead of out-of-range bits,
* false to just add characters corresponding to in-range bits.
* @return String representation of specified bits.
*/
public static String toStringBits(
byte bits,
int firstBitPos,
int lastBitPosExcl,
boolean bigEndian,
boolean padding) {
checkBitPositionsByte(firstBitPos, lastBitPosExcl);
return toStringBits_0_32_bitPosAlreadyChecked(8,bits, firstBitPos, lastBitPosExcl, bigEndian, padding);
}
/**
* @param firstBitPos First bit position (inclusive).
* @param lastBitPosExcl Last bit position (exclusive).
* @param bigEndian True for bits to be added in big endian order (MSBit to LSBit)
* false for little endian order.
* @param padding True if underscores must be added instead of out-of-range bits,
* false to just add characters corresponding to in-range bits.
* @return String representation of specified bits.
*/
public static String toStringBits(
short bits,
int firstBitPos,
int lastBitPosExcl,
boolean bigEndian,
boolean padding) {
checkBitPositionsShort(firstBitPos, lastBitPosExcl);
return toStringBits_0_32_bitPosAlreadyChecked(16,bits, firstBitPos, lastBitPosExcl, bigEndian, padding);
}
/**
* @param firstBitPos First bit position (inclusive).
* @param lastBitPosExcl Last bit position (exclusive).
* @param bigEndian True for bits to be added in big endian order (MSBit to LSBit)
* false for little endian order.
* @param padding True if underscores must be added instead of out-of-range bits,
* false to just add characters corresponding to in-range bits.
* @return String representation of specified bits.
*/
public static String toStringBits(
int bits,
int firstBitPos,
int lastBitPosExcl,
boolean bigEndian,
boolean padding) {
checkBitPositionsInt(firstBitPos, lastBitPosExcl);
return toStringBits_0_32_bitPosAlreadyChecked(32,bits, firstBitPos, lastBitPosExcl, bigEndian, padding);
}
/**
* @param firstBitPos First bit position (inclusive).
* @param lastBitPosExcl Last bit position (exclusive).
* @param bigEndian True for bits to be added in big endian order (MSBit to LSBit)
* false for little endian order.
* @param padding True if underscores must be added instead of out-of-range bits,
* false to just add characters corresponding to in-range bits.
* @return String representation of specified bits.
*/
public static String toStringBits(
long bits,
int firstBitPos,
int lastBitPosExcl,
boolean bigEndian,
boolean padding) {
checkBitPositionsLong(firstBitPos, lastBitPosExcl);
final int bitSize = 64;
final int bitSizeM1 = bitSize-1;
final int lastBitPos = lastBitPosExcl-1;
if (padding) {
final int nbrOfChars = bitSize;
final char[] chars = new char[nbrOfChars];
int bitIndex = bitSizeM1;
if (bigEndian) {
final int firstBitIndex = bitSizeM1-lastBitPos;
final int lastBitIndex = bitSizeM1-firstBitPos;
while (bitIndex > lastBitIndex) {
chars[bitSizeM1-bitIndex] = '_';
--bitIndex;
}
while (bitIndex >= firstBitIndex) {
chars[bitSizeM1-bitIndex] = (char)('0'+((bits>>bitIndex)&1));
--bitIndex;
}
while (bitIndex >= 0) {
chars[bitSizeM1-bitIndex] = '_';
--bitIndex;
}
} else {
while (bitIndex > lastBitPos) {
chars[bitIndex] = '_';
--bitIndex;
}
while (bitIndex >= firstBitPos) {
chars[bitIndex] = (char)('0'+((bits>>bitIndex)&1));
--bitIndex;
}
while (bitIndex >= 0) {
chars[bitIndex] = '_';
--bitIndex;
}
}
return new String(chars);
} else {
final int nbrOfChars = (lastBitPosExcl - firstBitPos);
final char[] chars = new char[nbrOfChars];
if (bigEndian) {
final int firstBitIndex = bitSizeM1-lastBitPos;
final int lastBitIndex = bitSizeM1-firstBitPos;
int bitIndex = lastBitIndex;
while (bitIndex >= firstBitIndex) {
chars[lastBitIndex-bitIndex] = (char)('0'+((bits>>bitIndex)&1));
--bitIndex;
}
} else {
int bitIndex = lastBitPos;
while (bitIndex >= firstBitPos) {
chars[bitIndex-firstBitPos] = (char)('0'+((bits>>bitIndex)&1));
--bitIndex;
}
}
return new String(chars);
}
}
/*
* toString (floating points)
*
* toStringCSN(double) and toStringNoCSN(double)
* could be made faster, by using directly internals
* of Double.toString(double), but this would require
* copy-paste of much tricky code from JDK, and
* the overhead of our little rework is relatively
* negligible.
*/
/**
* @param value A double value.
* @return String representing the specified value,
* using "computerized scientific notation",
* which Double.toString(double) uses for non-infinite
* values, when |value| < 1e-3 or |value| >= 1e7.
*/
public static String toStringCSN(double value) {
// Quick case (also to get rid of +-0.0,
// for which Double.toString(double) doesn't use CSN).
if (value == 0.0) {
if (Double.doubleToRawLongBits(value) < 0) {
return "-0.0E0";
} else {
return "0.0E0";
}
}
final double abs = Math.abs(value);
if ((abs >= NO_CSN_MIN_BOUND_INCL) && (abs < NO_CSN_MAX_BOUND_EXCL)) {
final boolean neg = (value < 0.0);
final String rawAbs = Double.toString(abs);
if (abs >= 1.0) {
/*
* 0123456
* 12.3456 ===> 1.23456E1
* 123.0 ===> 1.23E2
*/
final int dotIndex = rawAbs.indexOf((int)'.');
final int powerOfTen = dotIndex-1;
final StringBuilder sb = new StringBuilder();
if (neg) {
sb.append('-');
}
// Adding unit-or-above digits, with dot after first one.
sb.append(rawAbs.charAt(0));
sb.append('.');
sb.append(rawAbs,1,dotIndex);
if ((value != (int)value) || (abs < 10.0)) {
// Adding below-unit digits (possibly just 0 if abs < 10.0,
// to end up for example with "3.0E0" instead of "3.E0").
sb.append(rawAbs,dotIndex+1,rawAbs.length());
}
sb.append('E');
sb.append(CHAR_BY_DIGIT[powerOfTen]);
return sb.toString();
} else {
/*
* 012345678
* 0.0123456 ===> 1.23456E-2
* 0.01 ===> 1.0E-2
*/
int nonZeroIndex = 1;
while (rawAbs.charAt(++nonZeroIndex) == '0') {
}
// Negative.
final int powerOfTen = 1-nonZeroIndex;
final int nbrOfSignificantDigitsPastDot = (rawAbs.length() - (nonZeroIndex+1));
final StringBuilder sb = new StringBuilder();
if (neg) {
sb.append('-');
}
sb.append(rawAbs.charAt(nonZeroIndex));
sb.append('.');
if (nbrOfSignificantDigitsPastDot > 0) {
// If bug 4428022 make rawAbs being something like "0.0010",
// we add the last '0' here after the dot, which is fine.
sb.append(rawAbs,nonZeroIndex+1,rawAbs.length());
} else {
sb.append('0');
}
sb.append("E-");
sb.append(CHAR_BY_DIGIT[-powerOfTen]);
return sb.toString();
}
} else {
return Double.toString(value);
}
}
/**
* @param value A double value.
* @return String representing the specified value,
* not in "computerized scientific notation",
* which Double.toString(double) uses for non-infinite
* values, when |value| < 1e-3 or |value| >= 1e7.
*/
public static String toStringNoCSN(double value) {
// Quick case.
// Should also work with long instead of int,
// but less obvious (due to roundings...),
// and we just want to speed up the more common
// case of "small" integer values.
final int intValue = (int)value;
if (value == intValue) {
if (value == 0.0) {
if (Double.doubleToRawLongBits(value) < 0) {
return "-0.0";
} else {
return "0.0";
}
} else {
return Integer.toString(intValue)+".0";
}
}
final String raw = Double.toString(value);
final double abs = Math.abs(value);
if (abs >= NO_CSN_MAX_BOUND_EXCL) {
if (abs == Double.POSITIVE_INFINITY) {
return raw;
}
/*
* 0123456789
* 1.234567E5 ===> 123456.7
* 1.23456E5 ===> 123456.0 (adding 0)
* 1.23E5 ===> 123000.0
* 1.0E5 ===> 100000.0
*/
// "." close to start, so using indexOf.
final int dotIndex = raw.indexOf((int)'.');
// "E" close to end, so using lastIndexOf.
final int eIndex = raw.lastIndexOf((int)'E');
final int powerOfTen = Integer.parseInt(raw.substring(eIndex+1));
final int nbrOfSignificantLoDigits = (eIndex - dotIndex - 1);
final int nbrOfZerosToAddBeforeDot = (powerOfTen - nbrOfSignificantLoDigits);
int start;
int end;
final StringBuilder sb = new StringBuilder();
sb.append(raw,0,dotIndex);
if (nbrOfZerosToAddBeforeDot >= 0) {
// Can copy all digits that were between '.' and 'E'.
sb.append(raw,dotIndex+1,eIndex);
for (int i=0;i<nbrOfZerosToAddBeforeDot;i++) {
sb.append('0');
}
sb.append(".0");
} else {
start = dotIndex+1;
sb.append(raw,start,end = start+powerOfTen);
sb.append('.');
start = end;
sb.append(raw,start,end = eIndex);
}
return sb.toString();
} else if (abs < NO_CSN_MIN_BOUND_INCL) {
// Not +-0.0 since already handled.
/*
* 01234567
* 1.234E-4 ===> 0.0001234
* 1.0E-4 ===> 0.0001
*/
// "." close to start, so using indexOf.
final int dotIndex = raw.indexOf((int)'.');
// "E" close to end, so using lastIndexOf.
final int eIndex = raw.lastIndexOf((int)'E');
// Negative.
final int powerOfTen = Integer.parseInt(raw.substring(eIndex+1));
final int nbrOfZerosToAddAfterDot = (-powerOfTen-1);
final StringBuilder sb = new StringBuilder();
if (value < 0.0) {
sb.append("-0.");
} else {
sb.append("0.");
}
for (int i=0;i<nbrOfZerosToAddAfterDot;i++) {
sb.append('0');
}
// First raw digit.
sb.append(raw,dotIndex-1,dotIndex);
if ((eIndex == dotIndex + 2) && (raw.charAt(dotIndex+1) == '0')) {
// Char past dot is alone and '0': no need to add it.
} else {
// Raw digits that were past dot.
sb.append(raw,dotIndex+1,eIndex);
}
return sb.toString();
} else {
// abs in [0.001,1e7[.
if ((abs < 1.0) && (raw.charAt(raw.length()-1) == '0')) {
// Workaround for bug 4428022 (Double.toString(0.004) returns
// "0.0040", same with 0.001 etc.).
return raw.substring(0, raw.length()-1);
} else {
return raw;
}
}
}
//--------------------------------------------------------------------------
// PRIVATE METHODS
//--------------------------------------------------------------------------
private NumbersUtils() {
}
/*
*
*/
/**
* Had such isInXXX methods, and corresponding checkXXX methods,
* but they seem actually slower in practice, so just keeping this
* code here in case some day it becomes faster than regular isInXXX.
*
* Only works for non-empty ranges, i.e. such as min <= max.
* This treatment being designed for optimization, min <= max
* is not checked.
*
* @return True if the specified value is in the specified range (inclusive), false otherwise.
*/
private static boolean dontUseMe_isInNonEmptyRange_(int min, int max, int a) {
// Using modulo arithmetic.
return (Integer.MIN_VALUE+(a-min) <= Integer.MIN_VALUE+(max-min));
}
/*
*
*/
private static int minSignedIntForBitSize_noCheck(int bitSize) {
// i.e. (-1<<(bitSize-1))
return (Integer.MIN_VALUE>>(32-bitSize));
}
private static long minSignedLongForBitSize_noCheck(int bitSize) {
// i.e. (-1L<<(bitSize-1))
return (Long.MIN_VALUE>>(64-bitSize));
}
private static int maxSignedIntForBitSize_noCheck(int bitSize) {
// i.e. (1<<(bitSize-1))-1
return (Integer.MAX_VALUE>>(32-bitSize));
}
private static long maxSignedLongForBitSize_noCheck(int bitSize) {
// i.e. (1L<<(bitSize-1))-1
return (Long.MAX_VALUE>>(64-bitSize));
}
/*
*
*/
/**
* @throws IllegalArgumentException if the specified radix is out of range.
*/
private static int computeNbrOfDigits_negValue(int negValue, int radix) {
checkRadix(radix);
final int maxNbrOfDigits = MAX_NBR_OF_NEG_INT_DIGITS_BY_RADIX[radix];
int p = radix;
for (int i=1;i<maxNbrOfDigits;i++) {
if (negValue > -p) {
return i;
}
p *= radix;
}
return maxNbrOfDigits;
}
/**
* @throws IllegalArgumentException if the specified radix is out of range.
*/
private static int computeNbrOfDigits_negValue(long negValue, int radix) {
checkRadix(radix);
final int maxNbrOfDigits = MAX_NBR_OF_NEG_LONG_DIGITS_BY_RADIX[radix];
long p = radix;
for (int i=1;i<maxNbrOfDigits;i++) {
if (negValue > -p) {
return i;
}
p *= radix;
}
return maxNbrOfDigits;
}
/*
*
*/
private static boolean checkBitPositions(int firstBitPos, int lastBitPosExcl, int bitSize) {
if ((firstBitPos < 0) || (firstBitPos > lastBitPosExcl) || (lastBitPosExcl > bitSize)) {
throw new IllegalArgumentException(
"bit positions (first="+firstBitPos+",lastExcl="+lastBitPosExcl
+") must verify 0 <= first <= lastExcl <= "+bitSize);
}
return true;
}
/**
* Common method for byte, short and int.
* Could be a bit faster to have specific methods for byte and short,
* but not much, and that would also make more messy (byte-)code.
*
* @param bitSize Must be in [0,32].
*/
private static String toStringBits_0_32_bitPosAlreadyChecked(
int bitSize,
int bits,
int firstBitPos,
int lastBitPosExcl,
boolean bigEndian,
boolean padding) {
final int bitSizeM1 = bitSize-1;
final int lastBitPos = lastBitPosExcl-1;
if (padding) {
final int nbrOfChars = bitSize;
final char[] chars = new char[nbrOfChars];
int bitIndex = bitSizeM1;
if (bigEndian) {
final int firstBitIndex = bitSizeM1-lastBitPos;
final int lastBitIndex = bitSizeM1-firstBitPos;
while (bitIndex > lastBitIndex) {
chars[bitSizeM1-bitIndex] = '_';
--bitIndex;
}
while (bitIndex >= firstBitIndex) {
chars[bitSizeM1-bitIndex] = (char)('0'+((bits>>bitIndex)&1));
--bitIndex;
}
while (bitIndex >= 0) {
chars[bitSizeM1-bitIndex] = '_';
--bitIndex;
}
} else {
while (bitIndex > lastBitPos) {
chars[bitIndex] = '_';
--bitIndex;
}
while (bitIndex >= firstBitPos) {
chars[bitIndex] = (char)('0'+((bits>>bitIndex)&1));
--bitIndex;
}
while (bitIndex >= 0) {
chars[bitIndex] = '_';
--bitIndex;
}
}
return new String(chars);
} else {
final int nbrOfChars = (lastBitPosExcl - firstBitPos);
final char[] chars = new char[nbrOfChars];
if (bigEndian) {
final int firstBitIndex = bitSizeM1-lastBitPos;
final int lastBitIndex = bitSizeM1-firstBitPos;
int bitIndex = lastBitIndex;
while (bitIndex >= firstBitIndex) {
chars[lastBitIndex-bitIndex] = (char)('0'+((bits>>bitIndex)&1));
--bitIndex;
}
} else {
int bitIndex = lastBitPos;
while (bitIndex >= firstBitPos) {
chars[bitIndex-firstBitPos] = (char)('0'+((bits>>bitIndex)&1));
--bitIndex;
}
}
return new String(chars);
}
}
}