aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorPJ Fanning <fanningpj@apache.org>2021-08-07 18:36:12 +0000
committerPJ Fanning <fanningpj@apache.org>2021-08-07 18:36:12 +0000
commitb1bf15e2b555bcf48a5248dd6c8a6e1b3de70310 (patch)
treea30ede01f8de391360c9d8cb7a593fb9aa6e8223
parent9b697eeb76d5adb1f4c287e8afe5e7b88560ff9a (diff)
downloadpoi-b1bf15e2b555bcf48a5248dd6c8a6e1b3de70310.tar.gz
poi-b1bf15e2b555bcf48a5248dd6c8a6e1b3de70310.zip
[bug-49202] add PERCENTRANK.EXC function
git-svn-id: https://svn.apache.org/repos/asf/poi/trunk@1892088 13f79535-47bb-0310-9956-ffa450edef68
-rw-r--r--poi/src/main/java/org/apache/poi/ss/formula/atp/PercentRankExcFunction.java14
-rw-r--r--poi/src/main/java/org/apache/poi/ss/formula/functions/PercentRank.java18
2 files changed, 13 insertions, 19 deletions
diff --git a/poi/src/main/java/org/apache/poi/ss/formula/atp/PercentRankExcFunction.java b/poi/src/main/java/org/apache/poi/ss/formula/atp/PercentRankExcFunction.java
index bc4d887245..8391666615 100644
--- a/poi/src/main/java/org/apache/poi/ss/formula/atp/PercentRankExcFunction.java
+++ b/poi/src/main/java/org/apache/poi/ss/formula/atp/PercentRankExcFunction.java
@@ -25,7 +25,6 @@ import org.apache.poi.ss.formula.functions.PercentRank;
import java.math.BigDecimal;
import java.math.RoundingMode;
import java.util.ArrayList;
-import java.util.Collections;
import java.util.List;
/**
@@ -132,12 +131,8 @@ final class PercentRankExcFunction implements FreeRefFunction {
}
if (!recurse || closestMatchBelow == x || closestMatchAbove == x) {
int lessThanCount = 0;
- int greaterThanCount = 0;
- int matchesCount = 0;
for (Double d : numbers) {
if (d < x) lessThanCount++;
- else if (d > x) greaterThanCount++;
- else matchesCount++;
}
BigDecimal result = new BigDecimal((double)(lessThanCount + 1) / (double)(numbers.size() + 1));
return new NumberEval(PercentRank.round(result, significance, RoundingMode.DOWN));
@@ -150,13 +145,8 @@ final class PercentRankExcFunction implements FreeRefFunction {
if (!(aboveRank instanceof NumberEval)) {
return aboveRank;
}
- NumberEval below = (NumberEval)belowRank;
- NumberEval above = (NumberEval)aboveRank;
- double diff = closestMatchAbove - closestMatchBelow;
- double pos = x - closestMatchBelow;
- double rankDiff = above.getNumberValue() - below.getNumberValue();
- BigDecimal result = new BigDecimal(below.getNumberValue() + (rankDiff * (pos / diff)));
- return new NumberEval(PercentRank.round(result, significance, RoundingMode.HALF_UP));
+ return PercentRank.interpolate(x, closestMatchBelow, closestMatchAbove,
+ (NumberEval)belowRank, (NumberEval)aboveRank, significance);
}
}
}
diff --git a/poi/src/main/java/org/apache/poi/ss/formula/functions/PercentRank.java b/poi/src/main/java/org/apache/poi/ss/formula/functions/PercentRank.java
index 0eecb75c07..10d67f7cc8 100644
--- a/poi/src/main/java/org/apache/poi/ss/formula/functions/PercentRank.java
+++ b/poi/src/main/java/org/apache/poi/ss/formula/functions/PercentRank.java
@@ -133,17 +133,21 @@ public final class PercentRank implements Function {
if (!(aboveRank instanceof NumberEval)) {
return aboveRank;
}
- NumberEval below = (NumberEval)belowRank;
- NumberEval above = (NumberEval)aboveRank;
- double diff = closestMatchAbove - closestMatchBelow;
- double pos = x - closestMatchBelow;
- double rankDiff = above.getNumberValue() - below.getNumberValue();
- BigDecimal result = new BigDecimal(below.getNumberValue() + (rankDiff * (pos / diff)));
- return new NumberEval(round(result, significance, RoundingMode.HALF_UP));
+ return interpolate(x, closestMatchBelow, closestMatchAbove, (NumberEval)belowRank, (NumberEval)aboveRank, significance);
}
}
@Internal
+ public static NumberEval interpolate(double x, double closestMatchBelow, double closestMatchAbove,
+ NumberEval belowRank, NumberEval aboveRank, int significance) {
+ double diff = closestMatchAbove - closestMatchBelow;
+ double pos = x - closestMatchBelow;
+ double rankDiff = aboveRank.getNumberValue() - belowRank.getNumberValue();
+ BigDecimal result = new BigDecimal(belowRank.getNumberValue() + (rankDiff * (pos / diff)));
+ return new NumberEval(round(result, significance, RoundingMode.HALF_UP));
+ }
+
+ @Internal
public static double round(BigDecimal bd, int significance, RoundingMode rounding) {
//the rounding in https://support.microsoft.com/en-us/office/percentrank-function-f1b5836c-9619-4847-9fc9-080ec9024442
//is very inconsistent, this hodge podge of rounding modes is the only way to match Excel results