Python生命游戏实现原理及过程解析(附源代码)


Posted in Python onAugust 01, 2019

1. 生命游戏是什么

生命游戏是英国数学家约翰·何顿·康威在1970年发明的细胞自动机。它包括一个二维矩形世界,这个世界中的每个方格居住着一个活着的或死了的细胞。一个细胞在下一个时刻生死取决于相邻八个方格中活着的或死了的细胞的数量。如果相邻方格活着的细胞数量过多,这个细胞会因为资源匮乏而在下一个时刻死去;相反,如果周围活细胞过少,这个细胞会因太孤单而死去。

规则看起来很简单,但却能演绎出无穷无尽的内容。

Python生命游戏实现原理及过程解析(附源代码)

滑翔者:每4个回合"它"会向右下角走一格。虽然细胞早就是不同的细胞了,但它能保持原本的形态。

Python生命游戏实现原理及过程解析(附源代码)

轻量级飞船:它的周期是4,每2个回合会向右边走一格。

Python生命游戏实现原理及过程解析(附源代码)

脉冲星:它的周期为3,看起来像一颗周期爆发的星星。

Python生命游戏实现原理及过程解析(附源代码)

更复杂的图案。

来体会一下这些作品的脑洞以及震撼:

史诗般的生命游戏 http://www.iqiyi.com/w_19rsq435c9.html

用生命游戏实现生命游戏:http://www.bilibili.com/video/av616329/index.html

2. Console版:一个简单的Python实现

生命游戏的规则其实很简单。我们可以把计算机中的宇宙想象成是一堆方格子构成的封闭空间,尺寸为N的空间就有N*N个格子。

而每一个格子都可以看成是一个生命体,每个生命都有生和死两种状态,如果该格子生就显示蓝色,死则显示白色。每一个格子旁边都有邻居格子存在,如果我们把3*3的9个格子构成的正方形看成一个基本单位的话,那么这个正方形中心的格子的邻居就是它旁边的8个格子。

每个格子的生死遵循下面的原则:

1. 如果一个细胞周围有3个细胞为生(一个细胞周围共有8个细胞),则该细胞为生(即该细胞若原先为死,则转为生,若原先为生,则保持不变) 。

2. 如果一个细胞周围有2个细胞为生,则该细胞的生死状态保持不变;

3. 在其它情况下,该细胞为死(即该细胞若原先为生,则转为死,若原先为死,则保持不变)

设定图像中每个像素的初始状态后依据上述的游戏规则演绎生命的变化,由于初始状态和迭代次数不同,将会得到令人叹服的优美图案。

我们用#代表活的细胞,空格表示死的细胞,那么我们可以用控制台打印字符、清屏来模拟生命游戏。我的代码在github上:

https://github.com/Pleiades0428/GameOfLife/blob/master/Demo/gameOfLife.py

游戏世界尺寸为60x20,随机生成初始状态,循环边界,按任意键进入下一帧,q退出。

单纯的看这段程序,好像并没有什么问题,代码逻辑正确、清晰。

效果图:

Python生命游戏实现原理及过程解析(附源代码)

3. Python列表生成式

我们来尝试一些python的高级特性,比如列表生成式。

例如,在生成初始值时,我们一般这样写:

screen = []
width = 60
height = 20
def Init():
 for i in range(height):
  line = []
  for j in range(width):
   if random.random() > 0.8:
    line.append('#')
   else:
    line.append(' ')
  screen.append(line)

如果用列表生成式,我们可以这样写:

def Init():
  global screen
  screen = [['#' if random.random() > 0.8 else ' ' for i in range(width)] for j in range(height)]

注意这里必须用global声明,否则screen将默认作为函数内的局部变量。这里用了两层列表生成式来生成一个二维数组。

列表生成式很好很强大,如果用好能大大提高效率。但会牺牲一定的可读性,如果单个表达式写的过于复杂,那就变成write-only了。尤其是在团队开发情况下,可读性日益重要。

重写后的代码:

https://github.com/Pleiades0428/GameOfLife/blob/master/Demo/gameOfLife.1.py

如果仅仅是作为练习,这样就已经足够好了,简洁易读。

4. 重构:面向对象与重用

可是我们还不能满足,我们来给生命插上面向对象的翅膀,在模块化的天空中翱翔。对,就是让他跟别的模块搞对象!

先来定义一个类GameOfLifeWorld,之前那些丑陋的全局变量,让他们统统变成成员变量,再也不能在外兴风作浪。

class GameOfLifeWorld:

 width = 100
 height = 100
 cells = []
…略

然后把UI层剥离,只保留游戏的核心逻辑。

代码:

https://github.com/Pleiades0428/GameOfLife/blob/master/Demo/gameOfLifeWorld.py

5. GUI:Tkinter的调用

