我们要使用Python编写贪吃蛇游戏,需要使用到pygame模块,即在PyCharm终端输入
pip install pygame
安装完毕即可。
在pygame有三个对象比较重要,分别是
pygame.display—设置场景显示,包括页面大小,页面标题,页面更新(刷新)等;
pygame.time–设置一切与时间相关的设置,游戏的帧频,游戏的持续时间等;
pygame.event–设置与事件相关的处理,比如鼠标点击事件,键盘按下事件等;
pygame.draw–绘制图形到界面上。
要做贪吃蛇游戏,首先第一步就是搭建一个初始界面,具体包括设置界面的size,设置关闭界面事件,设置帧频以及页面背景渲染和更新页面等,具体代码如下所示:
import pygame
# 定义显示窗口的W(宽) H(高)
W = 960
H = 800
size = (960, 600)
pygame.init() # 初始化界面
window = pygame.display.set_mode(size)
pygame.display.set_caption("贪吃蛇大作战")
showWindow = True
clock = pygame.time.Clock() #时钟控制
while showWindow:
# 处理事件
for event in pygame.event.get():
if event.type == pygame.QUIT:
showWindow = False
# 页面渲染
pygame.draw.rect(window, (255, 255, 255), (0, 0, W, H))
pygame.display.flip() #更新整个待显示的Surface对象到屏幕上
clock.tick(30)# 设置帧频
此时运行程序可以显示一个正常的界面,点击右上角关闭可以关闭程序!不仅如此,而且对页面进行了简单的渲染,即改变了界面的背景颜色为白色;
我们可以将整个界面按照行row与列col来进行划分,因此页面中的每一块都对应着一个坐标(row,col),这样我们只需要设置相应的行row与列col,即可绘制出蛇头与食物;
具体代码如下所示:
class Point:#使用Point类来接受row col
def __init__(self, row=0, col=0):
self.row = row
self.col = col
def copy(self):
return Point(self.row, self.col)
ROW = 40 #行数
COL = 30 #列数
def rect(point, color):
cell_width = W/COL
cell_height = H/ROW
left = point.col*cell_width
top = point.row*cell_height
pygame.draw.rect(
window, color,
(left, top, cell_width, cell_height)
)
接下来在游戏循环中绘制蛇头与食物即可,如下所示:
# 定义蛇头
head = Point(row=int(ROW/2), col=int(COL/2))# 在页面中心定义蛇头
head_color = (0, 128, 128) # 用RGB表示颜色
# 定义食物
food = ge_food() #ge_food()是随机产生食物的函数 防止食物与蛇重合
food_color = (255, 255, 0)
# 画蛇头
rect(head, head_color)
# 画食物
rect(food, food_color)
我们需要知道蛇身是随着蛇头移动的,因此,首先我们就要使得蛇头移动起来,而蛇头在界面中使用row与col唯一定位,我们只需要明白其中的移动逻辑即可,具体如下:
left-->head.col-1,row不变
right-->head.col+1,row不变
top-->head.row-1,col不变
down-->head.row+1,col不变
同时需要知道,蛇头在向右移的时候不能再改变直接方向向左移,这样不合理;向上移动的时候不能直接改变方向向下移动;其他方向类似;
虽然我们每次改变的的行列数都是1,但是我们是将上述的改变写在游戏循环中,因此在界面上我们可以看见蛇头不断的移动。
# 处理事件
direct = 'left' #默认移动方向为left
While showWindow:
for event in pygame.event.get():
if event.type == pygame.QUIT:
showWindow = False
if event.type == pygame.KEYDOWN:
if event.key == pygame.K_UP:
if direct=='left' or direct=='right':
direct = "top"
elif event.key == pygame.K_DOWN:
if direct == 'left' or direct == 'right':
direct = "down"
elif event.key == pygame.K_LEFT:
if direct=='top' or direct=='down':
direct = 'left'
elif event.key == pygame.K_RIGHT:
if direct == 'top' or direct == 'down':
direct = 'right'
# 移动蛇头
if direct == 'left':
head.col-=1 # 注意 direct = 'left'与head.col-=1不能写在一起 因为蛇头要一直移动
elif direct == 'right':
head.col+=1
elif direct == 'top':
head.row-=1
else:
head.row+=1
绘制蛇体也很简单,我们只需要将蛇体的坐标存放在一个snakes列表中,当蛇头移动时,我们只需要将蛇体按照蛇头的方向依次移动,并且舍弃列表中的最后一个元素即可。简单来说,我们编写程序时,就要在蛇头移动之前,将蛇头坐标插入到蛇体列表snakes中,并舍弃snakes中的最后一个坐标即可。
snake_color = (128, 128, 128)
snakes = [] #定义蛇身列表
# 将蛇头插入到snakes列表中
snakes.insert(0, head.copy())
# 将最后一个元素删除
snakes.pop()
# 画蛇身
for snake in snakes:
rect(snake, snake_color)
这里我们的贪吃蛇游戏仍然有很多不足,即还要解决蛇撞墙的问题,蛇吃蛇身的问题以及蛇吃到食物就会增长的问题,这里将一一得到解决;
具体代码如下所示:
# 判断蛇是否吃到东西
eat = False
if head.row == food.row and head.col == food.col:# 蛇吃到食物
eat = True
if eat:# 吃到食物就要产生新的食物
food = ge_food() #随机产生食物的函数 返回Point
# 将蛇头插入到snakes列表中
snakes.insert(0, head.copy())
# 将最后一个元素删除
if not eat:
snakes.pop()
# 判断蛇是否死亡
dead = False
# 判断蛇是否撞墙
if head.col<0 or head.row<0 or head.row>=ROW or head.col>=COL:
dead = True
# 判断蛇是否撞蛇身
for snake in snakes:
if snake.row==head.row and snake.col==head.col:
dead = True
break
if dead:
showWindow = False
这里可以做的更加精细,这里我设置的是一旦蛇撞墙与蛇咬蛇身就直接退出界面。

