0 |& y6 b+ c# ]
6.2 绘制分形* S1 K3 w9 b) w7 |: `7 Q
分形势复杂的几何图案或形状,但却由简单的数学公式所生成。与圆形和矩形等几何图形相比,分形看上去并不规则,也没有明显的模式或描述。但如果仔细观察,你将发现模式的存在,即整个图形有无数个自身的副本构成。由于分形设计平面上的点的相同几何变换的重复引用,因此非常适合使用计算机程序来构建。这一章中,我们将学习如何绘制Barnsley蕨类植物、Sierpinski三角和Manderlbrot集合(后两个出现在编程挑战中),这些都是分形研究领域的常见例子。分形在自然界中比比皆是,常见的例子包括海岸线、树木和雪花。 + G) u% g+ t$ o" Z1 O' c, m5 R
6.2.1 平面上点的变化
/ ^; z8 Y; L8 ?8 C 创建分形的基本思想是点的变换。给定x-y平面上的一个点P(x,y),变幻的一个例子是P(x,y)->Q(x+1, y+1),这意味着应用变换后,创建好了一个新点Q,它位于P向上和向右各一个单位处。如果以Q点作为起始点,将得到另一个点R,它位于点Q向上和向右各一个单位处。假设起始点P位于(1,1),下图展示了这些点的位置。 + j% I7 b( X' E0 d' f
0 G9 r6 Z' ~& x6 p1 Y5 ~. y+ m P、Q、R三个点的位置图
* w- M, u2 j/ P9 o/ G* p+ }) I 8 ]$ k5 n A5 x, X+ d* ~' \8 p
因此,变换是一种描述点在x-y平面上移动的规则,从起始位置开始,每一次迭代使得点移动到新的位置。我们可以把变换想象成点在平面上的轨迹。现在换个思路考虑两个变换规则,每次迭代将从这两个规则中随机选择一个。两个规则如下: ' l9 b5 o1 o7 J% o. a) P1 c3 ?
规则1:P1(x,y) -> P2(x+1, y-1)
: {. ^5 D' i( s0 I7 o 规则2:P1(x,y) -> P2(x+1, y+1)
7 E8 ?; H: b J1 \! _- N 以P(1, 1)作为起始点,如果我们执行4次迭代,将得到以下一系列的点:
- t F9 @, T9 q A7 R P1(1, 1) -> P2(2, 0) (规则1) & g6 u1 Y; f& t8 X% y! [- {6 g" f
P2(2, 0) -> P3(3, 1) (规则2) 7 \# i' E: S* ~
P3(3, 1) -> P4(4, 2) (规则2)
P4 G, Z1 t# g# u6 o P4(4, 2) -> P5(5, 1) (规则2)
8 r* F r7 V' J3 P/ r; l3 i5 N ...
2 K% q' L: m7 C4 P! t 两个变换规则是随机选择的,其中每个规则被选择的概率相同。无论选择哪一个规则,点都会右移,因为在两种规则中都会增加x坐标。当点向右移动时,他们会向上或向下移动,从而形成一条Z字路径。以下程序绘制了在指定的迭代次数内进行上述转换规则之一时点的路径: : O- C% e2 `1 f8 V. _, L
Example of selecting a transformation from two equally probable: N- {4 E$ ^% \" f0 ]/ h1 H3 u: ]
transformations4 ` F r; G. s3 i" ~. R! d3 j' C
9 v; T! W+ v% R8 D7 J import matplotlib.pyplot as plt8 b: S, E- \ X9 t# ^% ]7 G
import random+ T1 h. s0 H% O/ d e0 }
, c4 b1 H8 k% k
def transformation_1(p):
9 t' U& S) a! e* g, l9 D x = p[0]+ u' Q5 J4 ]8 r
y = p[1]
* v9 ^1 V& `; @! b9 Y0 P return x + 1, y - 1
; W* O8 j! T; e& D def transformation_2(p):
) |3 c( B* S! I* e! Z; ?# { x = p[0]" g3 z! G! o. E1 t
y = p[1]
4 ~# C/ T6 k$ ?, J6 P( X6 d return x + 1, y + 1
* P) S8 o4 w3 i def transform(p):6 j7 }+ X6 p6 y+ W: J0 t+ Z
# List of transformation functions
" s/ l1 a& s9 I8 o$ R8 o# v transformations = [transformation_1, transformation_2]
( D; |6 }( |( {% `; s; b7 s" a # pick a random transformation function and call it
) I! `* o' b% W. E t = random.choice(transformations)
& t# \( Z" J1 L7 F x, y = t(p)
3 L' E1 t+ e; W# A, a$ s( H; }$ V* ~ return x, y! a) G6 w4 v2 `: l
def build_trajectory(p, n):$ V8 g2 s/ E7 \% g8 b
x = [p[0]]* @8 X: B3 C. x# E
y = [p[1]]. ]; q$ _" V& _; q3 s& X
for i in range(n):
% V$ m1 U$ ~0 j p = transform(p)
$ W/ o% {; ?: x: S+ |+ c3 } x.append(p[0]): H( q1 Y+ R8 v) ~9 b9 F4 ]" A
y.append(p[1])' O$ s5 ~3 m2 {/ S. Y$ |, P
return x, y0 t) O) X) B$ v3 q& \( A
# t" m+ l1 J! K: {7 |) T; ]: S if __name__ == __main__:. f/ y6 k; q+ w8 W5 P" Q5 b4 Z& q
# Initial point
& D e) S N- E! Y* T/ F p = (1,1)2 ^ B0 i" U- L+ C# D& D! T
n = int(input(Enter the number of iterations: ))
: t' v! v8 q" @& [5 K3 ]6 R x , y = build_trajectory(p, n)
, }5 v9 D+ T2 b4 t! B0 F3 } # PLot+ I& `& Z- p: |5 X2 ` E/ ^# z
plt.plot(x, y)
$ d& q5 h0 k' V! X& Q; C5 a6 t plt.xlabel(X)
5 o8 t% v6 s; \( W0 U5 M% U B plt.ylabel(Y)
# o1 U l# g* _3 Z2 x plt.show()我们定义了两个函数transformation_1()和transformation_2(),分别对应于之前的两种变换规则,在transform()函数中,我们创建了一个包含这两个函数名字的列表,然后使用random.choice()函数从该列表中选择去哦中一个进行变换。现在假设我们选择了其中一个变换,蒋点P作为输入参数调用该变换函数,然后分别使用x,y来存储变换后的坐标并且返回它们。
5 ]% |0 b& y- f3 P) H R& J2 i 从列表中随机选择一个元素 % F% B' h* Q3 V( }$ X3 L' Z9 x3 S3 N
我们在第一个分形程序中看到的random.choice()函数可以用来从列表中随机选择一个元素,每一个元素被选中的概率相同,下面是一个例子: import random% D! T3 v3 P1 v! b) k7 s
>>> l = [1,2,3]
; Z1 P: V6 O* Q6 l/ U >>> random.choice(l). `- Y$ w8 D$ x- y% A
14 r" a! Z7 S0 R& s* ?1 q; W
>>> random.choice(l)
% Z% M5 J7 l. P 3
# h1 r6 t4 D- {" i5 y1 N >>> random.choice(l)
& _5 Z' k1 I; I+ w. [3 Y+ I# \ 3
, u+ X7 p: p6 E3 q) i: H >>> random.choice(l)
& h8 \/ \8 q* W6 a6 ?$ p( z& m/ x 1( d( D1 p8 t2 [$ W) C
>>> random.choice(l)) S/ ?0 M3 c, x) a
15 }/ q2 v9 e/ W) d0 F! \6 F
这个函数也适用于元组和字符串。在后一种情形中,函数从字符串中随即返回一个字符。
: J8 v0 G' h+ L6 y0 U( Q 当运行程序,他首先会询问迭代次数n,即应用变化的次数。然后调用build_trajectory()函数,该函数的输入参数为迭代次数n和起始点坐标p,此处设置初始点坐标为(1,1)。build_trajectory()函数重复调用n次transform()函数,并使用两个列表x和y分别存储变换后的x和y坐标。最后使用这两个列表来绘制图形。
) D4 j7 u/ L! _ 下面两张图分别显示了100次和10000次迭代的点的轨迹图,这两幅图中的Z字形运动都很明显,这种Z字形路径通常被称为一条线上的随机游走。 " g7 ^ U: Q: {8 o" r# i. @( e( d
, R. Y2 \) I; d 迭代100次的Z字路径
9 }+ A1 q/ B) S! I
3 t/ Z" j; Z, b* Z9 N0 b 6 Z6 a3 @) d3 n& t
迭代10000次的Z字路径 2 f/ d6 `* N+ c* x( {, w/ B
4 w. K8 a" ^1 \; E/ G- r& s 这个例子展示了创建分形的基本思想,即从一个点开始并对其重复应用某种变换。接下来,我们将看到一个用相同想法绘制Barnsley蕨类植物的例子。 & }: H$ Y6 |9 K; Q% d4 Q
6.2.2 绘制Barnsley蕨类植物
$ j* L- f) O7 `, F. }3 o 英国数学家Michael Barnsley描述了如何对一个点进行重复的简单变换,从而创建蕨类植物的结构(如下图)。 # L; d& ^* v! X6 a
( m6 |- J. F X; M. I8 o# _ 蕨类植物
1 o% c. v! O/ j) O/ e3 d! ]9 n $ ?$ n) Y; }4 \
他提出了以下步骤来创建类似于蕨类植物的结构,以(0,0)为初始点,按事先分配的概率随机选择下述某种变换。 - R7 i! U$ z# }4 v: _. Z$ b
(1)变换1(概率为0.85): - g+ T" t* ?# M
 1 x8 F, @) K( Y2 X/ C
(2)变换2(概率为0.07): - e7 e) d) T7 A, l2 V7 r
 3 w# C( R9 G. D& } L
(3)变换3(概率为0.07): ) H' O; S% C B( w3 w) x( S
 5 U0 U# E1 b8 R
(4)变换4(概率为0.01): : H% V9 }3 d* u' [/ q- V" K9 u1 X
 ( ?4 u8 l- s) n* q9 I& M" l. | P7 ~
上述每一个变换对应于蕨类植物的一部分。第一个变换被选中的概率最大,因此被执行的次数最多,从而产生了蕨类植物的茎和底部的叶子。第二个和第三个变换分别对应于左边和右边底部的叶子,第四个变换绘制了蕨类植物的茎。
) i8 F8 Z0 T( i( C4 H 这是一个非均匀概率选择的例子,我们在第五章中已经学习过,以下程序为指定的点数绘制Barnsley蕨类植物。 0 H- J6 S& P+ q: L. U
Draw a Barnsley Fern/ h" S" N/ n# e: j" R# B
3 X2 T# s4 N" F8 h' F2 }0 q( B7 l4 p
import random4 V' m3 }4 P! R- v- Q2 l. ]- `
import matplotlib.pyplot as plt9 h4 |% f/ u- y
def transformation_1(p):
% K* ~) @# v" U3 T R7 N* Z x = p[0]
9 n+ h' L8 v$ [0 n% P y = p[1]
. `0 w0 m6 _( J2 w9 j: |2 n) S; U x1 = 0.85*x + 0.04*y1 X+ N" _7 O, p2 }
y1 = -0.04*y + 0.85*y + 1.6& x4 s! R, T! H
return x1, y1# z- g" U8 y# C) f- g$ R! K
def transformation_2(p):
, r2 ~; e5 ]; [" b+ N0 x6 d, m x = p[0]& ^& H, b. _- i1 K
y = p[1]
! ], i) }+ B1 ^ x1 = 0.2*x - 0.26*y# }4 j+ J, B$ S) A* y% ?
y1 = 0.23*y + 0.22*y + 1.6( l4 @/ o5 P3 G8 Z# @( b
return x1, y1, c% }0 u7 Z" c% Z+ B% S) l( n) N
def transformation_3(p):
' L- Q4 i8 ]# d! y/ T- D# a. ~ x = p[0]3 e6 o' ~) f& v5 L0 m
y = p[1]
' ^3 ?7 M9 ^7 x( j y' ]; \- D x1 = -0.15*x + 0.28*y
3 T) B1 L+ O8 r1 a5 Z, \" o Q7 v y1 = 0.26*y + 0.24*y + 0.44
5 M" f9 _: u C- Y3 z9 d+ A; Z return x1, y1
* S5 W3 ^3 N: w7 R( ^5 R def transformation_4(p):, L( j% s) G) D& X$ P
x = p[0]- L2 C' i5 B& `% C( b+ M
y = p[1]
! ?) b1 z' y" K; w* W8 A5 f! D x1 = 0
& Y8 T% t" d q/ I3 R y1 = 0.16*y% E j4 {5 B- b$ v" t3 I. r
return x1, y1
* A9 j8 P, |, }# N def get_index(probability):
% k, F7 k+ A$ K r = random.random()! m0 a R( p. ?- W0 t
c_probability = 0! A. }9 D* c3 `- s: D' f
sum_probability = []
4 Q$ W) {/ D: T) ~5 S' B$ o% Y; t for p in probability:
- e* j0 D4 G+ q0 ^$ j2 H c_probability += p) A& P* l0 b; E: a2 \8 q# _1 z8 x
sum_probability.append(c_probability)0 k6 O7 a4 K- @) ?- o
for item, sp in enumerate(sum_probability):
$ v+ S8 M' f. }0 @* l) c if r < sp:" z$ i- w" N$ L
return item& x( A/ C, b) t2 y9 e- u' t
return len(probability) - 1
2 U9 r2 y$ O8 Y% l: R, E7 x) z- F5 a- z# u: g# A" \: E, ~ k* @2 j
def transform(p):' }5 d+ W: i3 g' l
# List of transformation functions
) o! r+ i) ?+ `; |6 O transformation = [transformation_1, transformation_2, transformation_3, transformation_4]+ C' H3 O& l: f* h+ H6 O
probability = [0.85, 0.07, 0.07, 0.01]$ M f! @4 I- Q' W1 [& _
# Pick a random transformation function and call it2 @& \% }$ @ Q$ W
tindex = get_index(probability)
8 u8 r/ |4 h) d* ` t = transformation[tindex]
3 g# @! n K9 G' V x, y = t(p)% _- k% o# S# s. ]2 j
return x, y& h" i0 }1 Y% V) E1 Q! f6 |
def draw_fern(n):
! b2 Q- L# B$ P1 e" r* @0 ~ # We start with (0,0)# E" M/ s7 R$ q; d! m' I. m# P
x = [0]3 p* g: B4 |5 v
y = [0]
( c$ D" M( D" c* z' O x1, y1 = 0, 0
$ @! W& z+ j: a& f% u* x/ ~& Z+ A' L for i in range(n):
% v/ S3 S* m' T. b x1, y1 = transform((x1, y1))
# F* p8 E& A' N x.append(x1)
) S2 q3 Z6 N3 D: l y.append(y1)
1 f& ], x0 @2 o6 R return x, y& X* q& p7 g1 y" a% p
if __name__ == __main__:1 P) Y# U4 m4 e$ U% G8 C! k
n = int(input(Enter the number of points in the Ferns: ))8 c! h$ ^) y2 H- G9 J
x, y = draw_fern(n)
, A* }8 u; V9 E3 U8 }, K7 ?- ] plt.plot(x, y, o): t: x, r; l7 N% X# X8 t! {, B. s
plt.title(Fern with {0} points.format(n))* c" M/ W# c, L' g
plt.show()
: K& G: m* Z6 j$ S 运行此程序,他首先询问蕨类植物图中指定的点数,然后开始绘制该图。下面两张图分别展示了具有1000个点和10000个点的蕨类植物图。 ) T; ]& f$ Q' j! c+ A! e
. Y- ~! u4 G4 a# S
制定了1000个点的蕨类植物叶片
# I2 c( j% U. h/ E: j5 ]- J
& F+ h3 s+ O9 s1 _* x 7 q4 r& V. v# f
指定了10000个点的蕨类植物叶片 ' G7 u' }! Y/ _5 O6 Y! O9 u; _; z# g
X5 ]- C* J# ?+ r7 s( W
这4个变换规则分别在transformation_1(),transformation_2(),transformation_3(),transformation_4()函数中定义。get_index()函数返回按照给定概率下的一种随机变化的索引值,这个概率我们之前已经讲过了,不清楚的可以返回去在第五章非均匀概率那里再次学习。返回索引之后即可选择某一个具体的变换来应用。 - b7 c! p. M) T/ }
初始点(0,0)变换的次数即为程序输入中指定的蕨类植物点的个数。 * ?. W% }0 O! M5 s$ ^6 [( e' R
+ I! k9 @9 }6 A* e/ V
, A, ^5 z- g3 A- ^8 {$ A5 S5 x! h8 F& u# t; `
5 A) b! L" G4 d0 b |