有了上一步的铺垫,我们终于可以让Tkinter粉墨登场了。Tkinter是著名的UI库,Python自带的Tkinter是一个精简版,不过也够我们用的了。

我们这里用到的主要是Canvas,Button控件。Canvas画布用来绘制游戏区,Button用来交互。

代码:

https://github.com/Pleiades0428/GameOfLife/blob/master/Demo/gameOfLifeWorld.py

效果:

Python生命游戏实现原理及过程解析(附源代码)

以上就是这样,项目我还会继续改进,希望大家喜欢。

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持三水点靠木。

Python 相关文章推荐
python的re模块应用实例
Sep 26 Python
Python 2与Python 3版本和编码的对比
Feb 14 Python
TensorFlow实现Logistic回归
Sep 07 Python
Python中几种属性访问的区别与用法详解
Oct 10 Python
详解django+django-celery+celery的整合实战
Mar 19 Python
Python turtle绘画象棋棋盘
Aug 21 Python
PyCharm专业最新版2019.1安装步骤(含激活码)
Oct 09 Python
Python爬虫库BeautifulSoup获取对象(标签)名,属性,内容,注释
Jan 25 Python
Django 删除upload_to文件的步骤
Mar 30 Python
Django配置Bootstrap, js实现过程详解
Oct 13 Python
解决PyCharm无法使用lxml库的问题(图解)
Dec 22 Python
Python 一键获取电脑浏览器的账号密码
May 11 Python
Python中list的交、并、差集获取方法示例
Aug 01 #Python
详解pandas使用drop_duplicates去除DataFrame重复项参数
Aug 01 #Python
Python使用Pandas对csv文件进行数据处理的方法
Aug 01 #Python
python使用writerows写csv文件产生多余空行的处理方法
Aug 01 #Python
python psutil模块使用方法解析
Aug 01 #Python
python读写csv文件并增加行列的实例代码
Aug 01 #Python
Flask框架重定向,错误显示,Responses响应及Sessions会话操作示例
Aug 01 #Python
You might like
详解Yii2 定制表单输入字段的标签和样式
2017/01/04 PHP
redis+php实现微博(三)微博列表功能详解
2019/09/23 PHP
thinkphp5.1框架模板赋值与变量输出示例
2020/05/25 PHP
javascript full screen 全屏显示页面元素的方法
2013/09/27 Javascript
全面理解面向对象的 JavaScript(来自ibm)
2013/11/10 Javascript
JavaScript运行时库属性一览表
2014/03/14 Javascript
jQuery遍历DOM节点操作之filter()方法详解
2016/04/14 Javascript
javascript判断图片是否加载完成的方法推荐
2016/05/13 Javascript
input框中的name和id的区别
2016/11/16 Javascript
Spring boot 和Vue开发中CORS跨域问题解决
2018/09/05 Javascript
layer弹出子iframe层父子页面传值的实现方法
2018/11/22 Javascript
vue路由插件之vue-route
2019/06/13 Javascript
详解javascript void(0)
2020/07/13 Javascript
[57:09]DOTA2-DPC中国联赛 正赛 Phoenix vs Dynasty BO3 第一场 1月26日
2021/03/11 DOTA
Python UnicodeEncodeError: 'gbk' codec can't encode character 解决方法
2015/04/24 Python
Python参数解析模块sys、getopt、argparse使用与对比分析
2019/04/02 Python
Python利用requests模块下载图片实例代码
2019/08/12 Python
Python Web框架之Django框架文件上传功能详解
2019/08/16 Python
Python中list循环遍历删除数据的正确方法
2019/09/02 Python
python itsdangerous模块的具体使用方法
2020/02/17 Python
Python绘制全球疫情变化地图的实例代码
2020/04/20 Python
Python+kivy BoxLayout布局示例代码详解
2020/12/28 Python
Canvas在超级玛丽游戏中的应用详解
2021/02/06 HTML / CSS
土耳其玩具商店:Toyzz Shop
2019/08/02 全球购物
关于赌博的检讨书
2014/01/24 职场文书
销售员个人求职的自我评价
2014/02/10 职场文书
企业负责人任命书
2014/06/05 职场文书
工作所在部门证明
2014/09/21 职场文书
综合管理员岗位职责
2015/02/11 职场文书
2019让人心动的商业计划书
2019/06/27 职场文书
《家世》读后感:看家训的力量
2019/12/30 职场文书
Nginx反向代理多个服务器的实现方法
2021/03/31 Servers
golang 实现菜单树的生成方式
2021/04/28 Golang
Pandas数据类型之category的用法
2021/06/28 Python
Python与C++中梯度方向直方图的实现
2022/03/17 Python
CSS 实现角标效果的完整代码
2022/06/28 HTML / CSS