亚洲免费在线-亚洲免费在线播放-亚洲免费在线观看-亚洲免费在线观看视频-亚洲免费在线看-亚洲免费在线视频

Python實現(xiàn)打磚塊小游戲代碼實例

系統(tǒng) 1595 0

這次用Python實現(xiàn)的是一個接球打磚塊的小游戲,需要導(dǎo)入pygame模塊,有以下兩條經(jīng)驗總結(jié):

1.多父類的繼承2.碰撞檢測的數(shù)學(xué)模型

知識點稍后再說,我們先看看游戲的效果和實現(xiàn):

一、游戲效果

二、游戲代碼

            
#導(dǎo)入模塊
import pygame
from pygame.locals import *
import sys,random,time,math

class GameWindow(object):
	'''創(chuàng)建游戲窗口類'''
	def __init__(self,*args,**kw):		
		self.window_length = 600
		self.window_wide = 500
		#繪制游戲窗口,設(shè)置窗口尺寸
		self.game_window = pygame.display.set_mode((self.window_length,self.window_wide))
		#設(shè)置游戲窗口標題
		pygame.display.set_caption("CatchBallGame")
		#定義游戲窗口背景顏色參數(shù)
		self.window_color = (135,206,250)

	def backgroud(self):
		#繪制游戲窗口背景顏色
		self.game_window.fill(self.window_color)

class Ball(object):
	'''創(chuàng)建球類'''
	def __init__(self,*args,**kw):
		#設(shè)置球的半徑、顏色、移動速度參數(shù)
		self.ball_color = (255,215,0)		
		self.move_x = 1
		self.move_y = 1
		self.radius = 10

	def ballready(self):
		#設(shè)置球的初始位置、
		self.ball_x = self.mouse_x
		self.ball_y = self.window_wide-self.rect_wide-self.radius
		#繪制球,設(shè)置反彈觸發(fā)條件			
		pygame.draw.circle(self.game_window,self.ball_color,(self.ball_x,self.ball_y),self.radius)

	def ballmove(self):
		#繪制球,設(shè)置反彈觸發(fā)條件			
		pygame.draw.circle(self.game_window,self.ball_color,(self.ball_x,self.ball_y),self.radius)		
		self.ball_x += self.move_x
		self.ball_y -= self.move_y
		#調(diào)用碰撞檢測函數(shù)
		self.ball_window()
		self.ball_rect()
		#每接5次球球速增加一倍
		if self.distance < self.radius:
			self.frequency += 1
			if self.frequency == 5:
				self.frequency = 0
				self.move_x += self.move_x
				self.move_y += self.move_y
				self.point += self.point
		#設(shè)置游戲失敗條件
		if self.ball_y > 520:
			self.gameover = self.over_font.render("Game Over",False,(0,0,0))
			self.game_window.blit(self.gameover,(100,130))
			self.over_sign = 1

