Лучший способ перебрать SortedSet / SortedMap в Java в обратном направлении - PullRequest
19 голосов
/ 17 марта 2009

Мне нужно перебрать набор записей SortedMap (который является SortedSet) в обратном направлении. Код, который я пишу, чрезвычайно чувствителен к производительности, так как его будут вызывать из разных мест тысячи раз в секунду, а может и больше. Любой совет сделать это самым быстрым способом?

Ответы [ 4 ]

24 голосов
/ 17 марта 2009

В Java 1.6 вы можете использовать NavigableSet .

22 голосов
/ 19 ноября 2012

используйте это перед заполнением карты:

SortedMap sortedMap = new TreeMap(java.util.Collections.reverseOrder());
5 голосов
/ 17 марта 2009

Более быстрый способ итерации коллекции - взять ее копию в массиве. Это может быть повторено вперед и назад без создания объекта или даже вызова метода. Недостатком является то, что вам нужно обновлять его, а также свою коллекцию всякий раз, когда она меняется.

В следующем примере для итерации более 1000 элементов в прямом или обратном направлении требуется в среднем 1 208 нс.

import java.util.Comparator;
import java.util.Random;
import java.util.NavigableSet;
import java.util.TreeSet;

/*
Average time for iteration of 1000 elements was 1,208 ns
*/
public class Main {
    public static void main(String... args) {
        doPerfTest(true);
        doPerfTest(false);
    }

    private static void doPerfTest(boolean warmup) {
        NavigableSet<MyData> set = new TreeSet<MyData>(new MyCompataror());
        Random random = new Random();
        for (int i = 0; i < 1000; i++) {
            set.add(new MyData("text-" + random.nextLong(), random.nextInt()));
        }
        MyData[] myDatas = set.toArray(new MyData[set.size()]);
        long start = System.nanoTime();
        final int runs = 500 * 1000;
        for (int i = 0; i < runs; i+=2) {
            // forward iteration
            for (MyData md : myDatas) {

            }
            // reverse iteration
            for (int j = myDatas.length - 1; j >= 0; j--) {
                MyData md = myDatas[j];
            }
        }
        long time = System.nanoTime() - start;
        if (!warmup)
            System.out.printf("Average time for iteration of 1000 elements was %,d ns", time / runs);
    }

    static class MyCompataror implements Comparator<MyData> {
        public int compare(MyData o1, MyData o2) {
            int cmp = o1.text.compareTo(o2.text);
            if (cmp != 0)
                return cmp;
            return o1.value > o2.value ? +1 :
                    o1.value < o2.value ? -1 : 0;
        }
    }

    static class MyData {
        String text;
        int value;

        MyData(String text, int value) {
            this.text = text;
            this.value = value;
        }
    }
}

Теперь замените основной цикл, и среднее время станет 20,493.

// forward iteration
for(Iterator<MyData> it = set.iterator(); it.hasNext();) {
    MyData md = it.next();
}
// reverse iteration
for(Iterator<MyData> it = set.descendingIterator(); it.hasNext();) {
    MyData md = it.next();
}

Теперь давайте сравним это с тем, чтобы делать копию каждый раз (что, как я уже сказал, не так оптимально, как получение копии только при изменении), время уменьшается до 15 134 нс!

Таким образом, использование NavigableSet может быть самым медленным из трех рассмотренных вариантов.

2 голосов
/ 17 марта 2009

Вы можете определить SortedMap, такой как TreeMap с пользовательским Comparator , который меняет порядок сортировки. Если вам нужно выполнить итерацию в обоих направлениях, то возможно ли сохранить две копии структуры данных в памяти?

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...