0

我有一堆Map<Integer, TreeSet<Double>>对象,我将这些值传递给另一个方法。我NoSuchElementException在那个方法中得到了一个,所以开始调试,并看到调用Map#values()引发了这个异常。但是,当我遍历这些值时,它会完美地打印所有值。这是导致此问题的代码片段:

Map<Integer, TreeMultimap<String, Double>> kMap = Maps.newTreeMap();
for (int i = 0; i < 3; i++) {
    TreeMultimap<String, Double> treeMultimap = TreeMultimap.create();
    treeMultimap.putAll("a" + i, Lists.newArrayList(0.0, 0.06, 0.17, 0.23));
    treeMultimap.putAll("b" + i, Lists.newArrayList(0.0, 0.16, 0.34, 0.49));
    kMap.put(i, treeMultimap);
}

Map<Integer, TreeSet<Double>> rescaledKMap =
    Maps.transformEntries(kMap, ToRescaledValueMap(1.2));

Iterable<TreeSet<Double>> treeSets = rescaledKMap.values();

// Prints out each element correctly. No exception is thrown.
for (TreeSet<Double> set : treeSets)
    System.out.println(Joiner.on(',').join(set));

// NoSuchElementException thrown in getMeanPlot() before anything happens there.
TreeMap<Double, Double> mean_rescaledKMap = 
    getMeanPlot(rescaledKMap.values());

签名是getMeanPlot(Iterable<TreeSet<Double>>)

可迭代对象在传递给第二个方法之前如何正确,但在第二个方法开始时抛出异常?

堆栈跟踪

堆栈跟踪将Maps.EntryTransformer对象显示ToRescaledValueMap为罪魁祸首。这是跟踪:

Exception in thread "main" java.util.NoSuchElementException
    at java.util.TreeMap$PrivateEntryIterator.nextEntry(TreeMap.java:1113)
    at java.util.TreeMap$EntryIterator.next(TreeMap.java:1151)
    at java.util.TreeMap$EntryIterator.next(TreeMap.java:1146)
    at com.google.common.collect.AbstractMapBasedMultimap$Itr.next(AbstractMapBasedMultimap.java:1145)
    at com.google.common.collect.Ordering.min(Ordering.java:460)
    at com.google.common.collect.Ordering.min(Ordering.java:479)
    at DistributionMetricExperiment$6.transformEntry

这与 Guava 的惰性初始化有什么关系吗?我可以编写自己的小方法而不是使用该Maps#transformEntries方法,但不知道这个错误背后的原因是......好吧......困扰着我。入口变换如下:

static Maps.EntryTransformer<Integer, TreeMultimap<String,Double>, TreeSet<Double>>
ToRescaledValueMap(final double interval_length) {
    return new Maps.EntryTransformer<Integer, TreeMultimap<String,Double>, TreeSet<Double>>()
    {
        public TreeSet<Double> transformEntry(Integer i, TreeMultimap<String,Double> mmap) {
            double mmap_min = Ordering.natural().min(mmap.values());
            double mmap_max = Ordering.natural().max(mmap.values());
            TreeSet<Double> rescaledValues = Sets.newTreeSet();
            for (double val : mmap.values())
                rescaledValues.add(interval_length * (val - mmap_min)/(mmap_max - mmap_min));
            return rescaledValues;
        }
    };
}

编辑

  • 我不知道这是否重要,但想补充一点,我也发出了Iterable<TreeSet<Double>getMeanPlot. 即使在那里,它也可以打印,然后抛出异常。
  • 对于在变量名中使用下划线,我深表歉意。这些变量反映了我在代码背后的数学中使用的下标。我知道命名约定,但在这段代码中,我更专注于能够通过查看变量来识别变量的数学含义。
4

1 回答 1

1

实际上,您打印的是TreeSet<Double> set,这会误导您认为您的收藏已被填充。因为您的错误不是 onset而是 on TreeMultimap<String,Double> mmap:您执行Ordering.natural().min(mmap),这就是问题所在。Guava 指定在空集合中查找最小(或最大)元素将导致抛出 a NoSuchElementException,这就是您在此处所拥有的。

于 2014-03-07T00:27:40.917 回答