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

功能实现:

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

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

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

代码如下:

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. LINQ操作List<T>

    LINQ操作List<T>主要包括: 1.筛选 List<string> stcdList = stcdArray.ToList<string>() .FindAl ...

  2. 使用http维持socket长连接

    项目中有遇到问题如下: 1.旧版的cs服务,因为每个用户和唯一的长连接是在登录后绑定的,并且所有的消息报文均是基于该长连接去发送接收的,所以要求node服务要维持一个长连接,然后根据该用户获取长连接, ...

  3. vscode下eslint代码规范

    直接上规范吧: // 将设置放入此文件中以覆盖默认设置 { "editor.fontSize": 17, "editor.tabSize": 2, " ...

  4. 三、spark简介

    一.简介 spark的官网:http://spark.apache.org/ spark解决了什么问题? 我们都知道hadoop,hadoop以一个非常容易使用的编程模型解决了大数据的两大难题: 1) ...

  5. 撩课-Java每天5道面试题第26天

    161.简述一下springMVC当中的视图解析器 请求处理方法执行完成后,最终返回一个 ModelAndView 对象 对于那些返回 String,View 或 ModeMap 等类型的处理方法 S ...

  6. Java集成groovy之GroovyShell、GroovyScriptEngine、GroovyClassLoader

    GroovyClassLoader 用 Groovy 的 GroovyClassLoader ,动态地加载一个脚本并执行它的行为.GroovyClassLoader是一个定制的类装载器,负责解释加载J ...

  7. Java枚举、静态导入、自动拆装箱、增强for循环、可变参数

    一.枚举简介 1.什么是枚举? 需要在一定范围内取值,这个值只能是这个范围内中的任意一个 现实场景:交通信号灯,有三种颜色,但是每次只能亮三种颜色里面的任意一个 2.使用一个关键字 enum enum ...

  8. java 简单计算器

    package com.direct.demo; import java.text.DecimalFormat; import java.util.Scanner; public class Calc ...

  9. Mysql分布式部署高可用集群方案

    HAproxy+Mycat +MySQL主从集群高可用方案 1.         HAproxy高可用方案: haproxy+keepalived,利用keepalived的VIP浮动能力,(多台ha ...

  10. UOJ42. 【清华集训2014】Sum

    传送门 Sol \((-1)^a=1-2(a~mod~2)=1-2a+4\lfloor\frac{a}{2}\rfloor\) 那么原式变成 \(n-2\sum_{i=1}^{n}\lfloor d\ ...