74.htm 9.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179
  1. <!--
  2. demo.design 3D programming FAQ
  3. Idea, texts, screenshots:
  4. Andrew A. Aksyonoff,
  5. shodan@chat.ru
  6. Web-design, illustrations:
  7. Andrey Samoilov,
  8. asy@sense.simbirsk.su
  9. -->
  10. <html>
  11. <head>
  12. <title>demo.design 3D programming FAQ. Разное. Алгоритм "бегущих кубиков" для полигонизации изоповерхностей.</title>
  13. <link rel=stylesheet href="../style.css" type="text/css">
  14. </head>
  15. <script language="javascript">
  16. <!--//
  17. browser = navigator.appName;
  18. version = parseFloat(navigator.appVersion);
  19. if (browser == "Netscape" && version >= 3.0) { jsenabled = 1; } else
  20. if (browser == "Microsoft Internet Explorer" && version >= 3.0) { jsenabled = 1; } else { jsenabled = 0; }
  21. function swap(img,ref) { if (jsenabled) {document.images[img].src = ref;} }
  22. function loadtocache(img,ref) { cache[img] = new Image(); cache[img].src = ref; }
  23. if (jsenabled) {
  24. cache = new Array();
  25. loadtocache(0,"../img/xdl.gif");
  26. loadtocache(1,"../img/xfaq.gif");
  27. loadtocache(2,"../img/xlinks.gif");
  28. loadtocache(3,"../img/xauthor.gif");
  29. loadtocache(4,"../img/xe.gif");
  30. loadtocache(5,"../img/xprev.gif");
  31. loadtocache(6,"../img/xnext.gif");}
  32. //-->
  33. </script>
  34. <body bgcolor=white><center>
  35. <!-- Title -->
  36. <img src="../img/b.gif" width=500 height=1 alt=""><br>
  37. <img src="../img/t.gif" width=500 height=1 alt=""><br>
  38. <img src="../img/b.gif" width=500 height=1 alt=""><br>
  39. <img src="../img/t.gif" width=500 height=2 alt=""><br>
  40. <table width=500 cellpadding=0 cellspacing=0 border=0>
  41. <td><img src="../img/t.gif" width=5 height=1 alt=""><a href="../main.htm" onmouseover="swap('logo','../img/xe.gif');" onmouseout="swap('logo','../img/e.gif');"><img src="../img/e.gif" name=logo width=60 height=50 hspace=10 border=0 alt=" в самое начало "></a></td>
  42. <td><p class=pagetitle><img src="../img/t.gif" width=265 height=1 alt=""><br>demo.design<br>3D programming FAQ</td>
  43. <td align=center><p class=navy><a href="../download.htm" onmouseover="swap('dl','../img/xdl.gif');" onmouseout="swap('dl','../img/dl.gif');"><img src="../img/dl.gif" name=dl width=40 height=40 border=0 hspace=5 alt=" download "></a><br>download</td>
  44. <td align=center><p class=navy><a href="../links.htm" onmouseover="swap('links','../img/xlinks.gif');" onmouseout="swap('links','../img/links.gif');"><img src="../img/links.gif" name=links width=40 height=40 border=0 hspace=5 alt=" коллекция линков "></a><br>links</td>
  45. <td align=center><p class=navy><a href="../author.htm" onmouseover="swap('author','../img/xauthor.gif');" onmouseout="swap('author','../img/author.gif');"><img src="../img/author.gif" name=author width=40 height=40 border=0 hspace=5 alt=" автора! "></a><br>author</td>
  46. </table>
  47. <img src="../img/t.gif" width=500 height=4 alt=""><br><img src="../img/b.gif" width=500 height=1 alt=""><br>
  48. <!-- Head -->
  49. <table width=500 cellpadding=0 cellspacing=10 border=0><td><div align=justify>
  50. <p class=title>
  51. <img src="../img/b7.gif" width=70 height=70 align=left hspace=0 alt="">
  52. <img src="../img/t.gif" width=5 height=70 align=left hspace=0 alt="">
  53. РАЗНОЕ<br>7.4. Алгоритм "бегущих кубиков" для полигонизации изоповерхностей
  54. <!-- Article -->
  55. <p>Сооветствующий английский термин - marching cubes algorithm; приведен здесь,
  56. так сказать, just to avoid any confusion. Алгоритм предназначен для быстрого
  57. построения полигональной модели изоповерхности трехмерного скалярного поля,
  58. заданного значениями на равномерной сетке. Выражаясь менее академичным и,
  59. надеюсь, более понятным языком, этот алгоритм нужен для быстрого построения
  60. набора треугольных граней, достаточно хорошо приближающего изоповерхность,
  61. то есть такую поверхность, на которой определенная функция равна какой-то
  62. константе - изоуровню. Например, сфера радиуса 1 с центром в нуле - это
  63. изоповерхность функции f=1/(x*x+y*y+z*z) для изоуровня 1. Т.н. "3D капли",
  64. столь любимые ныне, тоже являются изоповерхностью, правда, немного более
  65. сложной функции (да, функция задана в трехмерном пространстве):
  66. <pre class=formula>
  67. n
  68. f = sum c[i]/((x[i]-x)*(x[i]-x)+(y[i]-y)*(y[i]-y)+(z[i]-z)*(z[i]-z)),
  69. i=1
  70. </pre>
  71. <p>где
  72. <p><table width=480 cellspacing=0 cellpadding=0 border=0 align=center>
  73. <tr><td><p class=expression>n</td><td>количество источников поля (капелек)</td></tr>
  74. <tr><td><p class=expression>c[i]</td><td>интенсивность каждого источника (радиус капельки)</td></tr>
  75. <tr><td><p class=expression>x[i], y[i], z[i]</td><td>координаты каждого источника (центра капельки)</td></tr>
  76. </table>
  77. <p>Работает алгоритм следующим образом.
  78. <p>Рассматриваем какой-то параллелипипед, внутри которого заведомо находится
  79. изоповерхность (или та ее часть, которую мы хотим нарисовать). Разбиваем
  80. его сеткой, ну, как бы разрезаем его на несколько меньших параллелипипедов,
  81. и считаем значения функции (поля) в узлах сетки, то есть вершинах этих самых
  82. маленьких параллелипипедов. Для большей ясности можно заменить длинное слово
  83. "параллелипипед" на слово "куб" и представить себе кубик Рубика. Теперь
  84. проходим по всем кубикам (дальше я буду везде использовать слово "кубик",
  85. надоело "параллелипипед" писать). Смотрим на значения функции в вершинах
  86. кубика. Если все они больше (или меньше) изоуровня - значит, кубик находится
  87. целиком над (или под) изоповерхностью, внутри этого кубика поверхности нет и
  88. мы его просто отбрасываем. Если же часть больше, а часть меньше, то некоторые
  89. ребра кубика пересекаются с изоповерхностью. Линейной интерполяцией приближаем
  90. эти точки пересечения и в зависимости от того, какие вершины находятся над
  91. изоповерхностью, а какие под, генерируем несколько треугольных граней. Все
  92. вершины этих граней - это как раз точки пересечения поверхности с ребрами.
  93. Как генерировать грани и пересечения каких ребер с поверхностью считать,
  94. смотрим по таблице, это зависит лишь от того, какие вершины находятся над
  95. поверхностью, а какие - под. Вершин восемь, состояний два - над и под. Это
  96. дает нам 256 возможных расположений, так что таблица не такая уж и большая.
  97. Индекс в таблице тоже генерируется совсем просто: если вершина находится
  98. над поверхностью, устанавливаем соответствующий этой вершине бит индекса,
  99. иначе - сбрасываем.
  100. <p>Вот простенький пример.
  101. <p><center><img src="illu/illu74a.gif" width=180 height=180 alt="рисунок (illu/illu74a.gif)"></center>
  102. <p>Пусть точка 6 находится под поверхностью, остальные - над. Тогда поверхность
  103. проходит через точки (*), приближаем их линейной интерполяцией и проводим
  104. через них треугольную грань. Какие ребра пересекаются с поверхностью, какие
  105. точки пересечния и как соединять - узнаем из таблиц по индексу; в данном
  106. случае индекс равен 11011111b=0DFh (установлены все биты, кроме 6го).
  107. <p>Алгоритм по шагам:
  108. <ul>
  109. <li><p>посчитать значения функции в узлах сетки (вершинах кубиков)
  110. <li><p>для каждого кубика:
  111. <ul>
  112. <li><p>посчитать индекс этого кубика в таблице. Каждый бит соответствует
  113. какой-то из вершин, если в ней значение функции больше изоуровня,
  114. то надо установить этот бит, иначе - сбросить
  115. <li><p>взять из таблицы по индексу кубика слово, каждый бит в котором
  116. определяет, пересекает ли соответствующее ему (биту) ребро кубика
  117. изоповерхность, или нет. Если да, то посчитать (простая линейная
  118. интерполяция между соответствующими ребру вершинами) координаты
  119. точки пересечения
  120. <li><p>взять из таблицы по индексу кубика число генерируемых граней и
  121. собственно тройки номеров ребер, (уже посчитанные) пересечения
  122. которых с изоповерхностью и будут вершинами нужных граней
  123. </ul></ul>
  124. <p>Неоднократно упоминавшиеся магические таблицы приведены в примерчике, там
  125. же приведен кусок кода, который довольно легко переделть под свой engine.
  126. Пример взят с <a href="http://www.mhri.edu.au/~pdb/modelling/polygonise">http://www.mhri.edu.au/~pdb/modelling/polygonise</a>, сам по себе
  127. он компилироваться НЕ будет, но все там написано правильно и никаких ошибок
  128. в таблицах нет (так что ищите ошибку в своем коде).
  129. </div>
  130. </td></table>
  131. <!-- Bottom Navigation -->
  132. <img src="../img/b.gif" width=500 height=1 alt=""><br><img src="../img/t.gif" width=500 height=2 alt=""><br>
  133. <table width=500 cellpadding=0 cellspacing=0 border=0>
  134. <td><img src="../img/t.gif" width=5 height=1 alt=""><a href="../main.htm" onmouseover="swap('logo2','../img/xe.gif');" onmouseout="swap('logo2','../img/e.gif');"><img src="../img/e.gif" name=logo2 width=60 height=50 hspace=10 border=0 alt=" в самое начало "></a></td>
  135. <td><p class=pagetitle><img src="../img/t.gif" width=265 height=1 alt=""><br>demo.design<br>3D programming FAQ</td>
  136. <td align=center><p class=navy><a href="73.htm" onmouseover="swap('prev','../img/xprev.gif');" onmouseout="swap('prev','../img/prev.gif');"><img src="../img/prev.gif" name=prev width=40 height=40 border=0 hspace=5 alt=" предыдущая статья "></a><br>previous</td>
  137. <td align=center><p class=navy><a href="../content.htm" onmouseover="swap('faq','../img/xfaq.gif');" onmouseout="swap('faq','../img/faq.gif');"><img src="../img/faq.gif" name=faq width=40 height=40 border=0 hspace=5 alt=" содержание "></a><br>content</td>
  138. <td align=center><p class=navy><a href="75.htm" onmouseover="swap('next','../img/xnext.gif');" onmouseout="swap('next','../img/next.gif');"><img src="../img/next.gif" name=next width=40 height=40 border=0 hspace=5 alt=" следующая статья "></a><br>next</td>
  139. </table>
  140. <img src="../img/t.gif" width=500 height=4 alt=""><br>
  141. <img src="../img/b.gif" width=500 height=1 alt=""><br>
  142. <img src="../img/t.gif" width=500 height=1 alt=""><br>
  143. <img src="../img/b.gif" width=500 height=1 alt=""><br>
  144. </center></body>
  145. </html>