Python趣味挑战之turtle库绘画飘落的银杏树
作者:yunyun云芸 发布时间:2023-07-21 21:46:40
标签:Python,turtle库,银杏树
一、导入所需的库
import turtle
import random
from math import *
二、生成斐波那契数列
斐波那契数列是指前两项的和加起来等于后一项的一个数列,这里使用了两个函数来生成斐波契那数列。
def Fibonacci_Recursion_tool(n): #斐波那契数列方法
if n <= 0:
return 0
elif n == 1:
return 1
else:
return Fibonacci_Recursion_tool(n - 1) + Fibonacci_Recursion_tool(n - 2)
def Fibonacci_Recursion(n): #生成斐波那契数列,并存入列表
result_list = []
for i in range(1, n + 3):
result_list.append(Fibonacci_Recursion_tool(i))
return result_list
调用函数生成一个数列如下:
yu = Fibonacci_Recursion(top) #生成斐波契那数列
print(yu)
运行结果如下:
三、定义生成叶子的方法
def leaf(x, y, node):#定义画叶子的方法
til = turtle.heading()
i = random.random()
an = random.randint(10, 180)
ye = random.randint(6, 9)/10
turtle.color(ye, ye*0.9, 0)
turtle.fillcolor(ye+0.1, ye+0.05, 0)
turtle.pensize(1)
turtle.pendown()
turtle.setheading(an + 90)
turtle.forward(8*i)
px = turtle.xcor()
py = turtle.ycor()
turtle.begin_fill()
turtle.circle(7.5*i, 120) # 画一段120度的弧线
turtle.penup() # 抬起笔来
turtle.goto(px, py) # 回到圆点位置
turtle.setheading(an + 90) # 向上画
turtle.pendown() # 落笔,开始画
turtle.circle(-7.5*i, 120) # 画一段120度的弧线
turtle.setheading(an + 100)
turtle.circle(10.5*i, 150)
turtle.end_fill() # 画一段150度的弧线
turtle.penup()
turtle.goto(x, y)
turtle.setheading(til)
turtle.pensize(node / 2 + 1)
四、定义生成树的方法
这里用x生成随机数,用if条件进行判断来决定要不要继续画分支,要不要画叶子,使树更加自然,无规律性,更好看一点,这样会导致你每次运行时,画出来的树都是不一样的。具体的细节,我已经加上了注释。如果想调整空中叶子的比例,树的分叉程度,修改if判断语句中的x取值范围,以增加概率或减小概率即可。至于如何达到你心中完美的效果就要慢慢去尝试了。
def draw(node, length, level, yu, button): #定义画树的方法
turtle.pendown()
t = cos(radians(turtle.heading()+5)) / 8 + 0.25
turtle.pencolor(t*1.6, t*1.2, t*1.4) #(r, g, b)颜色对应的RGB值
turtle.pensize(node/1.2) #画笔的尺寸
x = random.randint(0, 10) #生成随机数决定要画树枝还是画飘落的叶子
if level == top and x > 6: #此时画飘落的叶子,x范围太大会导致树太秃
turtle.forward(length) # 画树枝
yu[level] = yu[level] - 1
c = random.randint(2, 10)
for i in range(1, c):
leaf(turtle.xcor(), turtle.ycor(), node)
# 添加0.3倍的飘落叶子
if random.random() > 0.3:
turtle.penup()
# 飘落
t1 = turtle.heading()
an1 = -40 + random.random() * 40
turtle.setheading(an1)
dis = int(800 * random.random() * 0.5 + 400 * random.random() * 0.3 + 200 * random.random() * 0.2)
turtle.forward(dis)
turtle.setheading(t1)
turtle.right(90)
# 画叶子
leaf(turtle.xcor(), turtle.ycor(), node)
turtle.left(90)
# 返回
t2 = turtle.heading()
turtle.setheading(an1)
turtle.backward(dis)
turtle.setheading(t2)
elif level==top and x < 7 : #此时画枝叶,x范围太大会导致飘落的叶子太少
turtle.penup()
turtle.forward(length)
elif level>3 and (x>6) :# * 树枝以上,有40%的概率执行以下策略
turtle.pendown()
turtle.forward(length)
c = random.randint(4, 6)
for i in range(3, c):
leaf(turtle.xcor(), turtle.ycor(),node)
leaf(turtle.xcor(), turtle.ycor(),node)
button=1# jump"""
else:
turtle.forward(length) # 画树枝
yu[level] = yu[level] -1
if node > 0 and button == 0:
# 计算右侧分支偏转角度,在固定角度偏转增加一个随机的偏移量
right = random.random() * 5 + 17
# 计算左侧分支偏转角度,在固定角度偏转增加一个随机的偏移量
left = random.random() * 20 + 19
# 计算下一级分支的长度
child_length = length * (random.random() * 0.25 + 0.7)
# 右转一定角度,画右分支
r=random.randint(0, 1)
if r==1:
turtle.right(right)
level = level + 1
#print("level", level)
else:
turtle.left(right)
level = level + 1
#print("level", level)
draw(node - 1, child_length,level,yu,button)
yu[level] = yu[level] +1
if yu[level] > 1:
# 左转一定角度,画左分支
if r==1:
turtle.left(right + left)
draw(node - 1, child_length, level, yu,button)
# 将偏转的角度,转回
turtle.right(left)
yu[level] = yu[level] - 1
else:
turtle.right(right + left)
draw(node - 1, child_length, level, yu,button)
# 将偏转的角度,转回
turtle.left(left)
yu[level] = yu[level] - 1
else:
if r==1:
turtle.left(right + left)
turtle.right(left)
else:
turtle.right(right + left)
turtle.left(left)
turtle.penup()
#退回到上一级节点顶部位置
turtle.backward(length)
5.主函数部分
主函数中直接调用上述函数就行,top控制树的高度,turtle.speed控制画的速度,最后的turtle.write()用来书写最下方的签名。
```clike
if __name__ == '__main__':
turtle.setup(width=1.0, height=1.0) #设置全屏显示
turtle.hideturtle() # 隐藏turtle
turtle.speed(0) # 设置画笔移动的速度,0-10 值越小速度越快
# turtle.tracer(0,0) #设置动画的开关和延迟,均为0
turtle.penup() # 抬起画笔
turtle.left(90) # 默认方向为朝x轴的正方向,左转90度则朝上
turtle.backward(300) # 设置turtle的位置,朝下移动300
top = 9 #树高
yu = Fibonacci_Recursion(top) #生成斐波契那数列
yu.remove(yu[0])
#print(yu) 打印斐波那契数列
button = 0
draw(top, 120, 0, yu, button) # 调用函数开始绘制
turtle.write(" wsw", font=("微软雅黑", 14, "normal")) #生成签名
turtle.done()
运行程序后,“海龟”会帮你画出整棵树,你只需要看着它画就行,需要等待一定的时间,最后的一种成品如下,是想要的一半叶子在空中的感觉了,哈哈哈哈~
来源:https://blog.csdn.net/yunyun889901/article/details/117366517
0
投稿
猜你喜欢
- (1)int转strings := strconv.Itoa(i)等价于s := strconv.FormatInt(int64(i), 1
- 应该是开心网(kaixin.com)的宠物功能又升级了,这几次发来的邮件内容不仅不能让我开心,反而让我觉得很恶心。开心网注册也一段时间了,之
- 闭包的概念我们尝试从概念上去理解一下闭包。在一些语言中,在函数中可以(嵌套)定义另一个函数时,如果内部函数引用了外部函数的变量,则可能产生闭
- 一、前言在Python提供了强大的模块支持,主要体现为不仅在Python标准库中包含了大量的模块(称为标准模块),而且还有很多第三方模块,另
- 最近学了一点点python爬虫的知识,面向百度编程爬了一本小说之后感觉有点不满足,于是突发奇想尝试爬一本漫画下来看看。一、效果展示首先是我们
- 本文实例讲述了php查找指定目录下指定大小文件的方法。分享给大家供大家参考。具体实现方法如下:php查找文件大小的原理是遍历目录然后再利用f
- 今天下午主要做了个实验,是针对 测试表的列,进行添加,修改,删除的。做法如下: 增加一列: alter table emp4 add tes
- 数据类型是所有开发语言的基础,JavaScript虽然是一个弱类型的脚本语言,但是在数据类型上也有很多讲究的,看了淘宝UED玉伯的一篇文章,
- 我们经常会遇到多重查询问题,而长长的SQL语句往往让人丈二和尚摸不着头脑。特别是客户端部分填入查询条件时,如用普通方法将更是难上加难。以下巧
- 近日无事,想起以前曾打算过要做一个定时重启或关机的工具,便花了一点时间以hta的形式写了个,名为"Windows Timer&qu
- 学习前言……又看了很久的SSD算法,今天讲解一下训练部分的代码。预测部分的代码可以参照https
- asp如何获知页面上的图象的实际尺寸大小?见下面的两个asp文件:<!--#include virtual="/i
- 所需库的安装很多人问Pytorch要怎么可视化,于是决定搞一篇。tensorboardX==2.0tensorflow==1.13.2由于t
- 一、get//get请求function getUrl($url, $header = []){ $ch = cu
- 简介ORA-10458: standby database requires recoveryORA-01196: 文件 1 由于介质恢复会
- 原理:建一个栈,每次碰到一个新标签,就与栈顶的标签配对,如果配对,栈顶的标签就出栈,如果不配对,这个新标签就进栈,最终,栈如果是空的,说明所
- 好多网友问起来,·深度学习网址导航·深度学习整站系统 的后台管理能否增加批量删除功能,如何加:就是列出N篇文章或网址信息,每篇文章或网址前有
- PyQt5 事件处理机制PyQt为事件处理提供了两种机制:高级的信号与槽机制,以及低级的事件处理机制。信号与槽可以说是对事件处理机制的高级封
- 目录前言线程安全锁的作用Lock() 同步锁基本介绍使用方式死锁现象with语句RLock() 递归锁基本介绍使用方式with语句Condi
- ImageField的使用笔记今天完善作业写的订单系统,主要是给每一个菜品增加图片,看起来美观一些,但是没想到这个小小的需求花了我一天时间,