黄色的是随机产生的食物,而深色的是蛇头,蛇头吃到食物就吃长出蛇身。由于时间比较紧张,所以只是完成了基本的功能,后期会继续完善,包括多种游戏模式的设置,无尽模式,闯关模式等,还有背景图片背景音乐的加入,甚至可以用自定义的图片来设置蛇头的显示等等。
import pygame
import random
class Point:
def __init__(self, row=0, col=0):
self.row = row
self.col = col
def copy(self):
return Point(self.row, self.col)
# 定义显示窗口的W(宽) H(高)
W = 800
H = 600
snakes = [] #定义蛇身列表
def ge_food():
while True:
pos = Point(random.randint(0, ROW - 1), random.randint(0, COL - 1))
is_collide = False
if pos.row == head.row and pos.col == head.col: # 与蛇头重合
is_collide = True
# 与蛇身碰撞
for snake in snakes:
if (snake.row == pos.row and snake.col == pos.col):
is_collide = True
break
if not is_collide:
return pos
ROW = 40 #行数
COL = 30 #列数
size = (W, H)
pygame.init() # 初始化界面
window = pygame.display.set_mode(size)
pygame.display.set_caption("贪吃蛇大作战")
bak_color = (255, 255, 255)
# 定义蛇头
head = Point(row=int(ROW/2), col=int(COL/2))
head_color = (0, 128, 128)
# 定义食物
food = ge_food()
food_color = (255, 255, 0)
snake_color = (128, 128, 128)
direct = 'left'
def rect(point, color):
cell_width = W/COL
cell_height = H/ROW
left = point.col*cell_width
top = point.row*cell_height
pygame.draw.rect(
window, color,
(left, top, cell_width, cell_height)
)
showWindow = True
clock = pygame.time.Clock() #时钟控制
while showWindow:
# 处理事件
for event in pygame.event.get():
if event.type == pygame.QUIT:
showWindow = False
if event.type == pygame.KEYDOWN:
if event.key == pygame.K_UP:
if direct=='left' or direct=='right':
direct = "top"
elif event.key == pygame.K_DOWN:
if direct == 'left' or direct == 'right':
direct = "down"
elif event.key == pygame.K_LEFT:
if direct=='top' or direct=='down':
direct = 'left'
elif event.key == pygame.K_RIGHT:
if direct == 'top' or direct == 'down':
direct = 'right'
# 判断蛇是否吃到东西
eat = False
if head.row == food.row and head.col == food.col:# 蛇吃到食物
eat = True
if eat:# 吃到食物就要产生新的食物
food = ge_food()
# 将蛇头插入到snakes列表中
snakes.insert(0, head.copy())
# 将最后一个元素删除
if not eat:
snakes.pop()
# 移动蛇头
if direct == 'left':
head.col-=1 # 注意 direct = 'left'与head.col-=1不能写在一起 因为蛇头要一直移动
elif direct == 'right':
head.col+=1
elif direct == 'top':
head.row-=1
else:
head.row+=1
# 判断蛇是否死亡
dead = False
# 判断蛇是否撞墙
if head.col<0 or head.row<0 or head.row>=ROW or head.col>=COL:
dead = True
# 判断蛇是否撞蛇身
for snake in snakes:
if snake.row==head.row and snake.col==head.col:
dead = True
break
if dead:
showWindow = False
# 页面渲染
pygame.draw.rect(window, bak_color, (0, 0, W, H))
# 这里需要注意 绘制食物与蛇头要在绘制背景之后 因为黑色的背景颜色会覆盖一切
# 画蛇头
rect(head, head_color)
# 画蛇身
for snake in snakes:
rect(snake, snake_color)
# 画食物
rect(food, food_color)
pygame.display.flip() #更新整个待显示的Surface对象到屏幕上
clock.tick(15)# 设置帧频 可以用来控制蛇头移动的速度
我正在学习如何使用Nokogiri,根据这段代码我遇到了一些问题:require'rubygems'require'mechanize'post_agent=WWW::Mechanize.newpost_page=post_agent.get('http://www.vbulletin.org/forum/showthread.php?t=230708')puts"\nabsolutepathwithtbodygivesnil"putspost_page.parser.xpath('/html/body/div/div/div/div/div/table/tbody/tr/td/div
我有一个Ruby程序,它使用rubyzip压缩XML文件的目录树。gem。我的问题是文件开始变得很重,我想提高压缩级别,因为压缩时间不是问题。我在rubyzipdocumentation中找不到一种为创建的ZIP文件指定压缩级别的方法。有人知道如何更改此设置吗?是否有另一个允许指定压缩级别的Ruby库? 最佳答案 这是我通过查看rubyzip内部创建的代码。level=Zlib::BEST_COMPRESSIONZip::ZipOutputStream.open(zip_file)do|zip|Dir.glob("**/*")d
类classAprivatedeffooputs:fooendpublicdefbarputs:barendprivatedefzimputs:zimendprotecteddefdibputs:dibendendA的实例a=A.new测试a.foorescueputs:faila.barrescueputs:faila.zimrescueputs:faila.dibrescueputs:faila.gazrescueputs:fail测试输出failbarfailfailfail.发送测试[:foo,:bar,:zim,:dib,:gaz].each{|m|a.send(m)resc
很好奇,就使用rubyonrails自动化单元测试而言,你们正在做什么?您是否创建了一个脚本来在cron中运行rake作业并将结果邮寄给您?git中的预提交Hook?只是手动调用?我完全理解测试,但想知道在错误发生之前捕获错误的最佳实践是什么。让我们理所当然地认为测试本身是完美无缺的,并且可以正常工作。下一步是什么以确保他们在正确的时间将可能有害的结果传达给您? 最佳答案 不确定您到底想听什么,但是有几个级别的自动代码库控制:在处理某项功能时,您可以使用类似autotest的内容获得关于哪些有效,哪些无效的即时反馈。要确保您的提
假设我做了一个模块如下:m=Module.newdoclassCendend三个问题:除了对m的引用之外,还有什么方法可以访问C和m中的其他内容?我可以在创建匿名模块后为其命名吗(就像我输入“module...”一样)?如何在使用完匿名模块后将其删除,使其定义的常量不再存在? 最佳答案 三个答案:是的,使用ObjectSpace.此代码使c引用你的类(class)C不引用m:c=nilObjectSpace.each_object{|obj|c=objif(Class===objandobj.name=~/::C$/)}当然这取决于
我正在尝试使用ruby和Savon来使用网络服务。测试服务为http://www.webservicex.net/WS/WSDetails.aspx?WSID=9&CATID=2require'rubygems'require'savon'client=Savon::Client.new"http://www.webservicex.net/stockquote.asmx?WSDL"client.get_quotedo|soap|soap.body={:symbol=>"AAPL"}end返回SOAP异常。检查soap信封,在我看来soap请求没有正确的命名空间。任何人都可以建议我
关闭。这个问题是opinion-based.它目前不接受答案。想要改进这个问题?更新问题,以便editingthispost可以用事实和引用来回答它.关闭4年前。Improvethisquestion我想在固定时间创建一系列低音和高音调的哔哔声。例如:在150毫秒时发出高音调的蜂鸣声在151毫秒时发出低音调的蜂鸣声200毫秒时发出低音调的蜂鸣声250毫秒的高音调蜂鸣声有没有办法在Ruby或Python中做到这一点?我真的不在乎输出编码是什么(.wav、.mp3、.ogg等等),但我确实想创建一个输出文件。
我在我的项目目录中完成了compasscreate.和compassinitrails。几个问题:我已将我的.sass文件放在public/stylesheets中。这是放置它们的正确位置吗?当我运行compasswatch时,它不会自动编译这些.sass文件。我必须手动指定文件:compasswatchpublic/stylesheets/myfile.sass等。如何让它自动运行?文件ie.css、print.css和screen.css已放在stylesheets/compiled。如何在编译后不让它们重新出现的情况下删除它们?我自己编译的.sass文件编译成compiled/t
我想将html转换为纯文本。不过,我不想只删除标签,我想智能地保留尽可能多的格式。为插入换行符标签,检测段落并格式化它们等。输入非常简单,通常是格式良好的html(不是整个文档,只是一堆内容,通常没有anchor或图像)。我可以将几个正则表达式放在一起,让我达到80%,但我认为可能有一些现有的解决方案更智能。 最佳答案 首先,不要尝试为此使用正则表达式。很有可能你会想出一个脆弱/脆弱的解决方案,它会随着HTML的变化而崩溃,或者很难管理和维护。您可以使用Nokogiri快速解析HTML并提取文本:require'nokogiri'h
我想为Heroku构建一个Rails3应用程序。他们使用Postgres作为他们的数据库,所以我通过MacPorts安装了postgres9.0。现在我需要一个postgresgem并且共识是出于性能原因你想要pggem。但是我对我得到的错误感到非常困惑当我尝试在rvm下通过geminstall安装pg时。我已经非常明确地指定了所有postgres目录的位置可以找到但仍然无法完成安装:$envARCHFLAGS='-archx86_64'geminstallpg--\--with-pg-config=/opt/local/var/db/postgresql90/defaultdb/po