Improvements to formula evaluation treatment of -0.0. (Refinements to fix for bug 47198

git-svn-id: https://svn.apache.org/repos/asf/poi/trunk@797258 13f79535-47bb-0310-9956-ffa450edef68
This commit is contained in:
Josh Micich 2009-07-23 23:12:17 +00:00
parent a080338801
commit b7ba153dcb
8 changed files with 226 additions and 59 deletions

View File

@ -34,14 +34,17 @@ public final class PercentEval implements OperationEval {
if (args.length != 1) { if (args.length != 1) {
return ErrorEval.VALUE_INVALID; return ErrorEval.VALUE_INVALID;
} }
double d0; double d;
try { try {
ValueEval ve = OperandResolver.getSingleValue(args[0], srcRow, srcCol); ValueEval ve = OperandResolver.getSingleValue(args[0], srcRow, srcCol);
d0 = OperandResolver.coerceValueToDouble(ve); d = OperandResolver.coerceValueToDouble(ve);
} catch (EvaluationException e) { } catch (EvaluationException e) {
return e.getErrorEval(); return e.getErrorEval();
} }
return new NumberEval(d0 / 100); if (d == 0.0) { // this '==' matches +0.0 and -0.0
return NumberEval.ZERO;
}
return new NumberEval(d / 100);
} }
public int getNumberOfOperands() { public int getNumberOfOperands() {

View File

@ -19,7 +19,7 @@ package org.apache.poi.hssf.record.formula.eval;
/** /**
* Base class for all comparison operator evaluators * Base class for all comparison operator evaluators
* *
* @author Amol S. Deshmukh < amolweb at ya hoo dot com > * @author Amol S. Deshmukh < amolweb at ya hoo dot com >
*/ */
public abstract class RelationalOperationEval implements OperationEval { public abstract class RelationalOperationEval implements OperationEval {
@ -108,10 +108,7 @@ public abstract class RelationalOperationEval implements OperationEval {
if (vb instanceof NumberEval) { if (vb instanceof NumberEval) {
NumberEval nA = (NumberEval) va; NumberEval nA = (NumberEval) va;
NumberEval nB = (NumberEval) vb; NumberEval nB = (NumberEval) vb;
if (nA.getNumberValue() == nB.getNumberValue()) { // Excel considers -0.0 < 0.0 which is the same as Double.compare()
// Excel considers -0.0 == 0.0 which is different to Double.compare()
return 0;
}
return Double.compare(nA.getNumberValue(), nB.getNumberValue()); return Double.compare(nA.getNumberValue(), nB.getNumberValue());
} }
} }

View File

@ -26,13 +26,19 @@ abstract class TwoOperandNumericOperation implements OperationEval {
ValueEval ve = OperandResolver.getSingleValue(arg, srcCellRow, srcCellCol); ValueEval ve = OperandResolver.getSingleValue(arg, srcCellRow, srcCellCol);
return OperandResolver.coerceValueToDouble(ve); return OperandResolver.coerceValueToDouble(ve);
} }
public final Eval evaluate(Eval[] args, int srcCellRow, short srcCellCol) { public final Eval evaluate(Eval[] args, int srcCellRow, short srcCellCol) {
double result; double result;
try { try {
double d0 = singleOperandEvaluate(args[0], srcCellRow, srcCellCol); double d0 = singleOperandEvaluate(args[0], srcCellRow, srcCellCol);
double d1 = singleOperandEvaluate(args[1], srcCellRow, srcCellCol); double d1 = singleOperandEvaluate(args[1], srcCellRow, srcCellCol);
result = evaluate(d0, d1); result = evaluate(d0, d1);
if (result == 0.0) { // this '==' matches +0.0 and -0.0
// Excel converts -0.0 to +0.0 for '*', '/', '%', '+' and '^'
if (!(this instanceof SubtractEval)) {
return NumberEval.ZERO;
}
}
if (Double.isNaN(result) || Double.isInfinite(result)) { if (Double.isNaN(result) || Double.isInfinite(result)) {
return ErrorEval.NUM_ERROR; return ErrorEval.NUM_ERROR;
} }

View File

@ -20,7 +20,7 @@ package org.apache.poi.hssf.record.formula.eval;
/** /**
* @author Amol S. Deshmukh &lt; amolweb at ya hoo dot com &gt; * @author Amol S. Deshmukh &lt; amolweb at ya hoo dot com &gt;
* *
*/ */
public final class UnaryMinusEval implements OperationEval { public final class UnaryMinusEval implements OperationEval {
@ -41,6 +41,9 @@ public final class UnaryMinusEval implements OperationEval {
} catch (EvaluationException e) { } catch (EvaluationException e) {
return e.getErrorEval(); return e.getErrorEval();
} }
if (d == 0.0) { // this '==' matches +0.0 and -0.0
return NumberEval.ZERO;
}
return new NumberEval(-d); return new NumberEval(-d);
} }

View File

@ -26,16 +26,16 @@ import java.math.BigInteger;
* <ul> * <ul>
* <li>No more than 15 significant figures are output (java does 18).</li> * <li>No more than 15 significant figures are output (java does 18).</li>
* <li>The sign char for the exponent is included even if positive</li> * <li>The sign char for the exponent is included even if positive</li>
* <li>Special values (<tt>NaN</tt> and <tt>Infinity</tt>) get rendered like the ordinary * <li>Special values (<tt>NaN</tt> and <tt>Infinity</tt>) get rendered like the ordinary
* number that the bit pattern represents.</li> * number that the bit pattern represents.</li>
* <li>Denormalised values (between &plusmn;2<sup>-1074</sup> and &plusmn;2<sup>-1022</sup> * <li>Denormalised values (between &plusmn;2<sup>-1074</sup> and &plusmn;2<sup>-1022</sup>
* are displayed as "0"</sup> * are displayed as "0"</sup>
* </ul> * </ul>
* IEEE 64-bit Double Rendering Comparison * IEEE 64-bit Double Rendering Comparison
* *
* <table border="1" cellpadding="2" cellspacing="0" summary="IEEE 64-bit Double Rendering Comparison"> * <table border="1" cellpadding="2" cellspacing="0" summary="IEEE 64-bit Double Rendering Comparison">
* <tr><th>Raw bits</th><th>Java</th><th>Excel</th></tr> * <tr><th>Raw bits</th><th>Java</th><th>Excel</th></tr>
* *
* <tr><td>0x0000000000000000L</td><td>0.0</td><td>0</td></tr> * <tr><td>0x0000000000000000L</td><td>0.0</td><td>0</td></tr>
* <tr><td>0x3FF0000000000000L</td><td>1.0</td><td>1</td></tr> * <tr><td>0x3FF0000000000000L</td><td>1.0</td><td>1</td></tr>
* <tr><td>0x3FF00068DB8BAC71L</td><td>1.0001</td><td>1.0001</td></tr> * <tr><td>0x3FF00068DB8BAC71L</td><td>1.0001</td><td>1.0001</td></tr>
@ -96,8 +96,8 @@ import java.math.BigInteger;
* <tr><td>0x7FFFFFFFFFFFFFFFL</td><td>NaN</td><td>3.5953862697246E+308</td></tr> * <tr><td>0x7FFFFFFFFFFFFFFFL</td><td>NaN</td><td>3.5953862697246E+308</td></tr>
* <tr><td>0xFFF7FFFFFFFFFFFFL</td><td>NaN</td><td>2.6965397022935E+308</td></tr> * <tr><td>0xFFF7FFFFFFFFFFFFL</td><td>NaN</td><td>2.6965397022935E+308</td></tr>
* </table> * </table>
* *
* <b>Note</b>: * <b>Note</b>:
* Excel has inconsistent rules for the following numeric operations: * Excel has inconsistent rules for the following numeric operations:
* <ul> * <ul>
* <li>Conversion to string (as handled here)</li> * <li>Conversion to string (as handled here)</li>
@ -105,10 +105,10 @@ import java.math.BigInteger;
* <li>Conversion from text</li> * <li>Conversion from text</li>
* <li>General arithmetic</li> * <li>General arithmetic</li>
* </ul> * </ul>
* Excel's text to number conversion is not a true <i>inverse</i> of this operation. The * Excel's text to number conversion is not a true <i>inverse</i> of this operation. The
* allowable ranges are different. Some numbers that don't correctly convert to text actually * allowable ranges are different. Some numbers that don't correctly convert to text actually
* <b>do</b> get handled properly when used in arithmetic evaluations. * <b>do</b> get handled properly when used in arithmetic evaluations.
* *
* @author Josh Micich * @author Josh Micich
*/ */
public final class NumberToTextConverter { public final class NumberToTextConverter {
@ -119,49 +119,49 @@ public final class NumberToTextConverter {
private static final int FRAC_BITS_WIDTH = EXPONENT_SHIFT; private static final int FRAC_BITS_WIDTH = EXPONENT_SHIFT;
private static final int EXPONENT_BIAS = 1023; private static final int EXPONENT_BIAS = 1023;
private static final long FRAC_ASSUMED_HIGH_BIT = ( 1L<<EXPONENT_SHIFT ); private static final long FRAC_ASSUMED_HIGH_BIT = ( 1L<<EXPONENT_SHIFT );
private static final long EXCEL_NAN_BITS = 0xFFFF0420003C0000L; private static final long EXCEL_NAN_BITS = 0xFFFF0420003C0000L;
private static final int MAX_TEXT_LEN = 20; private static final int MAX_TEXT_LEN = 20;
private static final int DEFAULT_COUNT_SIGNIFICANT_DIGITS = 15; private static final int DEFAULT_COUNT_SIGNIFICANT_DIGITS = 15;
private static final int MAX_EXTRA_ZEROS = MAX_TEXT_LEN - DEFAULT_COUNT_SIGNIFICANT_DIGITS; private static final int MAX_EXTRA_ZEROS = MAX_TEXT_LEN - DEFAULT_COUNT_SIGNIFICANT_DIGITS;
private static final float LOG2_10 = 3.32F; private static final float LOG2_10 = 3.32F;
private NumberToTextConverter() { private NumberToTextConverter() {
// no instances of this class // no instances of this class
} }
/** /**
* Converts the supplied <tt>value</tt> to the text representation that Excel would give if * Converts the supplied <tt>value</tt> to the text representation that Excel would give if
* the value were to appear in an unformatted cell, or as a literal number in a formula.<br/> * the value were to appear in an unformatted cell, or as a literal number in a formula.<br/>
* Note - the results from this method differ slightly from those of <tt>Double.toString()</tt> * Note - the results from this method differ slightly from those of <tt>Double.toString()</tt>
* In some special cases Excel behaves quite differently. This function attempts to reproduce * In some special cases Excel behaves quite differently. This function attempts to reproduce
* those results. * those results.
*/ */
public static String toText(double value) { public static String toText(double value) {
return rawDoubleBitsToText(Double.doubleToLongBits(value)); return rawDoubleBitsToText(Double.doubleToLongBits(value));
} }
/* package */ static String rawDoubleBitsToText(long pRawBits) { /* package */ static String rawDoubleBitsToText(long pRawBits) {
long rawBits = pRawBits; long rawBits = pRawBits;
boolean isNegative = rawBits < 0; // sign bit is in the same place for long and double boolean isNegative = rawBits < 0; // sign bit is in the same place for long and double
if (isNegative) { if (isNegative) {
rawBits &= 0x7FFFFFFFFFFFFFFFL; rawBits &= 0x7FFFFFFFFFFFFFFFL;
} }
int biasedExponent = (int) ((rawBits & expMask) >> EXPONENT_SHIFT); int biasedExponent = (int) ((rawBits & expMask) >> EXPONENT_SHIFT);
if (biasedExponent == 0) { if (biasedExponent == 0) {
// value is 'denormalised' which means it is less than 2^-1022 // value is 'denormalised' which means it is less than 2^-1022
// excel displays all these numbers as zero, even though calculations work OK // excel displays all these numbers as zero, even though calculations work OK
return "0"; return isNegative ? "-0" : "0";
} }
int exponent = biasedExponent - EXPONENT_BIAS; int exponent = biasedExponent - EXPONENT_BIAS;
long fracBits = FRAC_ASSUMED_HIGH_BIT | rawBits & FRAC_MASK; long fracBits = FRAC_ASSUMED_HIGH_BIT | rawBits & FRAC_MASK;
// Start by converting double value to BigDecimal // Start by converting double value to BigDecimal
BigDecimal bd; BigDecimal bd;
if (biasedExponent == 0x07FF) { if (biasedExponent == 0x07FF) {
@ -175,26 +175,26 @@ public final class NumberToTextConverter {
isNegative = false; // except that the sign bit is ignored isNegative = false; // except that the sign bit is ignored
} }
bd = convertToBigDecimal(exponent, fracBits); bd = convertToBigDecimal(exponent, fracBits);
return formatBigInteger(isNegative, bd.unscaledValue(), bd.scale()); return formatBigInteger(isNegative, bd.unscaledValue(), bd.scale());
} }
private static BigDecimal convertToBigDecimal(int exponent, long fracBits) { private static BigDecimal convertToBigDecimal(int exponent, long fracBits) {
byte[] joob = { byte[] joob = {
(byte) (fracBits >> 48), (byte) (fracBits >> 48),
(byte) (fracBits >> 40), (byte) (fracBits >> 40),
(byte) (fracBits >> 32), (byte) (fracBits >> 32),
(byte) (fracBits >> 24), (byte) (fracBits >> 24),
(byte) (fracBits >> 16), (byte) (fracBits >> 16),
(byte) (fracBits >> 8), (byte) (fracBits >> 8),
(byte) (fracBits >> 0), (byte) (fracBits >> 0),
}; };
BigInteger bigInt = new BigInteger(joob); BigInteger bigInt = new BigInteger(joob);
int lastSigBitIndex = exponent-FRAC_BITS_WIDTH; int lastSigBitIndex = exponent-FRAC_BITS_WIDTH;
if(lastSigBitIndex < 0) { if(lastSigBitIndex < 0) {
BigInteger shifto = new BigInteger("1").shiftLeft(-lastSigBitIndex); BigInteger shifto = new BigInteger("1").shiftLeft(-lastSigBitIndex);
int scale = 1 -(int) (lastSigBitIndex/LOG2_10); int scale = 1 -(int) (lastSigBitIndex/LOG2_10);
BigDecimal bd1 = new BigDecimal(bigInt); BigDecimal bd1 = new BigDecimal(bigInt);
BigDecimal bdShifto = new BigDecimal(shifto); BigDecimal bdShifto = new BigDecimal(shifto);
return bd1.divide(bdShifto, scale, BigDecimal.ROUND_HALF_UP); return bd1.divide(bdShifto, scale, BigDecimal.ROUND_HALF_UP);
@ -208,10 +208,10 @@ public final class NumberToTextConverter {
if (scale < 0) { if (scale < 0) {
throw new RuntimeException("negative scale"); throw new RuntimeException("negative scale");
} }
StringBuffer sb = new StringBuffer(unscaledValue.toString()); StringBuffer sb = new StringBuffer(unscaledValue.toString());
int numberOfLeadingZeros = -1; int numberOfLeadingZeros = -1;
int unscaledLength = sb.length(); int unscaledLength = sb.length();
if (scale > 0 && scale >= unscaledLength) { if (scale > 0 && scale >= unscaledLength) {
// less than one // less than one
@ -226,7 +226,7 @@ public final class NumberToTextConverter {
} }
return sb.toString(); return sb.toString();
} }
private static int getNumberOfSignificantFiguresDisplayed(int exponent) { private static int getNumberOfSignificantFiguresDisplayed(int exponent) {
int nLostDigits; // number of significand digits lost due big exponents int nLostDigits; // number of significand digits lost due big exponents
if(exponent > 99) { if(exponent > 99) {
@ -241,19 +241,19 @@ public final class NumberToTextConverter {
} }
return DEFAULT_COUNT_SIGNIFICANT_DIGITS - nLostDigits; return DEFAULT_COUNT_SIGNIFICANT_DIGITS - nLostDigits;
} }
private static boolean needsScientificNotation(int nDigits) { private static boolean needsScientificNotation(int nDigits) {
return nDigits > MAX_TEXT_LEN; return nDigits > MAX_TEXT_LEN;
} }
private static void formatGreaterThanOne(StringBuffer sb, int nIntegerDigits) { private static void formatGreaterThanOne(StringBuffer sb, int nIntegerDigits) {
int maxSigFigs = getNumberOfSignificantFiguresDisplayed(nIntegerDigits); int maxSigFigs = getNumberOfSignificantFiguresDisplayed(nIntegerDigits);
int decimalPointIndex = nIntegerDigits; int decimalPointIndex = nIntegerDigits;
boolean roundCausedCarry = performRound(sb, 0, maxSigFigs); boolean roundCausedCarry = performRound(sb, 0, maxSigFigs);
int endIx = Math.min(maxSigFigs, sb.length()-1); int endIx = Math.min(maxSigFigs, sb.length()-1);
int nSigFigures; int nSigFigures;
if(roundCausedCarry) { if(roundCausedCarry) {
sb.insert(0, '1'); sb.insert(0, '1');
@ -292,11 +292,11 @@ public final class NumberToTextConverter {
if (pAbsExponent < 1) { if (pAbsExponent < 1) {
throw new IllegalArgumentException("abs(exponent) must be positive"); throw new IllegalArgumentException("abs(exponent) must be positive");
} }
int numberOfLeadingZeros = pAbsExponent-1; int numberOfLeadingZeros = pAbsExponent-1;
int absExponent = pAbsExponent; int absExponent = pAbsExponent;
int maxSigFigs = getNumberOfSignificantFiguresDisplayed(-absExponent); int maxSigFigs = getNumberOfSignificantFiguresDisplayed(-absExponent);
boolean roundCausedCarry = performRound(sb, 0, maxSigFigs); boolean roundCausedCarry = performRound(sb, 0, maxSigFigs);
int nRemainingSigFigs; int nRemainingSigFigs;
if(roundCausedCarry) { if(roundCausedCarry) {
@ -309,9 +309,9 @@ public final class NumberToTextConverter {
nRemainingSigFigs = countSignifantDigits(sb, 0 + maxSigFigs); nRemainingSigFigs = countSignifantDigits(sb, 0 + maxSigFigs);
sb.setLength(nRemainingSigFigs); sb.setLength(nRemainingSigFigs);
} }
int normalLength = 2 + numberOfLeadingZeros + nRemainingSigFigs; // 2 == "0.".length() int normalLength = 2 + numberOfLeadingZeros + nRemainingSigFigs; // 2 == "0.".length()
if (needsScientificNotation(normalLength)) { if (needsScientificNotation(normalLength)) {
if (sb.length()>1) { if (sb.length()>1) {
sb.insert(1, '.'); sb.insert(1, '.');
@ -319,7 +319,7 @@ public final class NumberToTextConverter {
sb.append('E'); sb.append('E');
sb.append('-'); sb.append('-');
appendExp(sb, absExponent); appendExp(sb, absExponent);
} else { } else {
sb.insert(0, "0."); sb.insert(0, "0.");
for(int i=numberOfLeadingZeros; i>0; i--) { for(int i=numberOfLeadingZeros; i>0; i--) {
sb.insert(2, '0'); sb.insert(2, '0');
@ -345,7 +345,7 @@ public final class NumberToTextConverter {
return; return;
} }
sb.append(val); sb.append(val);
} }
@ -391,12 +391,12 @@ public final class NumberToTextConverter {
while(sb.charAt(changeDigitIx) == '9') { while(sb.charAt(changeDigitIx) == '9') {
sb.setCharAt(changeDigitIx, '0'); sb.setCharAt(changeDigitIx, '0');
changeDigitIx--; changeDigitIx--;
// All nines, rounded up. Notify caller // All nines, rounded up. Notify caller
if(changeDigitIx < 0) { if(changeDigitIx < 0) {
return true; return true;
} }
} }
// no more '9's to round up. // no more '9's to round up.
// Last digit to be changed is still inside sb // Last digit to be changed is still inside sb
char prevDigit = sb.charAt(changeDigitIx); char prevDigit = sb.charAt(changeDigitIx);
sb.setCharAt(changeDigitIx, (char) (prevDigit + 1)); sb.setCharAt(changeDigitIx, (char) (prevDigit + 1));

View File

@ -22,11 +22,11 @@ import junit.framework.TestSuite;
/** /**
* Collects all tests the package <tt>org.apache.poi.hssf.record.formula.eval</tt>. * Collects all tests the package <tt>org.apache.poi.hssf.record.formula.eval</tt>.
* *
* @author Josh Micich * @author Josh Micich
*/ */
public class AllFormulaEvalTests { public class AllFormulaEvalTests {
public static Test suite() { public static Test suite() {
TestSuite result = new TestSuite(AllFormulaEvalTests.class.getName()); TestSuite result = new TestSuite(AllFormulaEvalTests.class.getName());
result.addTestSuite(TestAreaEval.class); result.addTestSuite(TestAreaEval.class);
@ -36,6 +36,7 @@ public class AllFormulaEvalTests {
result.addTestSuite(TestExternalFunction.class); result.addTestSuite(TestExternalFunction.class);
result.addTestSuite(TestFormulaBugs.class); result.addTestSuite(TestFormulaBugs.class);
result.addTestSuite(TestFormulasFromSpreadsheet.class); result.addTestSuite(TestFormulasFromSpreadsheet.class);
result.addTestSuite(TestMinusZeroResult.class);
result.addTestSuite(TestMissingArgEval.class); result.addTestSuite(TestMissingArgEval.class);
result.addTestSuite(TestPercentEval.class); result.addTestSuite(TestPercentEval.class);
result.addTestSuite(TestRangeEval.class); result.addTestSuite(TestRangeEval.class);

View File

@ -93,12 +93,21 @@ public final class TestEqualEval extends TestCase {
} }
/** /**
* Excel considers -0.0 to be equal to 0.0 * Bug 47198 involved a formula "-A1=0" where cell A1 was 0.0.
* Excel evaluates "-A1=0" to TRUE, not because it thinks -0.0==0.0
* but because "-A1" evaluated to +0.0
* <p/>
* Note - the original diagnosis of bug 47198 was that
* "Excel considers -0.0 to be equal to 0.0" which is NQR
* See {@link TestMinusZeroResult} for more specific tests regarding -0.0.
*/ */
public void testZeroEquality_bug47198() { public void testZeroEquality_bug47198() {
NumberEval zero = new NumberEval(0.0); NumberEval zero = new NumberEval(0.0);
NumberEval mZero = (NumberEval) UnaryMinusEval.instance.evaluate(new Eval[] { zero, }, 0, NumberEval mZero = (NumberEval) UnaryMinusEval.instance.evaluate(new Eval[] { zero, }, 0,
(short) 0); (short) 0);
if (Double.doubleToLongBits(mZero.getNumberValue()) == 0x8000000000000000L) {
throw new AssertionFailedError("Identified bug 47198: unary minus should convert -0.0 to 0.0");
}
Eval[] args = { zero, mZero, }; Eval[] args = { zero, mZero, };
BoolEval result = (BoolEval) EqualEval.instance.evaluate(args, 0, (short) 0); BoolEval result = (BoolEval) EqualEval.instance.evaluate(args, 0, (short) 0);
if (!result.getBooleanValue()) { if (!result.getBooleanValue()) {

View File

@ -0,0 +1,148 @@
/* ====================================================================
Licensed to the Apache Software Foundation (ASF) under one or more
contributor license agreements. See the NOTICE file distributed with
this work for additional information regarding copyright ownership.
The ASF 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.apache.poi.hssf.record.formula.eval;
import junit.framework.ComparisonFailure;
import junit.framework.TestCase;
import org.apache.poi.util.HexDump;
/**
* IEEE 754 defines a quantity '-0.0' which is distinct from '0.0'.
* Negative zero is not easy to observe in Excel, since it is usually converted to 0.0.
* (Note - the results of XLL add-in functions don't seem to be converted, so they are one
* reliable avenue to observe Excel's treatment of '-0.0' as an operand.)
* <p/>
* POI attempts to emulate Excel faithfully, so this class tests
* two aspects of '-0.0' in formula evaluation:
* <ol>
* <li>For most operation results '-0.0' is converted to '0.0'.</li>
* <li>Comparison operators have slightly different rules regarding '-0.0'.</li>
* </ol>
* @author Josh Micich
*/
public final class TestMinusZeroResult extends TestCase {
private static final double MINUS_ZERO = -0.0;
public void testSimpleOperators() {
// unary plus is a no-op
checkEval(MINUS_ZERO, UnaryPlusEval.instance, MINUS_ZERO);
// most simple operators convert -0.0 to +0.0
checkEval(0.0, UnaryMinusEval.instance, 0.0);
checkEval(0.0, PercentEval.instance, MINUS_ZERO);
checkEval(0.0, MultiplyEval.instance, MINUS_ZERO, 1.0);
checkEval(0.0, DivideEval.instance, MINUS_ZERO, 1.0);
checkEval(0.0, PowerEval.instance, MINUS_ZERO, 1.0);
// but SubtractEval does not convert -0.0, so '-' and '+' work like java
checkEval(MINUS_ZERO, SubtractEval.instance, MINUS_ZERO, 0.0); // this is the main point of bug 47198
checkEval(0.0, AddEval.instance, MINUS_ZERO, 0.0);
}
/**
* These results are hard to see in Excel (since -0.0 is usually converted to +0.0 before it
* gets to the comparison operator)
*/
public void testComparisonOperators() {
checkEval(false, EqualEval.instance, 0.0, MINUS_ZERO);
checkEval(true, GreaterThanEval.instance, 0.0, MINUS_ZERO);
checkEval(true, LessThanEval.instance, MINUS_ZERO, 0.0);
}
public void testTextRendering() {
confirmTextRendering("-0", MINUS_ZERO);
// sub-normal negative numbers also display as '-0'
confirmTextRendering("-0", Double.longBitsToDouble(0x8000100020003000L));
}
/**
* Uses {@link ConcatEval} to force number-to-text conversion
*/
private static void confirmTextRendering(String expRendering, double d) {
Eval[] args = { StringEval.EMPTY_INSTANCE, new NumberEval(d), };
StringEval se = (StringEval) ConcatEval.instance.evaluate(args, -1, (short)-1);
String result = se.getStringValue();
assertEquals(expRendering, result);
}
private static void checkEval(double expectedResult, OperationEval instance, double... dArgs) {
NumberEval result = (NumberEval) evaluate(instance, dArgs);
assertDouble(expectedResult, result.getNumberValue());
}
private static void checkEval(boolean expectedResult, OperationEval instance, double... dArgs) {
BoolEval result = (BoolEval) evaluate(instance, dArgs);
assertEquals(expectedResult, result.getBooleanValue());
}
private static Eval evaluate(OperationEval instance, double... dArgs) {
Eval[] evalArgs;
evalArgs = new Eval[dArgs.length];
for (int i = 0; i < evalArgs.length; i++) {
evalArgs[i] = new NumberEval(dArgs[i]);
}
Eval r = instance.evaluate(evalArgs, -1, (short)-1);
return r;
}
/**
* Not really a POI test - just shows similar behaviour of '-0.0' in Java.
*/
public void testJava() {
assertEquals(0x8000000000000000L, Double.doubleToLongBits(MINUS_ZERO));
// The simple operators consider all zeros to be the same
assertTrue(MINUS_ZERO == MINUS_ZERO);
assertTrue(MINUS_ZERO == +0.0);
assertFalse(MINUS_ZERO < +0.0);
// Double.compare() considers them different
assertTrue(Double.compare(MINUS_ZERO, +0.0) < 0);
// multiplying zero by any negative quantity yields minus zero
assertDouble(MINUS_ZERO, 0.0*-1);
assertDouble(MINUS_ZERO, 0.0*-1e300);
assertDouble(MINUS_ZERO, 0.0*-1e-300);
// minus zero can be produced as a result of underflow
assertDouble(MINUS_ZERO, -1e-300 / 1e100);
// multiplying or dividing minus zero by a positive quantity yields minus zero
assertDouble(MINUS_ZERO, MINUS_ZERO * 1.0);
assertDouble(MINUS_ZERO, MINUS_ZERO / 1.0);
// subtracting positive zero gives minus zero
assertDouble(MINUS_ZERO, MINUS_ZERO - 0.0);
// BUT adding positive zero gives positive zero
assertDouble(0.0, MINUS_ZERO + 0.0); // <<----
}
/**
* Just so there is no ambiguity. The two double values have to be exactly equal
*/
private static void assertDouble(double a, double b) {
long bitsA = Double.doubleToLongBits(a);
long bitsB = Double.doubleToLongBits(b);
if (bitsA != bitsB) {
throw new ComparisonFailure("value different to expected",
new String(HexDump.longToHex(bitsA)), new String(HexDump.longToHex(bitsB)));
}
}
}