Комментарии 45
за константное время O(1) выполняются операции вставки и удаления первого и последнего элемента, операции вставки и удаления элемента из середины списка (не учитывая время поиска позиции вставки элемента);
Как раз сложность поиска элемента в двусвязном списке как раз O(N), и это надо прямо указывать. А то у меня от этой вставки и удаления из середины за О(1) глаза на лоб вылезли.
Из приятного — наглядность картинок порадовала. Продолжайте пробовать писать, но что-то более существенное и интересное, ибо то, что написано здесь — прям для совсем начинающих.
перебор элементов списка с помощью итератора осуществляется за О(1)?
Перебор, очевидным образом осуществляется за O(n). Предыдущий комментарий был о том, что, если у нас есть итератор, который уже стоит на каком-то конкретном узле, скорость удаления этого узла никак не будет зависеть от количества элементов в LinkedList.
Да, и кроме того — иногда забывают, что ArrayList (по сути массив) занимает непрерывную область памяти и если попытаться создать такой список очень большого размера, то вполне реально упасть с ошибкой выделения памяти.
Далее, если мы имеем дело с большим объемом данных, то LinkedList тем более не подойдет, расходы памяти на него выше, чем у ArrayList. Каждый элемент оборачивается, небольшим, но все таки объектом Node. Когда ваша программа работает на пределе возможностей, универсального рецепта нет, но вероятно и от ArrayList придется отказаться тоже и работать с простыми массивами, как с наиболее низкоуровневыми объектами.
занимает непрерывную область памяти и если попытаться создать такой список очень большого размера, то вполне реально упасть с ошибкой выделения памяти.
Обычно JVM будет перемещать объекты пока не сможет выделить такую область памяти. То есть ошибка out of memory будет если банально кончится вся или почти вся свободная память. Но LinkedList занимает (по моим подсчетам) примерно в 4 раза больше места чем ArrayList, то есть он упадет с out of memory намного раньше, не говоря уже о бешеной нагрузке на сборщик мусора от миллионов объектов типа Node.
P.S. Есть разумеется фундаментальное ограничение на ArrayList и LinkedList в виде использования int в качестве их счетчика (а это 2 с небольшим миллиарда). Поэтому List не может быть больше 2 млд. с хвостиков значений, но это несколько гигабайт, а учитывая что в List хранятся только ссылки, то с объектами это десятки гигабайт на одну структуру. Как правило, для большинства даже серверных задач это с головой хватает (если вам нужно в памяти одного процесса держать десятки гигабайт данных в одном List — у вас что не так со архитектурой приложения).
Зря вы не упомянули add/remove в итераторе, это существенная часть внутреннего устройства (единственная реализация за O(1))
Рекомендую прочитать весь тред.
twitter.com/joshbloch/status/583813919019573248
1. Не существует задач когда стоит использовать LinkedList,
2. Если у вас такая задача см. пункт 1,
Надо часто удалять из середины? Используйте ArrayList заменяя элементы на null, а потом их игнорируяя. При слишком большом кол-ве null — можно создать новый лист без null.
Надо всавлять в начало — переверните ArrayList и вставляйте в конец, С двух сторон — просто используйте 2 листа,
Надо вставлять в середину много элементом — используйте временный лист и addAll,
В конце концов, самое лучшее используйте TreeList из Apache коллекций. Не хотите /не можете тянуть зависимость? Просто скопируйте ее код.
Это осуждалось уже в моей статье
Придумайте любую задачу и тест производительности, а перепишу с использованием ArrayList (на крайний случай двух ArrayList) реализацию быстрее. Я уже не говорю о TreeList от апача, которые быстрее будет практически всегда.
И это не говоря о диких расходах памяти (намного больших чем ArrayList) и мусора для сборщика
public class MyList {
private List<Long> list = new ArrayList<Long>(1500000000);
private int firstIndex;
public Long remove(int index) {
Long old = list.set(index + firstIndex, null);
if(index == 0) {
firstIndex++;
}
return old;
}
...
}
При вставке в первый элемент проводим обратную операцию (на случай вставки до удаления можно изначально начинать с firstIndex равным 10-30% от ожидаемого размера коллекции).
И раз уж мы тут говорим про сравнение листов — то ArrayList на 3 миллиарда элементов тоже не сделаешь.
А вы предлагает саблисты использовать. Что ошибкоопасно.
Не хотите писать руками используйте (или скопируйте) TreeList от апача.
Да и итератор если получить — нас итератор от основного листа вернётся.
Кто сказал? Сложно написать свой итератор?
И раз уж мы тут говорим про сравнение листов — то ArrayList на 3 миллиарда элементов тоже не сделаешь.
С такими запросами вам классы BigList нужны
О чем вы спорите я тоже не знаю.
Он работает примерно следующим образом: храним массив (одним куском) + его размер, и два указателя на него: на первый элемент дека и на следующий после последнего.
При добавлении в конец или начало: смотрим, есть ли у нас свободное место в массиве. Если нет — выделяем новый массив в X (от 1 до 2) раз больше текущего и все элементы перемещаем в него. Дальше добавляем новый элемент по указателю на конец или начало, сдвигая его в нужную сторону. Сложность получается O(N) в случае увеличения размера, но, учитывая экспоненциальность выделения памяти, в среднем O(1).
В итоге у нас добавление и удаление элементов как в начало, так и в конец за O(1), а еще за O(1) поиск элемента по индексу и вообще все cache friendly.
Например, я вижу такое существенное отличие в данном случае, как экономия 6 млрд указателей (48 Гб памяти) по сравнению с LinkedList. Да и скорость итерирования за счет лучшего использования кеша будет лучше.
Для хранения, скажем, 3 млрд 32-битных чисел в массиве нам нужно 3*4=12 Гб памяти, а для хранения их же в списке нам уже потребуется 3*(4 + 2*8)=60 Гб памяти. Как может такое быть, что первое не умещается в память, а второе умещается?
А вот в линкдлисте таких ограничений нет.
Правда нет? То есть то что все методы интерфейса List принимают только int в качестве индекса вас не смущает?
А то что внутренне поле size у LinkedList тоже int — тоже не наводит на смутные сомнения, что больше Integer.MAX_VALUE элементов в LinkedList никак не влезет?
Поавила использования LinkedList:
- Не существует задач когда стоит использовать LinkedList,
- Если у вас такая задача см. пункт 1,
А как насчёт кейса, когда нужна стабильная производительность? Когда просадок на вырастание внутреннего массива надо избегать?
1. Если перед вами лист дерева (структура данных или растение) — называйте его листом.
2. Если перед вами лист бумаги — называйте его листом.
2. Если перед вами ArrrayList или LinkedList — называйте их списками (точнее, списочным массивом и связанным списком). Вы, конечно, можете называть их и листами, но это так же неправильно, как называть их деревьями, массивами, графами, кошками или космическими кораблями.
Что «под капотом» у LinkedList?