Получение объекта по номеру — различия между версиями
Dima32ml (обсуждение | вклад) (→Описание алгоритма) |
Dima32ml (обсуждение | вклад) |
||
Строка 9: | Строка 9: | ||
Комбинаторные объекты занумерованы с <tex>0</tex>. Переход к нумерации с единицы можно сделать с помощью одной операции декремента перед проходом алгоритма. | Комбинаторные объекты занумерованы с <tex>0</tex>. Переход к нумерации с единицы можно сделать с помощью одной операции декремента перед проходом алгоритма. | ||
'''function''' num2object(numOfObject: '''int''') | '''function''' num2object(numOfObject: '''int''') | ||
− | '''for''' i = 1 '''to''' n | + | '''for''' i = 1 '''to''' n |
− | '''for''' j = 1 '''to''' k | + | '''for''' j = 1 '''to''' k |
− | '''if''' j- | + | '''if''' j-й элемент можно поставить на i-e место |
'''if''' numOfObject >= (количество комбинаторных объектов с префиксом object[1..i-1] и элементом j на месте i) | '''if''' numOfObject >= (количество комбинаторных объектов с префиксом object[1..i-1] и элементом j на месте i) | ||
numOfObject -= (количество комбинаторных объектов с префиксом object[1..i-1] и элементом j на месте i) | numOfObject -= (количество комбинаторных объектов с префиксом object[1..i-1] и элементом j на месте i) | ||
Строка 24: | Строка 24: | ||
Заметим, что всем префиксам на каждом шаге будет соответствовать диапазон номеров одинакового размера, (так как количество всевозможных суффиксов зависит только от длины) то есть можем просто посчитать "количество диапазонов, которые идут до нас" (количество цифр уже полностью занятых перестановками с меньшим номером) за <tex>O(1) </tex>: | Заметим, что всем префиксам на каждом шаге будет соответствовать диапазон номеров одинакового размера, (так как количество всевозможных суффиксов зависит только от длины) то есть можем просто посчитать "количество диапазонов, которые идут до нас" (количество цифр уже полностью занятых перестановками с меньшим номером) за <tex>O(1) </tex>: | ||
− | *k {{---}} номер искомой последовательности | + | *<tex>\mathtt{k}</tex> {{---}} номер искомой последовательности |
− | *n! {{---}} количество перестановок размера <tex>n</tex>. | + | *<tex>\mathtt{n!}</tex> {{---}} количество перестановок размера <tex>n</tex>. |
− | *permutation[n] {{---}} искомая перестановка. | + | *<tex>\mathtt{permutation[n]}</tex> {{---}} искомая перестановка. |
− | *was[n] {{---}} использовали ли мы уже эту цифру в перестановке. | + | *<tex>\mathtt{was[n]}</tex> {{---}} использовали ли мы уже эту цифру в перестановке. |
На <tex>i</tex>-ом шаге: | На <tex>i</tex>-ом шаге: | ||
− | *alreadyWas {{---}} сколько цифр уже полностью заняты перестановками с меньшим номером. | + | *<tex>\mathtt{alreadyWas}</tex> {{---}} сколько цифр уже полностью заняты перестановками с меньшим номером. |
− | *мы должны поставить ту цифру, которая еще полностью не занята, то есть цифру с номером alreadyWas + 1. Среди цифр, которых еще нет в нашем префиксе, считаем, что это цифра j. | + | *мы должны поставить ту цифру, которая еще полностью не занята, то есть цифру с номером <tex>alreadyWas + 1</tex>. Среди цифр, которых еще нет в нашем префиксе, считаем, что это цифра <tex>j</tex>. |
На <tex>j</tex>-ом шаге: | На <tex>j</tex>-ом шаге: | ||
− | *curFree {{---}} если элемент с номером <tex>j</tex> свободен, то он имеет номер curFree среди всех свободных элементов с 1 по <tex>j</tex> | + | *<tex>\mathtt{curFree}</tex> {{---}} если элемент с номером <tex>j</tex> свободен, то он имеет номер curFree среди всех свободных элементов с <tex>1</tex> по <tex>j</tex> |
'''function''' num2permutation(k: '''int''') | '''function''' num2permutation(k: '''int''') | ||
'''for''' i = 1 '''to''' n '''do''' | '''for''' i = 1 '''to''' n '''do''' | ||
Строка 54: | Строка 54: | ||
'''return''' permutation | '''return''' permutation | ||
− | Данный алгоритм работает за <tex>O(n^2)</tex>, так как в случае перестановок <tex>n=k</tex>. Мы можем посчитать все n! за <tex>O(n) </tex>. Асимптотику можно улучшить | + | Данный алгоритм работает за <tex>O(n^2)</tex>, так как в случае перестановок <tex>n=k</tex>. Мы можем посчитать все <tex>n!</tex> за <tex>O(n) </tex>. Асимптотику можно улучшить |
− | до <tex>O(n \log {n}) </tex>, если использовать структуры данных (например, [[Декартово дерево|декартово дерево]] по неявному ключу), которые позволяют искать <tex>i</tex>- | + | до <tex>O(n \log {n}) </tex>, если использовать структуры данных (например, [[Декартово дерево|декартово дерево]] по неявному ключу), которые позволяют искать <tex>i</tex>-й элемент множества и удалять элемент |
множества за <tex>O( \log {n}) </tex>. | множества за <tex>O( \log {n}) </tex>. | ||
Строка 62: | Строка 62: | ||
При построении битовых векторов можно не проверять условие возможности постановки какого-то объекта на текущее место. На каждый позиции может стоять один из двух элементов, независимо от того, какие элементы находятся в префиксе. Так как у нас всего два возможных элемента, упростим второй цикл до условия: | При построении битовых векторов можно не проверять условие возможности постановки какого-то объекта на текущее место. На каждый позиции может стоять один из двух элементов, независимо от того, какие элементы находятся в префиксе. Так как у нас всего два возможных элемента, упростим второй цикл до условия: | ||
− | *bitvector[n] {{---}} искомый битовый вектор | + | *<tex>\mathtt{bitvector[n]}</tex> {{---}} искомый битовый вектор |
− | *numOfBitvector {{---}} номер искомого вектора среди всех битовых векторов | + | *<tex>\mathtt{numOfBitvector}</tex> {{---}} номер искомого вектора среди всех битовых векторов |
− | *pow(2, n) {{---}} <tex>2^{n}</tex> количество битовых векторов длины <tex>n</tex> | + | *<tex>\mathtt{pow(2, n)}</tex> {{---}} <tex>2^{n}</tex> количество битовых векторов длины <tex>n</tex> |
'''function''' num2bitvector(numOfBitvector: '''int''') | '''function''' num2bitvector(numOfBitvector: '''int''') | ||
'''for''' i = 1 '''to''' n '''do''' | '''for''' i = 1 '''to''' n '''do''' | ||
Строка 78: | Строка 78: | ||
== См. также == | == См. также == | ||
*[[Получение номера по объекту|Получение номера по объекту]] | *[[Получение номера по объекту|Получение номера по объекту]] | ||
+ | |||
+ | == Источники информации == | ||
*Программирование в алгоритмах / С. М. Окулов. — М.: БИНОМ. Лаборатория знаний, 2002. стр.31 - ISBN 5-94774-010-9 | *Программирование в алгоритмах / С. М. Окулов. — М.: БИНОМ. Лаборатория знаний, 2002. стр.31 - ISBN 5-94774-010-9 | ||
[[Категория: Дискретная математика и алгоритмы]] | [[Категория: Дискретная математика и алгоритмы]] | ||
[[Категория: Комбинаторика]] | [[Категория: Комбинаторика]] |
Версия 16:39, 10 декабря 2014
Описание алгоритма
Получаем элементы объекта по порядку: сначала определим какой элемент будет стоять на первом месте, потом на втором и так далее. Считаем, что мы нашли первые
элементов объекта. Для всех вариантов элемента, который может стоять на позиции с номером , посчитаем диапазон номеров, который будет соответствовать объектам с данным префиксом. Если искомый номер входит в один из диапазонов, то, очевидно, мы нашли элемент, который должен стоять на месте с номером . Диапазоны номеров не пересекаются, значит на это место больше нельзя поставить никакой другой элемент.- В начале каждого шага — номер нужного объекта среди тех, у которых префикс до -го элемента лексикографически равен префиксу нашего объекта.
- — количество мест в комбинаторном объекте (например, битовый вектор длины )
- — количество различных элементов, которые могут находиться в данном комбинаторном объекте. Например, для битового вектора , поскольку возможны только и . Все элементы занумерованы в лексикографическом порядке, начиная с .
Комбинаторные объекты занумерованы с
. Переход к нумерации с единицы можно сделать с помощью одной операции декремента перед проходом алгоритма.function num2object(numOfObject: int) for i = 1 to n for j = 1 to k if j-й элемент можно поставить на i-e место if numOfObject >= (количество комбинаторных объектов с префиксом object[1..i-1] и элементом j на месте i) numOfObject -= (количество комбинаторных объектов с префиксом object[1..i-1] и элементом j на месте i) else object[i] = j return object
Сложность алгоритма — комбинаторных объектов по номеру.
. Количества комбинаторных объектов с заданными префиксами считаются известными, и их подсчет в сложности не учитывается. Стоит отметить, что подсчет количества комбинаторных объектов с заданным префиксом зачастую является задачей с достаточно большой вычислительной сложностью. Приведем примеры получения некоторыхПерестановки
Рассмотрим алгоритм получения лексикографическом порядке перестановки размера . Заметим, что всем префиксам на каждом шаге будет соответствовать диапазон номеров одинакового размера, (так как количество всевозможных суффиксов зависит только от длины) то есть можем просто посчитать "количество диапазонов, которые идут до нас" (количество цифр уже полностью занятых перестановками с меньшим номером) за :
-ой в- — номер искомой последовательности
- — количество перестановок размера .
- — искомая перестановка.
- — использовали ли мы уже эту цифру в перестановке.
На
-ом шаге:- — сколько цифр уже полностью заняты перестановками с меньшим номером.
- мы должны поставить ту цифру, которая еще полностью не занята, то есть цифру с номером . Среди цифр, которых еще нет в нашем префиксе, считаем, что это цифра .
На
-ом шаге:- — если элемент с номером свободен, то он имеет номер curFree среди всех свободных элементов с по
function num2permutation(k: int) for i = 1 to n do alreadyWas = k div (n-i)! k = k mod (n-i)! curFree = 0 for j = 1 to n if was[j] == false curFree++ if curFree == alreadyWas + 1 permutation[i] = j was[j] = true return permutation
Данный алгоритм работает за декартово дерево по неявному ключу), которые позволяют искать -й элемент множества и удалять элемент множества за .
, так как в случае перестановок . Мы можем посчитать все за . Асимптотику можно улучшить до , если использовать структуры данных (например,Битовые вектора
Рассмотрим алгоритм получения
-ого в лексикографическом порядке битового вектора размера . При построении битовых векторов можно не проверять условие возможности постановки какого-то объекта на текущее место. На каждый позиции может стоять один из двух элементов, независимо от того, какие элементы находятся в префиксе. Так как у нас всего два возможных элемента, упростим второй цикл до условия:- — искомый битовый вектор
- — номер искомого вектора среди всех битовых векторов
- — количество битовых векторов длины
function num2bitvector(numOfBitvector: int) for i = 1 to n do if numOfBitvector >= pow(2, (n-i)) numOfBitvector -= pow(2, (n-i)) bitvector[i] = 1 else bitvector[i] = 0 return bitvecor
Данный алгоритм работает за
, так как в случае битовых векторов не зависит от .См. также
Источники информации
- Программирование в алгоритмах / С. М. Окулов. — М.: БИНОМ. Лаборатория знаний, 2002. стр.31 - ISBN 5-94774-010-9