Динамический массив - Dynamic array
В Информатика, а динамический массив, растущий массив, изменяемый размер массива, динамический стол, изменяемый массив, или же список массивов это произвольный доступ, список переменного размера структура данных который позволяет добавлять или удалять элементы. Он поставляется со стандартными библиотеками на многих современных основных языках программирования. Динамические массивы преодолевают предел статического массивы, которые имеют фиксированную емкость, которую необходимо указать при выделении.
Динамический массив - это не то же самое, что динамически распределяется массив, который является множество чей размер фиксируется при выделении массива, хотя динамический массив может использовать такой массив фиксированного размера в качестве серверной части.[1]
Динамические массивы и емкость ограниченного размера
Простой динамический массив может быть создан путем выделения массива фиксированного размера, обычно большего, чем количество элементов, требуемых немедленно. Элементы динамического массива хранятся непрерывно в начале базового массива, а оставшиеся позиции ближе к концу базового массива зарезервированы или не используются. Элементы могут быть добавлены в конец динамического массива в постоянное время используя зарезервированное пространство, пока это пространство не будет использовано полностью. Когда все пространство израсходовано и нужно добавить дополнительный элемент, тогда необходимо увеличить размер базового массива фиксированного размера. Обычно изменение размера является дорогостоящим, поскольку оно включает выделение нового базового массива и копирование каждого элемента из исходного массива. Элементы могут быть удалены с конца динамического массива за постоянное время, так как изменение размера не требуется. Количество элементов, используемых содержимым динамического массива, - это его логический размер или же размер, а размер базового массива называется динамическим массивом емкость или же физический размер, что является максимально возможным размером без перемещения данных.[2]
Массив фиксированного размера будет достаточным в приложениях, где максимальный логический размер фиксирован (например, по спецификации) или может быть вычислен до выделения массива. Динамический массив может быть предпочтительнее, если:
- максимальный логический размер неизвестен или его трудно вычислить до выделения массива
- считается, что максимальный логический размер, указанный в спецификации, вероятно, изменится
- амортизированная стоимость изменения размера динамического массива не оказывает существенного влияния на производительность или быстродействие
Геометрическое расширение и амортизированная стоимость
Чтобы избежать многократных затрат на изменение размера, динамические массивы изменяют размер на большую величину, например удваивают размер, и используют зарезервированное пространство для будущего расширения. Операция добавления элемента в конец может работать следующим образом:
функция insertEnd(dynarray а, элемент е) если (а.размер == а.емкость) // изменить размер в два раза больше его текущей емкости: а.емкость ← а.емкость * 2 // (копируем содержимое в новое место в памяти) а[а.размер] ← е а.размер ← а.размер + 1
В качестве п вставляются элементы, емкости образуют геометрическая прогрессия. Расширение массива любой постоянной пропорцией а гарантирует, что вставка п элементы берет О(п) общее время, что означает, что каждая вставка занимает амортизированный постоянное время. Многие динамические массивы также освобождают часть базового хранилища, если его размер падает ниже определенного порога, например 30% от емкости. Этот порог должен быть строго меньше 1 /а для того, чтобы предоставить гистерезис (обеспечить стабильную полосу, чтобы избежать повторного роста и сжатия) и поддерживать смешанные последовательности вставок и удалений с амортизированной постоянной стоимостью.
Динамические массивы - распространенный пример при обучении амортизированный анализ.[3][4]
Фактор роста
Фактор роста для динамического массива зависит от нескольких факторов, включая компромисс между пространством и временем и алгоритмы, используемые в самом распределителе памяти. Фактор роста а, среднее время одной операции вставки о а/(а−1), а количество потерянных ячеек ограничено сверху величиной (а−1)п[нужна цитата ]. Если распределитель памяти использует первоочередное распределение алгоритм, то значения фактора роста, такие как а = 2 может привести к нехватке памяти при расширении динамического массива, даже если значительный объем памяти все еще доступен.[5] Были проведены различные дискуссии об идеальных значениях факторов роста, включая предложения по Золотое сечение а также значение 1,5.[6] Однако во многих учебниках а = 2 для простоты и анализа.[3][4]
Ниже приведены факторы роста, используемые несколькими популярными реализациями:
Выполнение | Фактор роста (а) |
---|---|
Java ArrayList[1] | 1.5 (3/2) |
Python PyListObject[7] | ~ 1,125 (п + п >> 3) |
Microsoft Visual C ++ 2013[8] | 1.5 (3/2) |
G ++ 5.2.0[5] | 2 |
Лязг 3.6[5] | 2 |
Безумие Facebook / FBVector[9] | 1.5 (3/2) |
Ржавчина Vec[10] | 2 |
Спектакль
Связанный список | Множество | Динамический массив | Сбалансированное дерево | Случайный список доступа | Дерево хешированных массивов | |
---|---|---|---|---|---|---|
Индексирование | Θ (п) | Θ (1) | Θ (1) | Θ (журнал n) | Θ (журнал n)[11] | Θ (1) |
Вставить / удалить в начале | Θ (1) | Нет данных | Θ (п) | Θ (журнал n) | Θ (1) | Θ (п) |
Вставить / удалить в конце | Θ (1) когда последний элемент известен; Θ (п) когда последний элемент неизвестен | Нет данных | Θ (1) амортизированный | Θ (журнал п) | Нет данных [11] | Θ (1) амортизированный |
Вставить / удалить в середине | время поиска + Θ (1)[12][13] | Нет данных | Θ (п) | Θ (журнал п) | Нет данных [11] | Θ (п) |
Потраченное впустую пространство (средний) | Θ (п) | 0 | Θ (п)[14] | Θ (п) | Θ (п) | Θ (√п) |
Динамический массив имеет производительность, аналогичную массиву, с добавлением новых операций для добавления и удаления элементов:
- Получение или установка значения по определенному индексу (постоянное время)
- Итерация элементов по порядку (линейное время, хорошая производительность кеша)
- Вставка или удаление элемента в середине массива (линейное время)
- Вставка или удаление элемента в конце массива (постоянное амортизированное время)
Динамические массивы обладают многими преимуществами массивов, в том числе хорошими местонахождение ссылки и кеш данных использование, компактность (низкое использование памяти) и произвольный доступ. Обычно у них есть только небольшие фиксированные дополнительные накладные расходы для хранения информации о размере и емкости. Это делает динамические массивы привлекательным инструментом для создания тайник -дружелюбный структуры данных. Однако в таких языках, как Python или Java, которые применяют семантику ссылок, динамический массив обычно не будет хранить фактические данные, а скорее будет хранить Рекомендации к данным, которые находятся в других областях памяти. В этом случае последовательный доступ к элементам в массиве фактически будет включать доступ к нескольким несмежным областям памяти, поэтому многие преимущества дружественности к кэшу этой структуры данных теряются.
В сравнении с связанные списки динамические массивы имеют более быстрое индексирование (постоянное время по сравнению с линейным временем) и, как правило, более быструю итерацию из-за улучшенной локальности ссылок; однако динамические массивы требуют линейного времени для вставки или удаления в произвольном месте, поскольку все последующие элементы должны быть перемещены, а связанные списки могут делать это за постоянное время. Этот недостаток смягчается за счет буфер промежутка и многоуровневый вектор варианты обсуждаются в Варианты ниже. Кроме того, в очень фрагментированный область памяти, может быть дорого или невозможно найти непрерывное пространство для большого динамического массива, тогда как связанные списки не требуют, чтобы вся структура данных хранилась непрерывно.
А сбалансированное дерево может хранить список, обеспечивая при этом все операции как с динамическими массивами, так и с связанными списками достаточно эффективно, но как вставка в конце, так и итерация по списку медленнее, чем для динамического массива, в теории и на практике из-за несмежного хранения и накладные расходы на обход дерева / манипуляции.
Варианты
Промежуточные буферы похожи на динамические массивы, но позволяют выполнять эффективные операции вставки и удаления, сгруппированные в одном и том же произвольном месте. Немного дек реализации используют массив дек, которые позволяют вставлять / извлекать / извлекать / вставлять амортизированное постоянное время на обоих концах вместо одного конца.
Goodrich[15] представил алгоритм динамического массива, названный многоуровневые векторы что обеспечивает O (n1 / к) производительность для вставок и удалений из любого места в массиве, и O (k) получение и установка, где k ≥ 2 - постоянный параметр.
Дерево хешированных массивов (HAT) - это алгоритм динамического массива, опубликованный Ситарски в 1996 году.[16] Порядок отходов дерева хешированных массивов n1/2 объем дискового пространства, где n - количество элементов в массиве. Алгоритм имеет амортизированную производительность O (1) при добавлении серии объектов в конец дерева хешированного массива.
В статье 1999 г.[17] Бродник и др. описать многоуровневую структуру данных динамического массива, которая тратит только n1/2 место для п элементы в любой момент времени, и они доказывают нижнюю границу, показывающую, что любой динамический массив должен тратить столько места, чтобы операции оставались амортизированными за постоянное время. Кроме того, они представляют вариант, при котором увеличение и уменьшение буфера не только амортизировало, но и в худшем случае постоянное время.
Багвелл (2002)[18] представили алгоритм VList, который можно адаптировать для реализации динамического массива.
Языковая поддержка
C ++ с std :: vector
и Ржавчина с std :: vec :: Vec
являются реализациями динамических массивов, как и ArrayList
[19] классы, снабженные Ява API и .NET Framework.[20]
Общий Список <>
класс, поставляемый с .NET Framework версии 2.0, также реализован с помощью динамических массивов. Болтовня с OrderedCollection
представляет собой динамический массив с динамическими начальным и конечным индексами, что делает удаление первого элемента также O (1).
Python с список
реализация типа данных представляет собой динамический массив.
Delphi и D реализовать динамические массивы в ядре языка.
Ада с Ada.Containers.Vectors
Общий пакет обеспечивает реализацию динамического массива для данного подтипа.
Многие языки сценариев, такие как Perl и Рубин предлагать динамические массивы как встроенные примитивный тип данных.
Несколько кросс-платформенных фреймворков предоставляют реализации динамических массивов для C, включая CFArray
и CFMutableArray
в Основной фундамент, и GArray
и GPtrArray
в GLib.
Common Lisp обеспечивает элементарную поддержку векторов изменяемого размера, позволяя настраивать встроенные множество
введите как регулируемый и место вставки указатель заполнения.
Рекомендации
- ^ а б См., Например, исходный код класса java.util.ArrayList из OpenJDK 6.
- ^ Ламберт, Кеннет Альфред (2009), «Физический размер и логический размер», Основы Python: от первых программ до структур данных, Cengage Learning, стр. 510, г. ISBN 978-1423902188
- ^ а б Гудрич, Майкл Т.; Тамассия, Роберто (2002), «1.5.2 Анализ реализации расширяемого массива», Разработка алгоритмов: основы, анализ и примеры в Интернете, Wiley, стр. 39–41..
- ^ а б Кормен, Томас Х.; Лейзерсон, Чарльз Э.; Ривест, Рональд Л.; Штейн, Клиффорд (2001) [1990]. «17.4 Динамические таблицы». Введение в алгоритмы (2-е изд.). MIT Press и McGraw-Hill. С. 416–424. ISBN 0-262-03293-7.
- ^ а б c "Вектор C ++ STL: определение, фактор роста, функции-члены". Архивировано из оригинал на 2015-08-06. Получено 2015-08-05.
- ^ «векторный фактор роста 1,5». comp.lang.c ++. модерируется. Группы Google.
- ^ Реализация объекта списка с github.com/python/cpython/, получено 23 марта 2020 г.
- ^ Брайс, Хади. «Анализ вектора C ++ STL: Часть 3 - Емкость и размер». Микромистерии. Получено 2015-08-05.
- ^ "фейсбук / глупость". GitHub. Получено 2015-08-05.
- ^ "ржавчина / ржавчина". GitHub. Получено 2020-06-09.
- ^ а б c Крис Окасаки (1995). «Чисто функциональные списки произвольного доступа». Материалы Седьмой Международной конференции по языкам функционального программирования и компьютерной архитектуре: 86–95. Дои:10.1145/224164.224187.
- ^ Основной доклад дня 1 - Бьярн Страуструп: стиль C ++ 11 в GoingNative 2012 на channel9.msdn.com с 45 минут или фольги с 44
- ^ Обработка чисел: почему вы никогда и НИКОГДА не должны использовать связанный список в своем коде снова в kjellkod.wordpress.com
- ^ Бродник, Андрей; Карлссон, Сванте; Седжвик, Роберт; Munro, JI; Demaine, ED (1999), Массивы с изменяемым размером в оптимальном времени и пространстве (Технический отчет CS-99-09) (PDF), Департамент компьютерных наук, Университет Ватерлоо
- ^ Гудрич, Майкл Т.; Клосс II, Джон Г. (1999), «Многоуровневые векторы: эффективные динамические массивы для ранговых последовательностей», Практикум по алгоритмам и структурам данных, Конспект лекций по информатике, 1663: 205–216, Дои:10.1007/3-540-48447-7_21, ISBN 978-3-540-66279-2
- ^ Ситарски, Эдвард (сентябрь 1996 г.), "HATs: деревья хешированных массивов", Аллея алгоритмов, Журнал доктора Добба, 21 (11)
- ^ Бродник, Андрей; Карлссон, Сванте; Седжвик, Роберт; Munro, JI; Demaine, ED (1999), Массивы с изменяемым размером в оптимальное время и пространство (PDF) (Технический отчет CS-99-09), Департамент компьютерных наук, Университет Ватерлоо
- ^ Багвелл, Фил (2002), Быстрые функциональные списки, хеш-списки, Deques и массивы переменной длины, EPFL
- ^ Javadoc включен
ArrayList
- ^ Класс ArrayList
внешняя ссылка
- Словарь алгоритмов и структур данных NIST: динамический массив
- VPOOL - Реализация динамического массива на языке Си.
- Коллекция - Профилировщик Java с явной поддержкой для отладки проблем, связанных с ArrayList и Vector.
- Структуры открытых данных - Глава 2 - Списки на основе массивов, Пэт Морин