3.2. Основные алгоритмы двумерного отсечения и их идеи

Алгоритмов двумерного отсечения огромное количество. Кратко рассмотрим некоторые из них:

Простой алгоритм двумерного отсечения
В этом алгоритме отрезок отсекается поочередно каждой из сторон окна, а для полученных точек пересечения проверяется их принадлежность внутренней области окна, т. е. корректность пересечения. Эта процедура применяется сначала к исходному отрезку Р1Р2 и получается отрезок Р12, а затем к отрезку Р12 и получается результирующий отрезок Р12'.
1. Вычисление кодов концевых точек отрезка.
2. Занесение этих кодов в 2 массива, размерностью 1х4 каждый.
3. Проверка полной видимости отрезка. Если отрезок полностью видим goto 13.
4. Проверка тривиальной невидимости отрезка. Если отрезок полностью невидим goto 13.
5. Отрезок может быть частично видим. Проверяется попадание концевых точек отрезка внутрь окна. Если концевые точки внутри окна есть, goto 8.
6. Внутри окна нет концов отрезка.
7. Инициализация номера конца отрезка. Если обработано оба конца отрезка, goto 13.
8. Проверяется вертикальность отрезка. Если отрезкок вертикален, goto 11.
9. Ищутся точки пересечения отрезка с левым и правым краями окна. Если пересечение обнаружено, goto 7.
10. Проверяется горизонтальность отрезка. Если отрезок горизонтален, goto 7.
11. Проверяется пересечение отрезка с верхним и нижним краями окна. Если пересечение обнаружено, goto 7.
12. Если пересечений не обнаружено, то отрезок невидим.
13. Завершение работы и вызов процедуры черчения.
14. Перейти к обработке следующего отрезка.

В примере 3.1 показано, как этот простой метод позволяет отбросить некорректные пересечения путем простого сравнения координат точек пересечения с координатами сторон окна.

Пример 3.1. Простое двумерное отсечение.
Рассмотрим отсекаюшее окно и отрезки, изображенные на рис. 3.2.

 

Рис. 3.2. Двумерное параметрическое отсечение

 

Наклон отрезка от Р1(- 3/2,1/6) до Р2(1/2,3/2) равен m = (y2 - y1) / (x2 - x1) = (3/2 - 1/6) / [1/2 - (- 3/2)] = 2/3. Его пересечения со сторонами окна таковы:
с левой: х = -1     y = (2/3)[-1 - (-3/2)] + 1/6 = 1/2
с правой: х = 1     y = (2/3)[1 - (-3/2)] + 1/6 = 11/6
(последнее число больше, чем yв, и поэтому отвергается)
с верхней: y = 1     x = -3/2 + (3/2)[1 - 1/6] = -1/4
c нижней: y = -1     х = -3/2 + (3/2)|-1 - 1/6] = -13/4
(последнee число меньше, чем хп, и поэтому отвергается)
Аналогично, отрезок от Р3(-3/2,-1) до Р4(3/2,2) имеет

m = y2 - y1

x2 - x1
= 2 - (-1)

3/2 - (-3/2)
= 1

и пересечения:
с левой: х = -1     y = (1)[-1 - (-3/2)] + (-1) = -1/2
с правой: х = 1     y = (1)[1 - (-3/2)] + (-1) = 3/2
(последнее число больше, чем yв, и поэтому отвергается)
с верхней: y = 1     x = -3/2 + (1)[1 - (-1)] = 1/2
c нижней: y = -1     х = -3/2 + (1)|-1 - (-1)] = -3/2
(последнee число меньше, чем хл, и поэтому отвергается)

