
Я хочу создать последовательность итераций ковра Серпинского. Ковер Серпинского создается путем деления квадрата на девять конгруэнтных подквадратов и удаления среднего, затем деления оставшихся подквадратов на девять меньших подквадратов и удаления среднего, затем повторения процесса.
Я создал диаграмму ниже с помощью tikz, рисуя и, где необходимо, заполняя каждый отдельный прямоугольник. Однако это утомительный и неэффективный процесс, особенно потому, что для следующей итерации потребуется еще 512 белых прямоугольников.
\begin{tikzpicture}
\fill (0, 0) rectangle (1, 1);
\begin{scope}[xshift = 1.5 cm]
\fill (0, 0) rectangle (1, 1);
\fill[color=white] (1/3, 1/3) rectangle (2/3, 2/3);
\end{scope}
\begin{scope}[xshift = 3 cm]
\fill (0, 0) rectangle (1, 1);
\fill[color=white] (1/3, 1/3) rectangle (2/3, 2/3);
\fill[color=white] (1/9, 1/9) rectangle (2/9, 2/9);
\fill[color=white] (4/9, 1/9) rectangle (5/9, 2/9);
\fill[color=white] (7/9, 1/9) rectangle (8/9, 2/9);
\fill[color=white] (1/9, 4/9) rectangle (2/9, 5/9);
\fill[color=white] (7/9, 4/9) rectangle (8/9, 5/9);
\fill[color=white] (1/9, 7/9) rectangle (2/9, 8/9);
\fill[color=white] (4/9, 7/9) rectangle (5/9, 8/9);
\fill[color=white] (7/9, 7/9) rectangle (8/9, 8/9);
\end{scope}
\begin{scope}[xshift = 4.5 cm]
\fill (0, 0) rectangle (1, 1);
\fill[color=white] (1/3, 1/3) rectangle (2/3, 2/3);
\fill[color=white] (1/9, 1/9) rectangle (2/9, 2/9);
\fill[color=white] (4/9, 1/9) rectangle (5/9, 2/9);
\fill[color=white] (7/9, 1/9) rectangle (8/9, 2/9);
\fill[color=white] (1/9, 4/9) rectangle (2/9, 5/9);
\fill[color=white] (7/9, 4/9) rectangle (8/9, 5/9);
\fill[color=white] (1/9, 7/9) rectangle (2/9, 8/9);
\fill[color=white] (4/9, 7/9) rectangle (5/9, 8/9);
\fill[color=white] (7/9, 7/9) rectangle (8/9, 8/9);
\fill[color=white] (4/81, 4/81) rectangle (5/81, 5/81);
\fill[color=white] (13/81, 4/81) rectangle (14/81, 5/81);
\fill[color=white] (22/81, 4/81) rectangle (23/81, 5/81);
\fill[color=white] (31/81, 4/81) rectangle (32/81, 5/81);
\fill[color=white] (40/81, 4/81) rectangle (41/81, 5/81);
\fill[color=white] (49/81, 4/81) rectangle (50/81, 5/81);
\fill[color=white] (58/81, 4/81) rectangle (59/81, 5/81);
\fill[color=white] (67/81, 4/81) rectangle (68/81, 5/81);
\fill[color=white] (76/81, 4/81) rectangle (77/81, 5/81);
\fill[color=white] (4/81, 13/81) rectangle (5/81, 14/81);
\fill[color=white] (22/81, 13/81) rectangle (23/81, 14/81);
\fill[color=white] (31/81, 13/81) rectangle (32/81, 14/81);
\fill[color=white] (49/81, 13/81) rectangle (50/81, 14/81);
\fill[color=white] (58/81, 13/81) rectangle (59/81, 14/81);
\fill[color=white] (76/81, 13/81) rectangle (77/81, 14/81);
\fill[color=white] (4/81, 22/81) rectangle (5/81, 23/81);
\fill[color=white] (13/81, 22/81) rectangle (14/81, 23/81);
\fill[color=white] (22/81, 22/81) rectangle (23/81, 23/81);
\fill[color=white] (31/81, 22/81) rectangle (32/81, 23/81);
\fill[color=white] (40/81, 22/81) rectangle (41/81, 23/81);
\fill[color=white] (49/81, 22/81) rectangle (50/81, 23/81);
\fill[color=white] (58/81, 22/81) rectangle (59/81, 23/81);
\fill[color=white] (67/81, 22/81) rectangle (68/81, 23/81);
\fill[color=white] (76/81, 22/81) rectangle (77/81, 23/81);
\fill[color=white] (4/81, 31/81) rectangle (5/81, 32/81);
\fill[color=white] (13/81, 31/81) rectangle (14/81, 32/81);
\fill[color=white] (22/81, 31/81) rectangle (23/81, 32/81);
\fill[color=white] (58/81, 31/81) rectangle (59/81, 32/81);
\fill[color=white] (67/81, 31/81) rectangle (68/81, 32/81);
\fill[color=white] (76/81, 31/81) rectangle (77/81, 32/81);
\fill[color=white] (4/81, 40/81) rectangle (5/81, 41/81);
\fill[color=white] (22/81, 40/81) rectangle (23/81, 41/81);
\fill[color=white] (58/81, 40/81) rectangle (59/81, 41/81);
\fill[color=white] (76/81, 40/81) rectangle (77/81, 41/81);
\fill[color=white] (4/81, 49/81) rectangle (5/81, 50/81);
\fill[color=white] (13/81, 49/81) rectangle (14/81, 50/81);
\fill[color=white] (22/81, 49/81) rectangle (23/81, 50/81);
\fill[color=white] (58/81, 49/81) rectangle (59/81, 50/81);
\fill[color=white] (67/81, 49/81) rectangle (68/81, 50/81);
\fill[color=white] (76/81, 49/81) rectangle (77/81, 50/81);
\fill[color=white] (4/81, 58/81) rectangle (5/81, 59/81);
\fill[color=white] (13/81, 58/81) rectangle (14/81, 59/81);
\fill[color=white] (22/81, 58/81) rectangle (23/81, 59/81);
\fill[color=white] (31/81, 58/81) rectangle (32/81, 59/81);
\fill[color=white] (40/81, 58/81) rectangle (41/81, 59/81);
\fill[color=white] (49/81, 58/81) rectangle (50/81, 59/81);
\fill[color=white] (58/81, 58/81) rectangle (59/81, 59/81);
\fill[color=white] (67/81, 58/81) rectangle (68/81, 59/81);
\fill[color=white] (76/81, 58/81) rectangle (77/81, 59/81);
\fill[color=white] (4/81, 67/81) rectangle (5/81, 68/81);
\fill[color=white] (22/81, 67/81) rectangle (23/81, 68/81);
\fill[color=white] (31/81, 67/81) rectangle (32/81, 68/81);
\fill[color=white] (49/81, 67/81) rectangle (50/81, 68/81);
\fill[color=white] (58/81, 67/81) rectangle (59/81, 68/81);
\fill[color=white] (76/81, 67/81) rectangle (77/81, 68/81);
\fill[color=white] (4/81, 76/81) rectangle (5/81, 77/81);
\fill[color=white] (13/81, 76/81) rectangle (14/81, 77/81);
\fill[color=white] (22/81, 76/81) rectangle (23/81, 77/81);
\fill[color=white] (31/81, 76/81) rectangle (32/81, 77/81);
\fill[color=white] (40/81, 76/81) rectangle (41/81, 77/81);
\fill[color=white] (49/81, 76/81) rectangle (50/81, 77/81);
\fill[color=white] (58/81, 76/81) rectangle (59/81, 77/81);
\fill[color=white] (67/81, 76/81) rectangle (68/81, 77/81);
\fill[color=white] (76/81, 76/81) rectangle (77/81, 77/81);
\end{scope}
\end{tikzpicture}
Я надеялся адаптировать решение Джейка с использованием систем Линденмайера кКак создать треугольник Серпинского в LaTeX?к прямоугольнику, как это сделал марсупилам для шестиугольника вФрактал Tikz - Шестиугольник Серпинского.
Я знаю, что вопросы о создании ковра Серпинского уже задавались. Ответ Марка Виброу наСоздание ковра Серпинского с помощью tikzне использует систему lindenmayer и заставляет мою систему зависать. Ответ Henri Menke наРисование простого фрактала в Tikzкрасиво, но становится совсем маленьким, когда я уменьшаю порядок фрактала и он располагается на вершине, а не на стороне. Я хочу сохранить каждый повтор одного и того же размера, как показано выше.
Я предполагаю, что если я устанавливаю order=\level, я хотел бы установить \squarewidth=9^\level, так как каждый квадрат разделен на девять квадратов, и что я должен установить angle=90, так как каждый угол является прямым. Однако я не понимаю, как адаптировать правила для символов X и Y, которые Джейк и marsupilam использовали в своих ответах для создания узоров, показанных выше.
решение1
Вот макрос, который размещает белые узлы в соответствующих местах.
\sierpinski[options]{levels}
Вариант, который я имею в виду, это scale=
, но rotate=
он тоже подойдет. Или xscale=
если вам нужны прямоугольники, которые не являются квадратами. Обязательно используйте transform shape
.
\documentclass{article}
\usepackage{tikz}
\newcommand{\sierpinski}[2][]{\tikz[#1]{
\draw[fill=black] rectangle(1,1);
\foreach \n[evaluate=\n as \m using \n-1, evaluate=\n as \s using 1/3^\n, evaluate=\m as \p using 3^\m] in {1,...,#2}{
\foreach \k[evaluate=\k as \x using (2*\k-1)/2/3^\m] in {1,...,\p}{
\foreach \j[evaluate=\j as \y using (2*\j-1)/2/3^\m] in {1,...,\p}{
\node[fill=white, minimum size=\s cm, inner sep=0] at (\x,\y){};
}}}}}
\begin{document}
\tikz{\draw[fill=black] rectangle(1,1);}\quad\sierpinski{1}\quad\sierpinski{2}\quad\sierpinski{3}\quad\sierpinski{4}
\end{document}
Вот \sierpinski[scale=3, transform shape]{5}
, это все, что может обработать моя машина. Это O(9ⁿ), так что приготовьтесь ждать.
решение2
с актуальной информациейhttps://ctan.org/pkg/pst-fractalи работает lualatex
:
\documentclass[pstricks]{standalone}
\usepackage{pst-fractal}
\begin{document}
\begin{pspicture}(18,3)
\multido{\iA=1+1,\iB=0+12}{6}{%
\psSierCarpet[scale=0.25,n=\iA](\iB,0.2)}
\end{pspicture}
\end{document}
и с опцией basecolor=red,linecolor=blue
:
решение3
Вы получили ответ tikz, который искали, поэтому я могу свободно добавить ответ, сделанный с помощью MetaPost/MetaFun. Вероятно, код можно оптимизировать, и, вероятно, можно сделать что-то подобное с помощью tikz (но я не могу). Первая версия рисует квадрат, зацикливает и очищает части, которые должны быть белыми. Вторая использует рекурсию.
Что касается сроков:
First version: 2.9s
Second version: 1.8s
Я обернул его в страницы MetaPost в ConTeXt lmtx. Файл можно скомпилировать с помощью context
.
\starttext
\startMPpage[offset=1dk]
vardef sierpinski(expr w, n) =
image(
fill unitsquare scaled w ;
for i = 1 upto n :
for j = 1 upto (3^(i-1)) :
for k = 1 upto (3^(i-1)) :
unfill unitsquare scaled (w/(3^i)) shifted ( (3*j-2)*w/(3^i), (3*k-2)*w/(3^i) ) ;
endfor
endfor
endfor)
enddef ;
for i = 1 upto 3 :
draw sierpinski(3cm,i) shifted (4*(i-1)*cm, 0 ) ;
draw sierpinski(3cm,i+3) shifted (4*(i-1)*cm, -4cm) ;
endfor ;
\stopMPpage
\startMPpage[offset=1dk]
vardef Sierpinski(expr w,n) =
save tmppic ;
picture tmppic ;
if n = 1 :
image(
fill unitsquare scaled w ;
unfill unitsquare scaled (w/3) shifted (w/3,w/3) ;
)
else :
tmppic := Sierpinski(w, n - 1) scaled 1/3 ;
image(
for i = 1 upto 3 :
for j = 1 upto 3 :
if ((i*j) <> 4) :
draw tmppic shifted (((i-1)/3)*w,((j-1)/3)*w) ;
fi
endfor
endfor
)
fi
enddef ;
for i = 1 upto 3 :
draw Sierpinski(3cm,i) shifted (4*(i-1)*cm, 0 ) ;
draw Sierpinski(3cm,i+3) shifted (4*(i-1)*cm, -4cm) ;
endfor ;
\stopMPpage
\stoptext
Вывод в обоих вариантах выглядит одинаково (насколько я могу судить), поэтому я показываю только один из них.
решение4
Вот версия на чистом LaTeX и рекурсии.
Очевидно, это работает только потому, что вы используете прямоугольники (которые очень легко размещаются с помощью tabular
).
Код
\documentclass{article}
\newcommand*\startCarpet[2][1em]{{%
\renewcommand*\arraystretch{0}%
\setlength\fboxsep{0pt}\setlength\fboxrule{#1}%
\edef\BOX{\noexpand\fbox{%
\noexpand\rule[-\the\dimexpr#1/2\relax]{0pt}{#1}\noexpand\rule{#1}{0pt}}}%
\def\TAB##1{\tabular{@{}c@{}c@{}c@{}}##1&##1&##1\\##1&&##1\\##1&##1&##1\endtabular}%
\def\level{#2}%
\def\doCarpet{%
\ifnum\level=0
\def\next{\TAB{\BOX}}%
\else
\def\next{%
\edef\level{\inteval{\level-1}}%
\TAB{\doCarpet}}%
\fi
\next}
\doCarpet}}
\begin{document}
\centering
\startCarpet{0}
\startCarpet[.3333em]{1}
\startCarpet[.1111em]{2}
\startCarpet[.117em]{3}
\end{document}