class Rect(object):
	'''創(chuàng)建球拍類'''
	def __init__(self,*args,**kw):
		#設(shè)置球拍顏色參數(shù)
		self.rect_color = (255,0,0)
		self.rect_length = 100
		self.rect_wide = 10

	def rectmove(self):
		#獲取鼠標位置參數(shù)
		self.mouse_x,self.mouse_y = pygame.mouse.get_pos()
		#繪制球拍,限定橫向邊界					
		if self.mouse_x >= self.window_length-self.rect_length//2:
			self.mouse_x = self.window_length-self.rect_length//2
		if self.mouse_x <= self.rect_length//2:
			self.mouse_x = self.rect_length//2
		pygame.draw.rect(self.game_window,self.rect_color,((self.mouse_x-self.rect_length//2),(self.window_wide-self.rect_wide),self.rect_length,self.rect_wide))

class Brick(object):
	def __init__(self,*args,**kw):
		#設(shè)置磚塊顏色參數(shù)
		self.brick_color = (139,126,102)
		self.brick_list = [[1,1,1,1,1,1],[1,1,1,1,1,1],[1,1,1,1,1,1],[1,1,1,1,1,1],[1,1,1,1,1,1]]
		self.brick_length = 80
		self.brick_wide = 20

	def brickarrange(self):		
		for i in range(5):
			for j in range(6):
				self.brick_x = j*(self.brick_length+24)
				self.brick_y = i*(self.brick_wide+20)+40
				if self.brick_list[i][j] == 1:
					#繪制磚塊
					pygame.draw.rect(self.game_window,self.brick_color,(self.brick_x,self.brick_y,self.brick_length,self.brick_wide))					
					#調(diào)用碰撞檢測函數(shù)
					self.ball_brick()										
					if self.distanceb < self.radius:
						self.brick_list[i][j] = 0
						self.score += self.point
		#設(shè)置游戲勝利條件
		if self.brick_list == [[0,0,0,0,0,0],[0,0,0,0,0,0],[0,0,0,0,0,0],[0,0,0,0,0,0],[0,0,0,0,0,0]]:
			self.win = self.win_font.render("You Win",False,(0,0,0))
			self.game_window.blit(self.win,(100,130))
			self.win_sign = 1

class Score(object):
	'''創(chuàng)建分數(shù)類'''
	def __init__(self,*args,**kw):		
		#設(shè)置初始分數(shù)
		self.score = 0
		#設(shè)置分數(shù)字體
		self.score_font = pygame.font.SysFont('arial',20)
		#設(shè)置初始加分點數(shù)
		self.point = 1
		#設(shè)置初始接球次數(shù)
		self.frequency = 0

	def countscore(self):
		#繪制玩家分數(shù)			
		my_score = self.score_font.render(str(self.score),False,(255,255,255))
		self.game_window.blit(my_score,(555,15))

class GameOver(object):
	'''創(chuàng)建游戲結(jié)束類'''
	def __init__(self,*args,**kw):
		#設(shè)置Game Over字體
		self.over_font = pygame.font.SysFont('arial',80)
		#定義GameOver標識
		self.over_sign = 0

class Win(object):
	'''創(chuàng)建游戲勝利類'''
	def __init__(self,*args,**kw):
		#設(shè)置You Win字體
		self.win_font = pygame.font.SysFont('arial',80)
		#定義Win標識
		self.win_sign = 0

class Collision(object):
	'''碰撞檢測類'''
	#球與窗口邊框的碰撞檢測
	def ball_window(self):
		if self.ball_x <= self.radius or self.ball_x >= (self.window_length-self.radius):
			self.move_x = -self.move_x
		if self.ball_y <= self.radius:
			self.move_y = -self.move_y

	#球與球拍的碰撞檢測
	def ball_rect(self):
		#定義碰撞標識
		self.collision_sign_x = 0
		self.collision_sign_y = 0

		if self.ball_x < (self.mouse_x-self.rect_length//2):
			self.closestpoint_x = self.mouse_x-self.rect_length//2
			self.collision_sign_x = 1
		elif self.ball_x > (self.mouse_x+self.rect_length//2):
			self.closestpoint_x = self.mouse_x+self.rect_length//2
			self.collision_sign_x = 2
		else:
			self.closestpoint_x = self.ball_x
			self.collision_sign_x = 3

		if self.ball_y < (self.window_wide-self.rect_wide):
			self.closestpoint_y = (self.window_wide-self.rect_wide)
			self.collision_sign_y = 1
		elif self.ball_y > self.window_wide:
			self.closestpoint_y = self.window_wide
			self.collision_sign_y = 2
		else:
			self.closestpoint_y = self.ball_y
			self.collision_sign_y = 3
		#定義球拍到圓心最近點與圓心的距離
		self.distance = math.sqrt(math.pow(self.closestpoint_x-self.ball_x,2)+math.pow(self.closestpoint_y-self.ball_y,2))
		#球在球拍上左、上中、上右3種情況的碰撞檢測
		if self.distance < self.radius and self.collision_sign_y == 1 and (self.collision_sign_x == 1 or self.collision_sign_x == 2):
			if self.collision_sign_x == 1 and self.move_x > 0:
				self.move_x = - self.move_x
				self.move_y = - self.move_y
			if self.collision_sign_x == 1 and self.move_x < 0:
				self.move_y = - self.move_y
			if self.collision_sign_x == 2 and self.move_x < 0:
				self.move_x = - self.move_x
				self.move_y = - self.move_y
			if self.collision_sign_x == 2 and self.move_x > 0:
				self.move_y = - self.move_y
		if self.distance < self.radius and self.collision_sign_y == 1 and self.collision_sign_x == 3:
			self.move_y = - self.move_y
		#球在球拍左、右兩側(cè)中間的碰撞檢測
		if self.distance < self.radius and self.collision_sign_y == 3:
			self.move_x = - self.move_x

	#球與磚塊的碰撞檢測
	def ball_brick(self):
		#定義碰撞標識
		self.collision_sign_bx = 0
		self.collision_sign_by = 0

		if self.ball_x < self.brick_x:
			self.closestpoint_bx = self.brick_x
			self.collision_sign_bx = 1
		elif self.ball_x > self.brick_x+self.brick_length:
			self.closestpoint_bx = self.brick_x+self.brick_length
			self.collision_sign_bx = 2
		else:
			self.closestpoint_bx = self.ball_x
			self.collision_sign_bx = 3

		if self.ball_y < self.brick_y:
			self.closestpoint_by = self.brick_y
			self.collision_sign_by = 1
		elif self.ball_y > self.brick_y+self.brick_wide:
			self.closestpoint_by = self.brick_y+self.brick_wide
			self.collision_sign_by = 2
		else:
			self.closestpoint_by = self.ball_y
			self.collision_sign_by = 3
		#定義磚塊到圓心最近點與圓心的距離
		self.distanceb = math.sqrt(math.pow(self.closestpoint_bx-self.ball_x,2)+math.pow(self.closestpoint_by-self.ball_y,2))
		#球在磚塊上左、上中、上右3種情況的碰撞檢測
		if self.distanceb < self.radius and self.collision_sign_by == 1 and (self.collision_sign_bx == 1 or self.collision_sign_bx == 2):
			if self.collision_sign_bx == 1 and self.move_x > 0:
				self.move_x = - self.move_x
				self.move_y = - self.move_y
			if self.collision_sign_bx == 1 and self.move_x < 0:
				self.move_y = - self.move_y
			if self.collision_sign_bx == 2 and self.move_x < 0:
				self.move_x = - self.move_x
				self.move_y = - self.move_y
			if self.collision_sign_bx == 2 and self.move_x > 0:
				self.move_y = - self.move_y
		if self.distanceb < self.radius and self.collision_sign_by == 1 and self.collision_sign_bx == 3:
			self.move_y = - self.move_y
		#球在磚塊下左、下中、下右3種情況的碰撞檢測
		if self.distanceb < self.radius and self.collision_sign_by == 2 and (self.collision_sign_bx == 1 or self.collision_sign_bx == 2):
			if self.collision_sign_bx == 1 and self.move_x > 0:
				self.move_x = - self.move_x
				self.move_y = - self.move_y
			if self.collision_sign_bx == 1 and self.move_x < 0:
				self.move_y = - self.move_y
			if self.collision_sign_bx == 2 and self.move_x < 0:
				self.move_x = - self.move_x
				self.move_y = - self.move_y
			if self.collision_sign_bx == 2 and self.move_x > 0:
				self.move_y = - self.move_y
		if self.distanceb < self.radius and self.collision_sign_by == 2 and self.collision_sign_bx == 3:
			self.move_y = - self.move_y
		#球在磚塊左、右兩側(cè)中間的碰撞檢測
		if self.distanceb < self.radius and self.collision_sign_by == 3:
			self.move_x = - self.move_x

class Main(GameWindow,Rect,Ball,Brick,Collision,Score,Win,GameOver):
	'''創(chuàng)建主程序類'''
	def __init__(self,*args,**kw):		
		super(Main,self).__init__(*args,**kw)
		super(GameWindow,self).__init__(*args,**kw)
		super(Rect,self).__init__(*args,**kw)
		super(Ball,self).__init__(*args,**kw)
		super(Brick,self).__init__(*args,**kw)
		super(Collision,self).__init__(*args,**kw)		
		super(Score,self).__init__(*args,**kw)
		super(Win,self).__init__(*args,**kw)
		#定義游戲開始標識
		start_sign = 0

		while True:			
			self.backgroud()
			self.rectmove()
			self.countscore()			
			
			if self.over_sign == 1 or self.win_sign == 1:
				break
			#獲取游戲窗口狀態(tài)
			for event in pygame.event.get():
				if event.type == pygame.QUIT:
					sys.exit()
				if event.type == MOUSEBUTTONDOWN:
					pressed_array = pygame.mouse.get_pressed()
					if pressed_array[0]:
						start_sign = 1
			if start_sign == 0:
				self.ballready()
			else:
				self.ballmove()

			self.brickarrange()

			#更新游戲窗口
			pygame.display.update()
			#控制游戲窗口刷新頻率
			time.sleep(0.010)

if __name__ == '__main__':
	pygame.init()
	pygame.font.init()
	catchball = Main()

          

三、知識點1.多父類的繼承

Python的繼承方式分為深度優(yōu)先和廣度優(yōu)先,Python2分經(jīng)典類的深度優(yōu)先搜索繼承方式(class A:)、 新式類的廣度優(yōu)先搜索繼承方式(class A(object):)2種,Python3經(jīng)典類與新式類的繼承方式與python2的新式類繼承方式一致,都為廣度優(yōu)先的繼承方式。

經(jīng)典類的深度優(yōu)先搜索繼承方式:

Python實現(xiàn)打磚塊小游戲代碼實例_第1張圖片

如圖所示
class B(A)
class C(A)
class D(B,C)

(1)若D類有構(gòu)造函數(shù),則重寫所有父類的繼承
(2)若D類沒有構(gòu)造函數(shù),B類有構(gòu)造函數(shù),則D類會繼承B類的構(gòu)造函數(shù)
(3)若D類沒有構(gòu)造函數(shù),B類也沒有構(gòu)造函數(shù),則D類會繼承 A類的構(gòu)造函數(shù),而不是C類的構(gòu)造函數(shù)
(4)若D類沒有構(gòu)造函數(shù),B類也沒有構(gòu)造函數(shù),A類也沒有構(gòu)造函數(shù),則D類才會繼承C類的構(gòu)造函數(shù)

新式類的廣度優(yōu)先搜索繼承方式:


Python實現(xiàn)打磚塊小游戲代碼實例_第2張圖片

如圖所示
class B(A)
class C(A)
class D(B,C)

(1)若D類有構(gòu)造函數(shù),則重寫所有父類的繼承
(2)若D類沒有構(gòu)造函數(shù),B類有構(gòu)造函數(shù),則D類會繼承B類的構(gòu)造函數(shù)
(3)若D類沒有構(gòu)造函數(shù),B類也沒有構(gòu)造函數(shù),則D類會繼承 C類的構(gòu)造函數(shù),而不是A類的構(gòu)造函數(shù)
(4)若D類沒有構(gòu)造函數(shù),B類也沒有構(gòu)造函數(shù),C類也沒有構(gòu)造函數(shù),則D類才會繼承A類的構(gòu)造函數(shù)

通過上面的分析,大家應(yīng)該清楚了Python中類的繼承順序,那么問題來了,如果我不想重寫父類的構(gòu)造函數(shù),要子類和父類的構(gòu)造函數(shù)都生效怎么辦?解決辦法需要用到super關(guān)鍵字,對直接父類對象的引用,可以通過super來訪問父類中被子類覆蓋的方法或?qū)傩浴?

            
class A(object):
	def __init__(self,*args,**kw)
class B(A):
	def __init__(self,*args,**kw)
		super(B,self).__init__(*args,**kw)
class C(A):
	def __init__(self,*args,**kw)
		super(C,self).__init__(*args,**kw)
class D(B,C):
	def __init__(self,*args,**kw)
		super(D,self).__init__(*args,**kw)
		super(B,self).__init__(*args,**kw)

          

2.碰撞檢測的數(shù)學(xué)模型

其實,編程問題到最后就是數(shù)學(xué)問題,這個游戲涉及到2D圓形與矩形的碰撞檢測問題:

碰撞檢測原理:通過找出矩形上離圓心最近的點,然后通過判斷該點與圓心的距離是否小于圓的半徑,若小于則為碰撞。

那如何找出矩形上離圓心最近的點呢?下面我們從 x 軸、y 軸兩個方向分別進行尋找。為了方便描述,我們先約定以下變量:

(1)矩形上離圓心最近的點為變量:closestpoint = [x, y]
(2)矩形 rect = [x, y, l, w] 左上角與長寬 length,wide
(3)圓形 circle = [x, y, r] 圓心與半徑


Python實現(xiàn)打磚塊小游戲代碼實例_第3張圖片

首先是 x 軸:
如果圓心在矩形的左側(cè)(if circle_x < rect_x),那么 closestpoint_x = rect_x。
如果圓心在矩形的右側(cè)(elif circle_x > rect_x + rect_l),那么 closestpoint_x = rect_x + rect_l。
如果圓心在矩形的正上下方(else),那么 closestpoint_x = circle_x。

同理,對于 y 軸:
如果圓心在矩形的上方(if circle_y < rect_y),那么 closestpoint_y = rect_y。
如果圓心在矩形的下方(elif circle_y > rect_y + rect_w)),那么 closestpoint_y = rect_y + rect_w。
圓形圓心在矩形的正左右兩側(cè)(else),那么 closestpoint_y = circle_y。

因此,通過上述方法即可找出矩形上離圓心最近的點了,然后通過“兩點之間的距離公式”得出“最近點”與“圓心”的距離,最后將其與圓的半徑相比,即可判斷是否發(fā)生碰撞。
distance=math.sqrt(math.pow(closestpoint_x-circle_x,2)+math.pow(closestpoint_y-circle_y,2))

if distance < circle.r :
return True ?C 發(fā)生碰撞
else :
return False ?C 未發(fā)生碰撞

以上所述是小編給大家介紹的Python打磚塊小游戲詳解整合,希望對大家有所幫助,如果大家有任何疑問請給我留言,小編會及時回復(fù)大家的。在此也非常感謝大家對腳本之家網(wǎng)站的支持!


更多文章、技術(shù)交流、商務(wù)合作、聯(lián)系博主

微信掃碼或搜索:z360901061

微信掃一掃加我為好友

QQ號聯(lián)系: 360901061

您的支持是博主寫作最大的動力,如果您喜歡我的文章,感覺我的文章對您有幫助,請用微信掃描下面二維碼支持博主2元、5元、10元、20元等您想捐的金額吧,狠狠點擊下面給點支持吧,站長非常感激您!手機微信長按不能支付解決辦法:請將微信支付二維碼保存到相冊,切換到微信,然后點擊微信右上角掃一掃功能,選擇支付二維碼完成支付。

【本文對您有幫助就好】

您的支持是博主寫作最大的動力,如果您喜歡我的文章,感覺我的文章對您有幫助,請用微信掃描上面二維碼支持博主2元、5元、10元、自定義金額等您想捐的金額吧,站長會非常 感謝您的哦!!!

發(fā)表我的評論
最新評論 總共0條評論
主站蜘蛛池模板: 香蕉尹人综合精品 | 欧美孕妇乱大交xxxxx | 999热在线精品观看全部 | 欧美大尺度 边吃奶边做 | 日本高清不卡二区 | 一区二区三区免费精品视频 | 国产精品久久久久a影院 | 久久精品国产精品亚洲精品 | 欧美日韩亚洲国产一区二区三区 | 欧美成人天天综合在线视色 | 欧美日韩国产高清一区二区三区 | 在线观看国产久青草 | 欧美精品亚洲精品日韩 | 中文字幕在线视频观看 | 四虎国产精品成人永久免费影视 | avtt亚洲一区中文字幕 | 91最新国产 | 欧美三级毛片 | 国产一级淫片a免费播放口欧美 | 亚洲性激情 | 香蕉福利久久福利久久香蕉 | 国产日韩精品视频一区二区三区 | 日韩欧美国产卡通动漫 | 免费国产不卡午夜福在线 | jizz丝袜| 久久婷婷是五月综合色狠狠 | 97se亚洲国产综合自在线观看 | 在线精品自拍 | 国产精品一区伦免视频播放 | 亚洲成a人片在线v观看 | 狠狠艹 | 偷亚洲偷国产欧美高清 | 91亚洲精品国产自在现线 | 国产成人毛片亚洲精品不卡 | 久九精品 | 99国产精品视频免费观看 | 久久综合中文字幕一区二区 | 欧美在线播放成人免费 | 国产麻豆高清视频在线第一页 | www.天天操 | 久久91精品国产91久久户 |