学习自小甲鱼视频教学(笔记)

功能实现:

在随机位置生成若干个小球以随机速度运动;

若小球运动出左边界则从右边界进入,上下边界同理;

若两小球相碰撞则都以相反速度运动分开。

代码如下:

1.尝试自己写碰撞检测函数(对比两球之间的圆心距离和半径即可)。

 import pygame
import sys
import math
from pygame.locals import *
from random import * # 面向对象的编程方法,定义一个球的类型
class Ball(pygame.sprite.Sprite):
def __init__(self, image, position, speed, bg_size):
# 初始化动画精灵
pygame.sprite.Sprite.__init__(self) self.image = pygame.image.load(image).convert_alpha()
self.rect = self.image.get_rect()
# 将小球放在指定位置
self.rect.left, self.rect.top = position
self.speed = speed
self.width, self.height = bg_size[0], bg_size[1] # 定义一个移动的方法
def move(self):
self.rect = self.rect.move(self.speed)
# 如果小球的左侧出了边界,那么将小球左侧的位置改为右侧的边界
# 这样便实现了从左边进入,右边出来的效果
if self.rect.right < 0:
self.rect.left = self.width
if self.rect.left > self.width:
self.rect.right = 0
if self.rect.bottom < 0:
self.rect.top = self.height
if self.rect.top > self.height:
self.rect.bottom = 0 def collide_check(item, target):
col_balls = []
for each in target:
distance = math.sqrt(
math.pow((item.rect.center[0] - each.rect.center[0]), 2) +
math.pow((item.rect.center[1] - each.rect.center[1]), 2))
if distance <= (item.rect.width + each.rect.width) / 2:
col_balls.append(each) return col_balls def main():
pygame.init() ball_image = 'ball.png'
bg_image = 'background.png'
running = True # 根据背景图片指定游戏界面尺寸
bg_size = width, height = 1024, 500
screen = pygame.display.set_mode(bg_size)
pygame.display.set_caption('Collision Spheres') background = pygame.image.load(bg_image).convert_alpha() # 用来存放小球对象的列表
balls = [] # 创建6个位置随机,速度随机的小球
BALL_NUM = 6
for i in range(BALL_NUM):
position = randint(0, width - 70), randint(0, height - 70)
speed = [randint(1, 6), randint(1, 6)]
ball = Ball(ball_image, position, speed, bg_size)
while collide_check(ball, balls):
ball.rect.left, ball.rect.top = randint(0, width - 70), randint(0, height - 70) balls.append(ball) clock = pygame.time.Clock() while running:
for event in pygame.event.get():
if event.type == QUIT:
sys.exit() screen.blit(background, (0, 0)) for each in balls:
each.move()
screen.blit(each.image, each.rect) for i in range(BALL_NUM):
item = balls.pop(i) if collide_check(item, balls):
item.speed[0] = -item.speed[0]
item.speed[1] = -item.speed[1] balls.insert(i, item) pygame.display.flip()
clock.tick(60) if __name__ == '__main__':
main()

2.使用sprite模块提供的碰撞检测函数。

sprite 模块提供了一个 spritecollide() 函数,用于检测某个精灵是否与指定组中的其他精灵发生碰撞。

spritecollide(sprite, group, dokill, collided = None)

第一个参数指定被检测的精灵

第二个参数指定一个组,由 sprite.Group() 生成

第三个参数设置是否从组中删除检测到碰撞的精灵

第四个参数设置一个回调函数,用于定制特殊的检测方法。如果该参数忽略,那么默认是检测精灵之间的 rect 是否产生重叠。

注意:

  实现圆形的碰撞检测,我们还需要指定 spritecollide() 函数的最后一个参数。

  sprite 模块中正好有一个 collide_circle() 函数用于检测两个圆之间是否发生碰撞...

  这个函数需要你的精灵对象中必须有一个 radius(半径)属性才行。

 import pygame
