Skip to content

Commit 0dcab25

Browse files
committed
WIP utility singleLabelConfusionMatrix(List<Prediction<MultiLabel>> predictions)
- partially represent false negatives by calling them false positives tied to some predicted label if there is one
1 parent 35735aa commit 0dcab25

File tree

1 file changed

+22
-2
lines changed

1 file changed

+22
-2
lines changed

MultiLabel/Core/src/test/java/org/tribuo/multilabel/IndependentMultiLabelTest.java

Lines changed: 22 additions & 2 deletions
Original file line numberDiff line numberDiff line change
@@ -24,6 +24,7 @@
2424
import java.util.logging.Level;
2525
import java.util.logging.Logger;
2626
import java.util.stream.Collectors;
27+
import java.util.stream.Stream;
2728

2829
import org.junit.jupiter.api.Assertions;
2930
import org.junit.jupiter.api.BeforeAll;
@@ -118,14 +119,19 @@ public static LabelConfusionMatrix singleLabelConfusionMatrix(final List<Predict
118119
}
119120

120121
public static List<Prediction<Label>> mkSingleLabelPredictions(List<Prediction<MultiLabel>> predictions) {
122+
return mkSingleLabelPredictions(predictions, false);
123+
}
124+
125+
public static List<Prediction<Label>> mkSingleLabelPredictions(List<Prediction<MultiLabel>> predictions,
126+
final boolean falseNegativeHeuristic) {
121127
return predictions.stream()
122128
.flatMap(p -> {
123129
final Set<Label> trueLabels = p.getExample().getOutput().getLabelSet();
124130
final Set<Label> predicted = p.getOutput().getLabelSet();
125131
// intersection(trueLabels, predicted) = true positives
126132
// predicted - trueLabels = false positives
127133
// trueLabels - predicted = false negatives
128-
return predicted.stream().map(pred -> {
134+
return Stream.concat(predicted.stream().map(pred -> {
129135
if (trueLabels.contains(pred)) {
130136
return mkPrediction(pred.getLabel(), pred.getLabel());
131137
} else if (trueLabels.size() == 1) {
@@ -134,7 +140,21 @@ public static List<Prediction<Label>> mkSingleLabelPredictions(List<Prediction<M
134140
// arbitrarily pick first trueLabel
135141
return mkPrediction(trueLabels.iterator().next().getLabel(), pred.getLabel());
136142
}
137-
});
143+
}),
144+
!falseNegativeHeuristic ? Stream.of() :
145+
// partially represent false negatives by calling them false positives tied to some predicted label if there is one
146+
trueLabels.stream().filter(t -> !predicted.contains(t)).flatMap(fnTrueLabel -> {
147+
if (predicted.isEmpty()) {
148+
// nothing to pin this on
149+
return Stream.of();
150+
} else if (predicted.size() == 1) {
151+
return Stream.of(mkPrediction(fnTrueLabel.getLabel(), predicted.iterator().next().getLabel()));
152+
} else {
153+
// arbitrarily pick first predicted label
154+
return Stream.of(mkPrediction(fnTrueLabel.getLabel(), predicted.iterator().next().getLabel()));
155+
}
156+
})
157+
);
138158
}).collect(Collectors.toList());
139159
}
140160

0 commit comments

Comments
 (0)