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

功能实现:

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

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

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

代码如下:

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. DDD学习笔记(一)

    最近开始筹备一个电商项目. 其实是公司的老本行了. 但今年公司希望在做项目的同时, 沉淀出一套针对电商的基础产品. 这样可以提高新项目的开发效率, 减少重复劳动. 那现如今, DDD(领域驱动设计)应 ...

  2. Python——第一个python程序helloworld

    安装了Python的环境之后,就是编写Python的代码了. 首先,我们来写一个简单的“hello world” 新建一个空白的txt文本,将后缀改为.py 改了后缀之后即变为Python程序的图标 ...

  3. JS实现最小生成树之普里姆(Prim)算法

    最小生成树: 我们把构造连通网的最小代价生成树称为最小生成树.经典的算法有两种,普利姆算法和克鲁斯卡尔算法. 普里姆算法打印最小生成树: 先选择一个点,把该顶点的边加入数组,再按照权值最小的原则选边, ...

  4. 常见IT英语单词

    lable标签,master精通.主人,reference参考,release发布,schema模式,component组件,persistence持久化,generate生成产生,plugin插件, ...

  5. 将Windows上的Oracle迁移至Linux

    迁移前提: 1.在安装Linux数据库实例时,注意选择的编码格式要与Windows的数据库实例一致. 迁移步骤 1.检查Linux上数据库实例的编译格式 SQL> select userenv( ...

  6. python中静态方法(@staticmethod)和类方法(@classmethod)的区别

    一般来说,要使用某个类的方法,需要先实例化一个对象再调用方法. 而使用@staticmethod或@classmethod,就可以不需要实例化,直接类名.方法名()来调用. 这有利于组织代码,把某些应 ...

  7. js生成qq客服在线代码

    说到QQ客服在线代码,随便那么百度谷歌一下就会出来,一般都是 <a target="blank" href="http://wpa.qq.com/msgrd?V=1 ...

  8. BZOJ1266 [AHOI2006]上学路线

    Description 可可和卡卡家住合肥市的东郊,每天上学他们都要转车多次才能到达市区西端的学校.直到有一天他们两人参加了学校的信息学奥林匹克竞赛小组才发现每天上学的乘车路线不一定是最优的. 可可: ...

  9. 出售基于Html5的高性能GIS平台源码

    远景地理信息系统平台软件 远景地理信息系统平台软件,简称"RemoteGIS",版本V1.0,它是基于HTML5自主研发的新一代WEBGIS平台,它使用Javascript开发,具 ...

  10. C++学习笔记(8)----C++类的大小

    C++类的大小 (i) 如下代码: #include<iostream> using namespace std; class CBase { }; class CDerive :publ ...