Алгоритм отсечения Сазерленда-Коэна.
В алгоритме Сазерленда-Коэна отрезок тоже разбивается сторонами окна. Отличие состоит в том, что здесь не производится проверки попадания точки пересечения внутрь окна, вместо этого каждая из пары получающихся частей отрезка сохраняется или отбрасывается в результате анализа кодов ее концевых точек. Для определения той из девяти областей, которой принадлежит конец ребра, вводится четырехразрядный (битовый) код. Крайний правый бит кода считается первым. В соответствующий бит заносится 1 при выполнении следующих условий:
для первого бита - если точка левее окна
для второго бита - если точка правее окна
для третьего бита - если точка ниже окна
для четвертого бита - если точка выше окна
В противном случае в бит заносится нуль.
Ключом к алгоритму Сазерленда-Коэна является информация о том, что одна из концевых точек отрезка лежит вне окна. Поэтому тот отрезок, который заключен между этой точкой и точкой пересечения, можно отвергнуть как невидимый. Фактически это означает замену исходной концевой точки на точку пересечения. В содержательных понятиях алгоритм Сазерленда - Коэна формулируется следующим образом:
Для каждой стороны окна выполнить:
1. Для каждого отрезка Р1P2 определить, не является ли он полностью видимым или может быть тривиально отвергнут как невидимый.
2. Если Р1 вне окна, то продолжить выполнение, иначе поменять Р1 и Р2 местами.
3. Заменить Р1 на точку пересечения Р1Р2 со стороной окна.
Этот алгоритм иллюстрирует следующий пример.

Пример 3.2. Алгоритм отсечения Саэерленда-Коэна.
Рассмотрим вновь отсечение отрезка Р1Р2 окном, показанным на рис. 3.2. Коды концевых точек Р1(- 3/2,1/6) и Р2(1/2,3/2) равны (0001) и (1000) соответственно. Этот отрезок не является ни полностью видимым, ни тривиально невидимым. Отрезок пересекает левую сторону окна. Р1 - вне окна.
Пересечение с левой стороной (х = -1) окна происходит в точке Р1'(-1,1/2). Замена Р1 на Р1' дает новый отрезок от Р1 (-1,1/2) до Р2(1/2,3/2).
Коды концевых точек Р1 и Р2 теперь стали (0000) и (1000) соответственно. Отрезок не является ни полностью видимым, ни тривиально невидимым.
Отрезок не пересекается с правой стороной окна. Перейти к нижней стороне.
Коды концевых точек Р1 и Р2 остаются по-прежнему равными (0000) в (1000) соотвественно.Отрезок не является ни полностью видимым, ни тривиально невидимым.
Отрезок не пересекается с нижней стороной окна. Перейти к верхней стороне.
Коды концевых точек Р1 и Р2 остаются равными (0000) и (1000) соответственно. Отрезок не является ни полностью видимым, ни тривиально невидимым.
Отрезок пересекается с верхней стороной окна. Р1 - не снаружи окна. Поменяв Р1 и Р2 местами, мы получим новый отрезок от Р1(1/2,3/2) до P2(-1,1/2).
Точка пересечення отрезка с верхней стороной окна (y = 1) равна Р1'(-1/4,1). Заменив Р1 на P1', получаем новый отрезок от Р1(-1/4,1) до P2(-1,1/2).
Теперь коды концевых точек Р1 и Р2 равны (0000) и (0000) соответственно. Oтрезок полностью видим.
Процедура завершена.
Начертить отрезок.

Алгоритм разбиения средней точкой.
В предыдущем алгоритме требовалось вычислить пересечение отрезка со стороной окна. Можно избежать непосредственного вычисления, если реализовать двоичный поиск такого пересечения путем деления отрезка его средней точкой. Программная реализация этого алгоритма медленнее, чем реализация предыдущего алгоритма. Аппаратная же реализация быстрее и эффективнее, поскольку аппаратные сложения и деления на 2 очень быстры. Деление на 2 аппаратно эквивалентно сдвигу каждого бита вправо.

В алгоритме используются коды концевых точек отрезка и проверки, выявляющие полную видимость и тривиальную невидимость отрезков. Те отрезки, которые только с помощью таких простых проверок нельзя отнести к одной из двух категорий, разбиваются на две равные части. Затем те же проверки применяются к каждой из половин до тех пор, пока не будет обнаружено пересечение со стороной окна или длина разделяемого отрезка не станет пренебрежимо малой, т. е. пока он не выродится в точку. После вырождения определяется видимость полученной точки. В результате процесс обнаружения пересечения сводится к двоичному поиску. Максимальное число разбиений пропорционально точности задания координат концевых точек отрезка.

