Эффективное пересечение двух списков<String> в Java?
-
18-09-2019 - |
Вопрос
Вопрос прост:
У меня есть два Списка
List<String> columnsOld = DBUtils.GetColumns(db, TableName);
List<String> columnsNew = DBUtils.GetColumns(db, TableName);
И мне нужно найти их пересечение.Есть ли быстрый способ достичь этого?
Решение
Вы можете использовать retainAll
способ:
columnsOld.retainAll (columnsNew);
Другие советы
Поскольку retainAll не будет касаться коллекции аргументов, это было бы быстрее:
List<String> columnsOld = DBUtils.GetColumns(db, TableName);
List<String> columnsNew = DBUtils.GetColumns(db, TableName);
for(int i = columnsNew.size() - 1; i > -1; --i){
String str = columnsNew.get(i);
if(!columnsOld.remove(str))
columnsNew.remove(str);
}
Пересечением будут значения, оставленные в columnsNew.Удаление уже сравниваемых значений из columnsOld уменьшит количество необходимых сравнений.
Использование Гуавы:
Sets.intersection(Sets.newHashSet(setA), Sets.newHashSet(setB))
Как насчет
private List<String> intersect(List<String> A, List<String> B) {
List<String> rtnList = new LinkedList<>();
for(String dto : A) {
if(B.contains(dto)) {
rtnList.add(dto);
}
}
return rtnList;
}
Существует хороший способ с потоками, который может сделать это в одной строке кода, и вы можете использовать два списка, которые не относятся к одному типу, что невозможно с помощью метода containsAll afaik:
columnsOld.stream().filter(c -> columnsNew.contains(c)).collect(Collectors.toList());
Пример для списков с разными типами.Если у вас есть связь между foo и bar, и вы можете получить bar-объект из foo, чем вы можете изменить свой поток:
List<foo> fooList = new ArrayList<>(Arrays.asList(new foo(), new foo()));
List<bar> barList = new ArrayList<>(Arrays.asList(new bar(), new bar()));
fooList.stream().filter(f -> barList.contains(f.getBar()).collect(Collectors.toList());
Если вы помещаете второй список в набор, скажите HashSet.И просто выполните итерацию по первому списку, проверяя наличие в наборе и удаляя, если его нет, ваш первый список в конечном итоге будет иметь нужное вам пересечение.Это будет намного быстрее, чем сохранять все или содержать в списке.Акцент здесь делается на использовании set вместо list .Поисковые запросы - O(1).firstList.retainAll (новый набор хэшей (secondList)) также будет работать.
используя retainAll, если вас не волнуют вхождения, в противном случае используя N.пересечение
a = N.asList(12, 16, 16, 17, 19);
b = N.asList(16, 19, 107);
a.retainAll(b); // [16, 16, 19]
N.println(a);
a = N.asList(12, 16, 16, 17, 19);
b = N.asList(16, 19, 107);
a = N.intersect(a, b);
N.println(a); // [16, 19]
N - это служебный класс в Счеты , используемые
используйте org.apache.commons.collections4.ListUtils#пересечение