Алгоритм Бентли — Оттманна
Алгоритм Бентли — Оттманна (1979) позволяет найти все точки пересечений прямолинейных отрезков на плоскости. В нем применяется метод выметающей прямой[1] (заметающей прямой[2], движущейся прямой[3], сканирующей линии[4]; англ. sweeping line). В методе используется вертикальная выметающая прямая, движущаяся слева направо, при этом отрезки, которые она пересекает при данной координате , можно упорядочить по координате , тем самым их можно сравнивать между собой (какой выше, какой ниже). Это сравнение можно осуществить, например, используя уравнение прямой, проходящей через две точки (отрезки заданы двумя своими конечными точками): , где , и , — координаты, соответственно, первой и второй точек отрезка. Выметающая прямая перемещается по так называемым точкам событий (левым и правым концам отрезков, а также точкам пересечения отрезков). После точки пересечения отрезки следует менять местами, так как, например, самый верхний из пересекающихся отрезков после точки пересечения становится самым нижним. Приведенный ниже алгоритм не рассчитан на случай, когда два отрезка пересекаются больше, чем в одной точке.
NB Выметающую прямую можно также представить как горизонтальную, движущуюся сверху вниз по координате , и сравнивать пересекающие её отрезки по координате .
Обработка вертикальных отрезков
Возникает проблема с вертикальным отрезком в том смысле как его сравнивать по выше/ниже с пересекающими отрезками. Для этого можно, например, считать, что если точка пересечения вертикального с не вертикальным отрезков находится ниже текущей координаты точки события, то вертикальный отрезок находится выше, если точка пересечения выше текущей координаты точки события, то вертикальный отрезок считается ниже пересекающего его. Если текущая координата равна координате точки события, то при удалении отрезка считать, что вертикальный отрезок ниже, при вставке же считать, что он выше.
Квадрат памяти
В худшем случае, когда, например, все отрезки, пересекаясь между собой, образуют прямоугольную сетку, будет точек пересечений, которые надо будет хранить. Чтобы избежать использования квадрата памяти в алгоритме, можно удалять точку пересечения отрезков, которые временно перестают быть соседними при данном положении выметающей прямой. Эти точки все равно будут снова найдены при последующих шагах алгоритма, когда данные отрезки снова станут соседними (Печ, Шерир 1991).
Алгоритм
В приведенном ниже псевдокоде используются:
- Q — динамические структура данных без повторений с логарифмическим временем поиска, вставки, удаления точек событий и поиска минимального элемента (например, красно-чёрное дерево, 2-3-дерево).
- T — динамические структура данных без повторений с логарифмическим временем поиска, вставки, удаления отрезков. В ней хранятся все отрезки, пересекающие выметающую прямую (например, красно-чёрное дерево, 2-3-дерево).
- q — точка события.
- newq — только что найденная точка пересечения отрезков, точка события.
- L(q) — множество отрезков, левый конец которых — q (для вертикальных отрезков левым считается нижний конец).
- R(q) — множество отрезков, правым концом которых является q.
- I(q) — множество отрезков, пересекающихся в q.
segmentsIntersections(points[]) 1) Инициализируются Q и T. В Q заносятся все концы отрезков, упорядоченные по координате x, при этом, если две точки совпали, то левая конечная точка отрезка помещается перед правой. Если совпали только x, то точка с меньшим y является меньшей. T ← ∅ 2) while Q != ∅ q ← min(Q); processPoint(q);
processPoint(q) 1) Найти в Q все отрезки, содержащие q; // они в Q будут соседними, так как точки события, которые содержатся в этих отрезках, имеют одинаковые координаты; 2) if (|L(q)| + |R(q)| + |I(q)| > 1) ИЛИ (|I(q)| > 0) then Выдать в ответ точку q; 3) if (|I(q)| = 0) И (|L(q)|+|R(q)| > 0) // в рассматриваемой точке все отрезки только начинаются или заканчиваются; then I(q) ← I(q) ∪ L(q) ∪ R(q); // добавить в I(q) L(q) и R(q) 4) Удалить из T R(q) и I(q); 5) Вставить в T I(q) и L(q); // из T были удалены все отрезки из множества I(q), теперь же вставляются обратно в измененном порядке после точки пересечения; 6) if (L(q)∪I(q) = ∅) ИЛИ (|I(q)| = |R(q)| - 1) then Найти в T верхнего и нижнего соседей q su и sl; newq = findIntersect(su, sl); if newq != NULL then add(Q, newq); 7) else Пусть s′ — самый верхний отрезок из L(q)∪I(q); Пусть su — верхний сосед s′ в T; Пусть s′′ — самый нижний отрезок из L(q)∪ I(q); Пусть sl — нижний сосед s′′ в T; newq = findIntersect(su, s′); if newq != NULL then add(Q, newq); newq = findIntersect(sl, s′′); if newq != NULL then add(Q, newq); // далее убираем квадрат памяти; newq = findIntersect(sl, su); if newq != NULL then delete(Q, newq); newq = findIntersect(s′′, su); if newq != NULL then delete(Q, newq); newq = findIntersect(sl, s′); if newq != NULL then delete(Q, newq);
findIntersect(sl, su) if sl и su пересекаются правее заметающей прямой (или на заметающей прямой выше текущей точки события) then return newq; else return NULL;
Анализ
Пусть — число отрезков, — число отрезков, пересекающих точку . Тогда время на инициализацию Q равно , на инициализацию T — . На поиск всех отрезков, проходящих через точку и обновление Q, требуется времени. На обновление T также времени. Суммарно: , где — число точек пересечения . .
Память , благодаря тому, что удаляются точки пересечения отрезков, которые перестали быть соседними, иначе было бы , где .
Примечания
- ↑ Кормен, Т., Лейзерсон, Ч., Ривест, Р., Штайн, К. Алгоритмы: построение и анализ = Introduction to Algorithms / Под ред. И. В. Красикова. — 2-е изд. — М.: Вильямс, 2005. — 1296 с. — ISBN 5-8459-0857-4.
- ↑ Прапарата Ф., Шеймос М. Вычислительная геометрия: Введение = Computational Geometry An introduction. — М.: Мир, 1989. — 478 с.
- ↑ Кормен, Т., Лейзерсон, Ч., Ривест, Р. Алгоритмы: построение и анализ = Introduction to Algorithms / Пер. с англ. под ред. А. Шеня. — М.: МЦНМО, 2000. — 960 с. — ISBN 5-900916-37-5.
- ↑ Ласло М. Вычислительная геометрия и компьютерная графика на C++. — М.: БИНОМ, 1997. — 304 с.
Литература
- Берг М., Чеонг О., Кревельд М., Овермарс М. Вычислительная геометрия. Алгоритмы и приложения = Computational Geometry: Algorithms and Applications. — М.: ДМК-Пресс, 2016. — 438 с. — ISBN 978-5-97060-406-9.
- Mark de Berg, Marc van Kreveld, Mark Overmars, Otfried Schwarzkopf. Computational Geometry: Algorithms and Applications. — Springer, 2000. — 368 с.
- Прапарата Ф., Шеймос М. Вычислительная геометрия: Введение = Computational Geometry An introduction. — М.: Мир, 1989. — 478 с.
- Ласло М. Вычислительная геометрия и компьютерная графика на C++. — М.: БИНОМ, 1997. — 304 с.
- Т. Кормен, Ч. Лейзерсон, Р. Ривест, К. Штайн. Алгоритмы. Построение и анализ = Introduction to Algorithms. — 2-e изд. — “Вильямс”, 2005. — 1296 с.
Ссылки
- Визуализатор — частный случай (алгоритм Шеймоса — Гоя, 1976 (определение наличия пересекающихся отрезков)).