Данный алгоритм можно формально разбить на три этапа:
Для каждой концевой точки отрезка:
1. Если концевая точка видима, то она будет наиболее удаленной видимой точкой. Процесс завершен. Иначе - продолжить.
2. Если отрезок тривиально характеризуется как невидимый, то выходная информация не формируется. Процесс завершен. Иначе - продолжить.
3. Грубо оценить наиболее удаленную видимую точку путем деления отрезка Р1Р2 средней точкой Рm. Применить вышеизложенные тесты к двум кускам Р1Рm и РmР2. Если РmР2 тривиально отвергается как невидимый, то средняя точка дает верхнюю оценку для наиболее удаленной видимой точки. Продолжить процедуру с отрезком P1Рm. Иначе - средняя точка дает оценку снизу для наиболее удаленной видимой точки. Продолжить процедуру с куском Р2Рm. Если отрезок становится настолько мал, что его средняя точка совпадает с его концами с машинной или наперед заданной точностью, то надо оценить ее видимость и закончить процесс. Конкретный пример лучше проиллюстрирует этот алгоритм.

Пример 3.3. Разбиение средней точкой.
Рассмотрим окно, имеющее экранные координаты левой, правой, нижней и верхней сторон: 0, 1023, 0, 1023 соотвественно. Экранные координаты концов отрезка с равны: Р1(-307,631) и Р2 (820,-136). Коды концевых точек равны: для Р1 - (0001), а для Р2 - (0100). Оба кода не равны нулю, поэтому отрезок не является полностью видимым. Логическое произведение кодов концевых точек равно (0000). Значит, отрезок нельзя тривиально отвергнуть как невидимый. Займемся поиском пересечений.
Координаты средней точки с учетом округления равны:

 

xm = x2 + x1

2
= 820 - 307

2
= 256.5 = 256
ym = y2 + y1

2
= -136 + 631

2
= 247.5 = 247

Код средней точки равен (0000). Оба отрезка Р1Рm и Р2Рm не являются ни полностью видимыми, ни тривиально невидимыми. Отложим на время отрезок Р2Рm и займемся отрезком Р1Рm. Процесс разбиения отрезков показан в табл. 3.1.

 

P1
P2
Pm
Примечания
-307, 631 820, -136 256, 247 Запомнить РmР2
Продолжить с Р1Рm
-307, 631 256, 247 -26, 439 Продолжить с РmР2
-26, 439 256, 247 115, 343 Продолжить с Р1Рm
-26, 439 115, 343 44, 391 Продолжить с Р1Рm
-26, 439 44, 391 9, 415 Продолжить с Р1Рm
-26, 439 9, 415 -9, 427 Продолжить с РmР2
-9, 427 9, 415 0, 421 Найдено пересечение
256, 247 820, -136 538, 55 Восстановить РmР2
Продолжить с РmР2
538, 55 820, -136 679, -41 Продолжить с Р1Рm
538, 55 679, -41 608, 7 Продолжить с РmР2
608, 7 679, -41 643, -17 Продолжить с Р1Рm
608, 7 643, -17 625, -5 Продолжить с Р1Рm
608, 7 625, -5 616, 1 Продолжить с РmР2
616, 1 625, -5 620, -2 Продолжить с Р1Рm
616, 1 620, -2 618, -1 Продолжить с Р1Рm
616, 1 618, -1 617, 0 Найдено пересечение

Таблица 3.1.

Использование точного уравнения отрезка P1P2 дает пересечения в точках (0, 422) и (620,0). Отличие этих значений от значений из табл. 3.1 объясняетcя погрешностью целочисленных округлений.

 

Назад
Компьютерная графика © 2014 ОСУ ИК Вход