Предикат "левый поворот" — различия между версиями
Komarov (обсуждение | вклад) |
|||
Строка 1: | Строка 1: | ||
− | Даны два отрезка, которые задаются начальной и конечной точками <tex>a,b\ \mathcal{2}\ \mathbb R^2</tex> и определяются как множества точек <tex>s\ =\ \{(1-t)a + tb,\ t\ \ | + | Даны два отрезка, которые задаются начальной и конечной точками <tex>a,b\ \mathcal{2}\ \mathbb R^2</tex> и определяются как множества точек <tex>s\ =\ \{(1-t)a + tb,\ t\ \in [0;1]\}</tex>. Требуется проверить существование множества их общих точек. Для определения этого факта в вычислительной геометрии используется предикат ''левый поворот'' (или ''по часовой стрелке''). |
Рассмотрим возможные расположения точек и самих отрезков относительно друг друга: | Рассмотрим возможные расположения точек и самих отрезков относительно друг друга: | ||
Строка 11: | Строка 11: | ||
<tex dpi = "120"> | <tex dpi = "120"> | ||
$$ | $$ | ||
− | + | \operatorname{LeftTurn}(a, b, c) =\left\{ | |
\begin{array}{rl} | \begin{array}{rl} | ||
-1 &\mbox{, for}\ (b - a)\times(c - a) < 0\\ | -1 &\mbox{, for}\ (b - a)\times(c - a) < 0\\ | ||
Строка 27: | Строка 27: | ||
Допустим, что все числа положительные и будем писать без модулей: | Допустим, что все числа положительные и будем писать без модулей: | ||
− | '''Замечание:''' при сложении складываются | + | '''Замечание:''' при сложении складываются абсолютные погрешности, при умножении складываются относительные погрешности. |
− | <tex dpi = "150"> \delta (b - a)\times(c - a) = A \varepsilon (\frac{(b_x + a_x)}{(b_x \cdot a_x)} + \frac{(c_y + a_y)}{(c_y \cdot a_y)}) + B \varepsilon (\frac{(b_y + a_y)}{(b_y \cdot a_y)} + \frac{(c_x + a_x)}{(c_x \cdot a_x)})</tex> | + | <tex dpi = "150"> \delta (b - a)\times(c - a) = A \varepsilon \left(\frac{(b_x + a_x)}{(b_x \cdot a_x)} + \frac{(c_y + a_y)}{(c_y \cdot a_y)}\right) + B \varepsilon \left(\frac{(b_y + a_y)}{(b_y \cdot a_y)} + \frac{(c_x + a_x)}{(c_x \cdot a_x)}\right)</tex> |
− | Заметим, что все координаты (а значит и наши вычисления) производятся в вещественных числах, а это значит, что при вычислениях мы можем допустить ошибку. Более точно определить знак нашего выражения поможет вычисление с [[Интервальная арифметика | | + | Заметим, что все координаты (а, значит, и наши вычисления) производятся в вещественных числах, а это значит, что при вычислениях мы можем допустить ошибку. Более точно, определить знак нашего выражения поможет вычисление с [[Интервальная арифметика |''интервальной арифметикой'']]. Все исходные переменные будут вырожденными интервалами. Из-за погрешностей, возникающих при округлении вещественных чисел, истинные значения операций нам будут неизвестны, но они обязательно будет содержаться в посчитанных интервалах. Для точного вычисления необходимо |
фильтрованное вычисление предикатов. | фильтрованное вычисление предикатов. | ||
Строка 42: | Строка 42: | ||
<tex dpi = 130>\mid\delta_i\mid \le \varepsilon</tex>; | <tex dpi = 130>\mid\delta_i\mid \le \varepsilon</tex>; | ||
− | Именно поэтому, когда угол между отрезками АВ и АС | + | Именно поэтому, когда угол между отрезками АВ и АС ''крайне мал'', мы можем получить неверное значение предиката. |
[[Файл:Tiny_angle.jpg]] | [[Файл:Tiny_angle.jpg]] | ||
=Bounding box= | =Bounding box= | ||
− | Ещё следует обратить внимание на граничные случаи, когда какие-то точки попадают на саму прямую. При этом возникает единственный особый случай, когда вышеописанные проверки ничего не дадут — случай, когда оба отрезка лежат на одной прямой. Этот случай надо рассмотреть отдельно. Для этого достаточно проверить, что проекции этих двух отрезков на оси X и Y пересекаются (часто эту проверку называют | + | Ещё следует обратить внимание на граничные случаи, когда какие-то точки попадают на саму прямую. При этом возникает единственный особый случай, когда вышеописанные проверки ничего не дадут — случай, когда оба отрезка лежат на одной прямой. Этот случай надо рассмотреть отдельно. Для этого достаточно проверить, что проекции этих двух отрезков на оси X и Y пересекаются (часто эту проверку называют ''проверкой на bounding box''). |
[[Файл:Bounting_box().png]] | [[Файл:Bounting_box().png]] |
Версия 13:24, 16 марта 2012
Даны два отрезка, которые задаются начальной и конечной точками
и определяются как множества точек . Требуется проверить существование множества их общих точек. Для определения этого факта в вычислительной геометрии используется предикат левый поворот (или по часовой стрелке). Рассмотрим возможные расположения точек и самих отрезков относительно друг друга:Определим, лежат ли точки концов отрезков по разные стороны от другого отрезка.
Определение: |
Распишем подробнее:
Какие при этом у нас будут погрешности? Допустим, что все числа положительные и будем писать без модулей:
Замечание: при сложении складываются абсолютные погрешности, при умножении складываются относительные погрешности.
Заметим, что все координаты (а, значит, и наши вычисления) производятся в вещественных числах, а это значит, что при вычислениях мы можем допустить ошибку. Более точно, определить знак нашего выражения поможет вычисление с интервальной арифметикой. Все исходные переменные будут вырожденными интервалами. Из-за погрешностей, возникающих при округлении вещественных чисел, истинные значения операций нам будут неизвестны, но они обязательно будет содержаться в посчитанных интервалах. Для точного вычисления необходимо фильтрованное вычисление предикатов.
;
Именно поэтому, когда угол между отрезками АВ и АС крайне мал, мы можем получить неверное значение предиката.
Bounding box
Ещё следует обратить внимание на граничные случаи, когда какие-то точки попадают на саму прямую. При этом возникает единственный особый случай, когда вышеописанные проверки ничего не дадут — случай, когда оба отрезка лежат на одной прямой. Этот случай надо рассмотреть отдельно. Для этого достаточно проверить, что проекции этих двух отрезков на оси X и Y пересекаются (часто эту проверку называют проверкой на bounding box).