python与分形0018 - 【教程】奥林匹克烯

知乎上有一个提问,怎么画奥林匹克烯,具体长下面这样子。

f2222b4b53f1307c6fc93e1a3eb82587.png

看到这个图的第一印象,就跟苯环差不多,六边形加上几条线就好了。

STEP 1:画一个普通六边形 Hexagon

这个代码比较简单,最简单的是:

for i in range(6):
    turtle.fd(100)
    turtle.right(60)
db873b2116cd12899ae45431dbe8b346.gif

STEP 2:画一个正放六边形Vertical Hexagon

我们从左下角的位置开始画,步骤:

  • 起笔到起始位置

  • 设置方向为正北

  • 开始画fd

  • 设置方向向右60°

  • 重复如上5次

代码:

turtle.penup()
turtle.goto((-100, 0))
turtle.seth(90)
turtle.pendown()
for i in range(6):
    turtle.fd(100)
    turtle.right(60)
ccb486d00602c89b612fa59bb1344880.gif

STEP 3:重构一把

重构原因:

  • 把画图封装到一个函数中

  • 5个六边形实际上是有关系的,我们需要把6个顶点记下来

重构后的代码:

def draw_vertical_hexagon(start_pos, length):
    all_pos = []
    turtle.penup()
    turtle.goto(start_pos)
    turtle.seth(90)
    turtle.pendown()
    for i in range(6):
        all_pos.append(turtle.pos())
        turtle.fd(length)
        turtle.right(60)
    return all_pos

STEP 4:增加苯环内部的线段

这些线段有如下特性:

  • 与外环的顶点连线与外环的夹角是60°

  • 内环是一个小的正六边形

基于如上两点,我们只要确定内环的起始位置线段长度就可以按照外环的画法把内环画出来。

数学推导见下图:

0aed1a366c7b1797b04b04743501ff88.png

图上的标注的夹角为60°,假设外环和内环的长度差为2d,

如果外环的起点坐标为(x, y)的话,

那么内环的起点坐标为(x+sqrt(3)*d, y+d)。

好了,已知外环的起始坐标和长度,直接求出了内环的起始坐标和长度。

可以直接上代码(这里的d不太符合我们的定义,但是也可以用,我懒得调了):

def draw_vertical_hexagon_with_ring(start_pos, length, length_ratio, ring):
    d = length//length_ratio
    all_pos = []
    turtle.penup()
    turtle.goto(start_pos)
    turtle.seth(90)
    turtle.pendown()
    for i in range(6):
        all_pos.append(turtle.pos())
        turtle.fd(length)
        turtle.right(60)

    turtle.penup()
    turtle.goto(start_pos[0]+math.sqrt(3)*d, start_pos[1]+d)
    turtle.seth(90)
    for i in range(6):
        if ring[i]:
            turtle.pendown()
        turtle.fd(length-d*2)
        turtle.right(60)
        turtle.penup()
    return all_pos

这个代码里面,我们巧妙的用了一个ring的参数,表示对应的外环是否有对应的内环。

  • 如果有对应的内环,那么我们就pendown。

  • 如果没有对应的内环,那么我们就penup。

  • 就这么简单。

4fe1c090eeac8cc51b67a925cfe82805.gif

STEP 5

万事俱备,只欠五环。

观察整个图形,我们发现每个环的起始绘图点是有规律的。

  • 第二个环的第一个点(起始点)是第一个环的第五个点。

  • 第三个环的第一个点(起始点)是第二个环的第五个点。

  • 第四个环的第一个点(起始点)与第一个环的第六个点差一个外环的边长。

  • 第五个环的第一个点(起始点)是第四个环的第五个点。

代码来了:

length = 100
length_ratio = 10
p0 = (-100,0)
all_pos0 = draw_vertical_hexagon_with_ring(p0, length, length_ratio, [1, 0, 1, 0, 1, 0])

p1 = all_pos0[4]
all_pos1 = draw_vertical_hexagon_with_ring(p1, length, length_ratio, [0, 0, 0, 0, 0, 0])

p2 = all_pos1[4]
all_pos2 = draw_vertical_hexagon_with_ring(p2, length, length_ratio, [1, 0, 1, 0, 1, 0])

p3 = (all_pos0[5][0], all_pos0[5][1]-length) 
all_pos3 = draw_vertical_hexagon_with_ring(p3, length, length_ratio, [0, 0, 0, 1, 0, 1])

p4 = all_pos3[4]
all_pos4 = draw_vertical_hexagon_with_ring(p4, length, length_ratio, [0, 0, 0, 0, 1, 0])
0d1927f1e7436582f60007827926072a.gif

视频

72603ad69fe0c2f2430371ee19b4a1a9.gif e3940adb28927cc9141cb17b26772320.png 86dc53821eefac706a62b885ffde0c5c.gif
  • 1
    点赞
  • 12
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值