import sys
from pygame.locals import *
from random import * # 面向对象的编程方法,定义一个球的类型
class Ball(pygame.sprite.Sprite):
def __init__(self, image, position, speed, bg_size):
# 初始化动画精灵
pygame.sprite.Sprite.__init__(self) self.image = pygame.image.load(image).convert_alpha()
self.rect = self.image.get_rect()
# 将小球放在指定位置
self.rect.left, self.rect.top = position
self.speed = speed
self.width, self.height = bg_size[0], bg_size[1]
self.radius = self.rect.width / 2 # 定义一个移动的方法
def move(self):
self.rect = self.rect.move(self.speed)
# 如果小球的左侧出了边界,那么将小球左侧的位置改为右侧的边界
# 这样便实现了从左边进入,右边出来的效果
if self.rect.right < 0:
self.rect.left = self.width
if self.rect.left > self.width:
self.rect.right = 0
if self.rect.bottom < 0:
self.rect.top = self.height
if self.rect.top > self.height:
self.rect.bottom = 0 def main():
pygame.init() ball_image = 'ball.png'
bg_image = 'background.png'
running = True # 根据背景图片指定游戏界面尺寸
bg_size = width, height = 1024, 500
screen = pygame.display.set_mode(bg_size)
pygame.display.set_caption('Collision Spheres') background = pygame.image.load(bg_image).convert_alpha() # 用来存放小球对象的列表
balls = []
group = pygame.sprite.Group() # 创建6个位置随机,速度随机的小球
BALL_NUM = 6
for i in range(BALL_NUM):
position = randint(0, width - 70), randint(0, height - 70)
speed = [randint(1, 6), randint(1, 6)]
ball = Ball(ball_image, position, speed, bg_size)
while pygame.sprite.spritecollide(ball, group, False, pygame.sprite.collide_circle):
ball.rect.left, ball.rect.top = randint(0, width - 70), randint(0, height - 70)
balls.append(ball)
group.add(ball) clock = pygame.time.Clock() while running:
for event in pygame.event.get():
if event.type == QUIT:
sys.exit() screen.blit(background, (0, 0)) for each in balls:
each.move()
screen.blit(each.image, each.rect) for each in group:
group.remove(each) if pygame.sprite.spritecollide(each, group, False, pygame.sprite.collide_circle):
each.speed[0] = -each.speed[0]
each.speed[1] = -each.speed[1] group.add(each) pygame.display.flip()
clock.tick(60) if __name__ == '__main__':
main()

本演示图片素材下载:https://files.cnblogs.com/files/GraceSkyer/20180203.zip

Pygame碰撞检测的更多相关文章

  1. Python游戏编程(Pygame)

    安装Pygame pip install pygame C:\Users> pip install pygame Collecting pygame Downloading https://fi ...

  2. Python-Day07-图形用户界面和游戏开发

    Python-100Day-学习打卡Author: Seven_0507Date: 2019-05-22123 文章目录Python图形用户界面和游戏开发1. tkinter模块2. Pygame进行 ...

  3. pygame 笔记-6 碰撞检测

    这一节学习碰撞检测,先看原理图: 2个矩形如果发生碰撞(即:图形有重叠区域),按上图的判断条件就能检测出来,如果是圆形,则稍微变通一下,用半径检测.如果是其它不规则图形,大多数游戏中,并不要求精确检测 ...

  4. 2015/11/7用Python写游戏,pygame入门(7):碰撞检测

    我们已经完成了飞机大战的大部分东西,但是游戏还是没有办法正式开玩,因为子弹并不能打掉飞机.只有完成了这一个工作,游戏才算基本成型. 今天的内容就非常简单了,就是做到这个碰撞检测,以及控制好子弹和飞机的 ...

  5. 【python游戏编程之旅】第六篇---pygame中的Sprite(精灵)模块和加载动画

    本系列博客介绍以python+pygame库进行小游戏的开发.有写的不对之处还望各位海涵. 直到现在我们已经学了很多pygame基础知识了,从这篇博客开始我们将会学习pygame中高级部分,更多和精灵 ...

  6. pygame编写贪吃蛇

    一直想用pygame做一个小游戏的,可是因为拖延症的缘故一直没有动,结果那天看到了一个12岁的国际友人小盆友用pygame做的一款塔防游戏,突然感觉已经落后超级远了,所以心血来潮做小游戏了.高中陪伴我 ...

  7. python(pygame)滑稽大战(类似飞机大战) 教程

    成品已录制视频投稿B站(本文目前实现了基础的游戏功能),点击观看项目稽忽悠不(github)地址:https://github.com/BigShuang/From-simple-to-Huaji 本 ...

  8. pygame-KidsCanCode系列jumpy-part17-mask-collide碰撞检测

    这节我们研究下pygame的几种碰撞检测模式: 如上图,左侧是默认的检测模式:基于矩形的检测(这也是性能最好的模式), 右侧是基于圆形的检测(性能略差于矩形检测). 矩形检测法虽然性能好,但是缺点也很 ...

  9. pygame-KidsCanCode系列jumpy-part3-重力及碰撞检测

    这个游戏叫jumpy,大致玩法就是模拟超级玛丽一样,可以不停在各个档板上跳动,同时受到重力的作用,会向下掉,如果落下时,没有站在档板上,就挂了. 这节,我们加入重力因素,继续改造sprites.py ...

随机推荐

  1. IDEA tomcat部署

    一.前言     1.CATALINA_HOME和CATALINA_BASE两个变量的区别:前者是tomcat的安装目录,后者是tomcat实例的目录.(安装一个tomcat,可以启动多个tomcat ...

  2. Vue 脱坑记 - 查漏补缺(汇总下群里高频询问的xxx及给出不靠谱的解决方案)

    前言 发现群里有些问题的提问重复率太高了,每次都去回答,回答的贼烦.这里做一个大体的汇总,废话不多说,直接开始给出方案,不是手把手..若是连问题和解决都看不懂的..应该去补充下基础知识 问题汇总 Q: ...

  3. SVN相关命令

    从http://subversion.tigris.org获取subversion for windows的版本,安装之后就有了svn.exe这个基于命令行的客户端工具.当然服务器端的程序也有了,这里 ...

  4. guava文档API制作成chm文件

    将HTML制作成CHM.EXE需要用到一个小工具“HUGECHM”,将HTML打包成CHM文件 1.下载guava的最新的版本,网址:https://github.com/google/guava/w ...

  5. Details.cshtml(118): error CS1001: 应输入标识符

    写了没定义 @Html.DisplayFor(model => model.)

  6. MongoDB的基本使用及java对MongoDB的基本增删改查

    MongoDB的特点 MongoDB 是文档存储数据库,存储结构灵活 MongoDB 支持复杂查询操作.支持序列 MongoDB 采用C++开发,可以做分布式扩展 MongoDB 采用BSON格式存储 ...

  7. Vue 错误:Avoid mutating a prop directly

    Avoid mutating a prop directly since the value will be overwritten whenever the parent component re- ...

  8. js延迟加载优化页面响应速度

    网页打开速度是衡量网站性能的一个极为重要的指标,今天就来说说如何通过JS延迟加载的方式提高页面响应速度: JS延迟加载的 含义:即等页面加载完成之后再加载 JavaScript 文件.作用:JS延迟加 ...

  9. 可持久化trie(BZOJ5338: [TJOI2018]xor)

    题面 BZOJ Sol 显然是要维护一个区域的 \(trie\) 树,然后贪心 区间 \(trie\) 树??? 可持久化 \(trie\) 树??? 直接参考主席树表示出区间的方法建立 \(trie ...

  10. C/C++遍历进程和进程ID的小工具

    原文:http://blog.csdn.net/qq78442761/article/details/54646010 当我们写某些具有破坏性的程序时就需要对进程进行遍历和提取ID 对于上述功能,我们 ...