《用Python+PyGame开发双人生存游戏!源码解析+完整开发思路分享》
导语
"你是否想过用Python开发一款可玩性高的双人合作游戏?本文将分享如何从零开始实现一款类《吸血鬼幸存者》的生存射击游戏!包含完整源码解析、角色系统设计、敌人AI逻辑等核心技术点,文末提供完整代码包下载!"
哈哈,怪物可以换成同学 的qq头像
游戏内容如下:

一、游戏展示 & 核心功能
-
游戏截图/GIF动图
(建议添加游戏实际运行画面,展示双人操作、敌人生成、技能特效等) -
核心玩法特性
- 双人本地合作模式(WASD vs 方向键控制)
- 两种可选角色:四方向攻击 vs 对角线攻击
- 动态敌人系统:普通敌人 + 精英Boss
- 角色成长体系:经验值升级/属性强化
- 时间限制生存模式(5分钟倒计时)
二、技术实现亮点
-
技术栈
- 语言:Python 3.x
- 核心库:PyGame
- 开发周期:约10小时
-
关键技术点
- 精灵(Sprite)系统:玩家/敌人/子弹的统一管理 - 基于三角函数的子弹轨迹计算(8方向射击) - 敌人AI:自动追踪玩家 + 精英怪弹幕攻击 - 动态难度系统:敌人生成速度随玩家等级提升 - 经验球漂浮动画(正弦函数实现) - 多菜单系统:主菜单/角色选择/游戏内HUD
三、代码结构解析
python
# 代码模块示意图
├── Assets/ # 资源文件夹
│ ├── image/ # 游戏素材(角色/敌人/背景图)
├── main.py # 主程序入口
│ ├── 核心类:
│ │ - Player # 玩家角色(移动/攻击/成长)
│ │ - Enemy # 基础敌人AI
│ │ - EliteEnemy # 精英Boss(弹幕攻击)
│ │ - Bullet # 子弹物理系统
│ │ - Button # 交互式GUI按钮
│ ├── 游戏流程:
│ │ - main_menu() # 主菜单
│ │ - role_selection() # 角色选择
│ │ - game_loop() # 核心游戏循环
四、关键代码解读
-
角色控制系统
python
class Player(pygame.sprite.Sprite):def get_attack_directions(self):# 角色1:四方向射击 | 角色2:对角线射击return ["up", "down", "left", "right"] if self.role_type == 1 else ["up_left", "up_right", ...] -
精英敌人弹幕算法
python
class EliteEnemy(Enemy):def shoot(self, target):# 45度间隔的8方向弹幕for angle in range(0, 360, 45):rad = math.radians(angle)bullet = EliteBullet(..., math.cos(rad)*speed, math.sin(rad)*speed) -
动态难度机制
python
# 敌人生成速度随玩家等级提升 enemy_spawn_interval = 60 - (sum(p.level for p in players) * 2)
五、如何运行游戏
-
环境准备
bash
pip install pygame -
文件结构要求
project/ ├── main.py └── image/├── role1.png # 角色1素材├── enemy.png # 敌人素材└── ... -
启动命令
bash
python main.py
六、开发心得 & 优化方向
-
踩坑经验
- PyGame精灵组的碰撞检测优化
- 双人模式下的事件冲突处理
- 游戏节奏平衡性调试
-
待优化项
- 添加音效系统
- 实现网络联机功能
- 增加更多角色/技能树
- 开发关卡编辑器
七、完整代码获取
"关注+私信回复【生存游戏】获取完整代码包和素材资源!"
骗你的,代码就在这,复制就能用!
# -*- coding: utf-8 -*-
import os
import pygame
import random
import math# 初始化配置
BASE_DIR = os.path.dirname(os.path.abspath(__file__))
IMAGE_DIR = os.path.join(BASE_DIR, 'image')RESOURCES = {'role1': os.path.join(IMAGE_DIR, 'role1.png'),'role2': os.path.join(IMAGE_DIR, 'role2.png'),'enemy': os.path.join(IMAGE_DIR, 'enemy.png'),'elite': os.path.join(IMAGE_DIR, 'elite.png'),'bullet': os.path.join(IMAGE_DIR, 'bullet.png'),'exp_orb': os.path.join(IMAGE_DIR, 'exp_orb.png'),'background': os.path.join(IMAGE_DIR, 'background.png')
}pygame.init()
WIDTH, HEIGHT = 800, 600
screen = pygame.display.set_mode((WIDTH, HEIGHT))
clock = pygame.time.Clock()# 颜色定义
WHITE = (255, 255, 255)
GRAY = (100, 100, 100)
BUTTON_COLOR = (50, 150, 50)
HOVER_COLOR = (70, 170, 70)
RED = (255, 0, 0)
YELLOW = (255, 255, 0)
ELITE_BULLET_COLOR = (255, 165, 0)font = pygame.font.Font(None, 24)def load_image(path, default_size=(30, 30)):try:image = pygame.image.load(path).convert_alpha()return pygame.transform.scale(image, default_size)except:surf = pygame.Surface(default_size)surf.fill(RED)return surfGAME_IMAGES = {'role1': load_image(RESOURCES['role1']),'role2': load_image(RESOURCES['role2']),'enemy': load_image(RESOURCES['enemy'], (20, 20)),'elite': load_image(RESOURCES['elite'], (40, 40)),'bullet': load_image(RESOURCES['bullet'], (10, 10)),'exp_orb': load_image(RESOURCES['exp_orb'], (10, 10)),'background': load_image(RESOURCES['background'], (WIDTH, HEIGHT))
}class Button:def __init__(self, text, x, y, w, h):self.rect = pygame.Rect(x, y, w, h)self.text = textself.color = BUTTON_COLORself.hover = Falsedef draw(self, surface):color = HOVER_COLOR if self.hover else BUTTON_COLORpygame.draw.rect(surface, color, self.rect, border_radius=5)text_surf = font.render(self.text, True, WHITE)text_rect = text_surf.get_rect(center=self.rect.center)surface.blit(text_surf, text_rect)def check_hover(self, mouse_pos):self.hover = self.rect.collidepoint(mouse_pos)class Player(pygame.sprite.Sprite):def __init__(self, controls, role_type, pos_offset=0, is_player2=False):super().__init__()self.role_type = role_typeself.image = GAME_IMAGES['role2' if role_type == 2 else 'role1']self.rect = self.image.get_rect(center=(WIDTH // 2 + pos_offset, HEIGHT // 2))self.speed = 5self.health = 100self.exp = 0self.level = 1self.max_exp = 100self.kills = 0self.controls = controlsdef update(self, keys):if keys[self.controls['up']]: self.rect.y -= self.speedif keys[self.controls['down']]: self.rect.y += self.speedif keys[self.controls['left']]: self.rect.x -= self.speedif keys[self.controls['right']]: self.rect.x += self.speedself.rect.clamp_ip(screen.get_rect())def get_attack_directions(self):return ["up", "down", "left", "right"] if self.role_type == 1 else ["up_left", "up_right", "down_left","down_right"]class Bullet(pygame.sprite.Sprite):def __init__(self, x, y, direction):super().__init__()self.image = GAME_IMAGES['bullet']self.rect = self.image.get_rect(center=(x, y))self.speed = 8self.start_pos = (x, y)self.max_distance = 300self.penetration = 2dir_mapping = {"up": (0, -1), "down": (0, 1),"left": (-1, 0), "right": (1, 0),"up_left": (-math.sqrt(0.5), -math.sqrt(0.5)),"up_right": (math.sqrt(0.5), -math.sqrt(0.5)),"down_left": (-math.sqrt(0.5), math.sqrt(0.5)),"down_right": (math.sqrt(0.5), math.sqrt(0.5))}dx_mult, dy_mult = dir_mapping[direction]self.dx = dx_mult * self.speedself.dy = dy_mult * self.speeddef update(self):self.rect.x += self.dxself.rect.y += self.dyif math.hypot(self.rect.x - self.start_pos[0], self.rect.y - self.start_pos[1]) > self.max_distance:self.kill()class Enemy(pygame.sprite.Sprite):def __init__(self):super().__init__()self.image = GAME_IMAGES['enemy']self.rect = self.image.get_rect(center=(random.choice([-100, WIDTH + 100]), random.randint(0, HEIGHT)))self.speed = 2def update(self, targets):if not targets: returnnearest = min(targets, key=lambda t: math.hypot(t.rect.x - self.rect.x, t.rect.y - self.rect.y))dx = nearest.rect.x - self.rect.xdy = nearest.rect.y - self.rect.ydist = math.hypot(dx, dy)if dist != 0:self.rect.x += dx / dist * self.speedself.rect.y += dy / dist * self.speedclass EliteEnemy(pygame.sprite.Sprite):def __init__(self):super().__init__()self.image = GAME_IMAGES['elite']self.rect = self.image.get_rect(center=(random.choice([-100, WIDTH + 100]), random.randint(0, HEIGHT)))self.speed = 1.5self.health = 50self.max_health = 50self.shoot_timer = 0self.bullet_speed = 5def update(self, targets):if not targets: returnnearest = min(targets, key=lambda t: math.hypot(t.rect.x - self.rect.x, t.rect.y - self.rect.y))dx = nearest.rect.x - self.rect.xdy = nearest.rect.y - self.rect.ydist = math.hypot(dx, dy)if dist != 0:self.rect.x += dx / dist * self.speedself.rect.y += dy / dist * self.speedself.shoot_timer += 1if self.shoot_timer >= 60:self.shoot(nearest)self.shoot_timer = 0def shoot(self, target):for angle in range(0, 360, 45):rad = math.radians(angle)bullet = EliteBullet(self.rect.centerx,self.rect.centery,math.cos(rad) * self.bullet_speed,math.sin(rad) * self.bullet_speed)bullets.add(bullet)all_sprites.add(bullet)class EliteBullet(pygame.sprite.Sprite):def __init__(self, x, y, dx, dy):super().__init__()self.image = pygame.Surface((15, 15))self.image.fill(ELITE_BULLET_COLOR)self.rect = self.image.get_rect(center=(x, y))self.dx = dxself.dy = dyself.max_distance = 400self.start_pos = (x, y)def update(self):self.rect.x += self.dxself.rect.y += self.dyif math.hypot(self.rect.x - self.start_pos[0], self.rect.y - self.start_pos[1]) > self.max_distance:self.kill()class ExpOrb(pygame.sprite.Sprite):def __init__(self, x, y):super().__init__()self.image = GAME_IMAGES['exp_orb']self.rect = self.image.get_rect(center=(x, y))self.float_timer = 0def update(self):self.float_timer += 1self.rect.y += math.sin(self.float_timer * 0.1) * 0.5def draw_hud(surface, players, time_left):time_text = font.render(f"Time: {time_left // 60:02}:{time_left % 60:02}", True, WHITE)surface.blit(time_text, (WIDTH // 2 - 60, 10))for i, player in enumerate(players):y_offset = 40 + i * 80pygame.draw.rect(surface, GRAY, (10, y_offset, 100, 10))health_width = int((player.health / 100.0) * 100)pygame.draw.rect(surface, RED, (10, y_offset, health_width, 10))pygame.draw.rect(surface, GRAY, (10, y_offset + 20, 100, 10))exp_width = int((player.exp / float(player.max_exp)) * 100)pygame.draw.rect(surface, YELLOW, (10, y_offset + 20, exp_width, 10))info_text = font.render(f"P{i + 1} Lv{player.level} K{player.kills}", True, WHITE)surface.blit(info_text, (10, y_offset + 40))def role_selection_menu(player_count):roles = []buttons = []descriptions = ["Role 1: 4-Direction Attack","Role 2: Diagonal Attack"]for i in range(player_count):y_base = 150 + i * 150buttons.append([Button(f"Player{i + 1} Role1", WIDTH // 2 - 250, y_base, 200, 50),Button(f"Player{i + 1} Role2", WIDTH // 2 + 50, y_base, 200, 50)])confirm_btn = Button("Start Game", WIDTH // 2 - 100, HEIGHT - 100, 200, 50)while True:screen.fill((30, 30, 30))mouse_pos = pygame.mouse.get_pos()for event in pygame.event.get():if event.type == pygame.QUIT:pygame.quit()return []if event.type == pygame.MOUSEBUTTONDOWN:for i, pair in enumerate(buttons):for j, btn in enumerate(pair):if btn.rect.collidepoint(mouse_pos):roles = roles[:i] + [j + 1] + roles[i + 1:] if len(roles) > i else roles + [j + 1]if confirm_btn.rect.collidepoint(mouse_pos) and len(roles) == player_count:return rolesdesc_y = 100for desc in descriptions:text = font.render(desc, True, WHITE)screen.blit(text, (WIDTH // 2 - text.get_width() // 2, desc_y))desc_y += 30for i, pair in enumerate(buttons):for j, btn in enumerate(pair):btn.check_hover(mouse_pos)btn.draw(screen)if i < len(roles) and roles[i] == j + 1:pygame.draw.rect(screen, YELLOW, btn.rect.inflate(10, 10), 3, border_radius=7)confirm_btn.check_hover(mouse_pos)confirm_btn.draw(screen)pygame.display.flip()clock.tick(30)def main_menu():buttons = [Button("1 Player", WIDTH // 2 - 100, HEIGHT // 2 - 50, 200, 50),Button("2 Players", WIDTH // 2 - 100, HEIGHT // 2 + 20, 200, 50)]while True:screen.fill((30, 30, 30))mouse_pos = pygame.mouse.get_pos()for event in pygame.event.get():if event.type == pygame.QUIT:pygame.quit()return (0, [])if event.type == pygame.MOUSEBUTTONDOWN:for i, btn in enumerate(buttons):if btn.rect.collidepoint(mouse_pos):roles = role_selection_menu(i + 1)if roles:return (i + 1, roles)title = font.render("Vampire Survivors", True, WHITE)screen.blit(title, (WIDTH // 2 - title.get_width() // 2, 100))for btn in buttons:btn.check_hover(mouse_pos)btn.draw(screen)pygame.display.flip()clock.tick(30)def game_loop(player_count, roles):background = GAME_IMAGES['background']controls = [{'up': pygame.K_w, 'down': pygame.K_s, 'left': pygame.K_a, 'right': pygame.K_d},{'up': pygame.K_UP, 'down': pygame.K_DOWN, 'left': pygame.K_LEFT, 'right': pygame.K_RIGHT}]players = pygame.sprite.Group()for i in range(player_count):player = Player(controls=controls[i],role_type=roles[i],pos_offset=-50 + i * 100,is_player2=(i == 1))players.add(player)all_sprites = pygame.sprite.Group(players)enemies = pygame.sprite.Group()bullets = pygame.sprite.Group()exp_orbs = pygame.sprite.Group()enemy_spawn_timer = 0attack_timer = 0start_ticks = pygame.time.get_ticks()time_limit = 300running = Truewhile running:screen.blit(background, (0, 0))keys = pygame.key.get_pressed()elapsed_seconds = (pygame.time.get_ticks() - start_ticks) // 1000time_left = max(time_limit - elapsed_seconds, 0)if time_left <= 0:running = Falsefor event in pygame.event.get():if event.type == pygame.QUIT:running = Falseenemy_spawn_timer += 1if enemy_spawn_timer >= 60 - (sum(p.level for p in players) * 2):if random.random() < 0.1:enemy = EliteEnemy()else:enemy = Enemy()enemies.add(enemy)all_sprites.add(enemy)enemy_spawn_timer = 0attack_timer += 1if attack_timer >= 30:for player in players:for direction in player.get_attack_directions():bullet = Bullet(player.rect.centerx, player.rect.centery, direction)bullets.add(bullet)all_sprites.add(bullet)attack_timer = 0for player in players:player.update(keys)enemies.update(players)bullets.update()exp_orbs.update()for bullet in bullets:hits = pygame.sprite.spritecollide(bullet, enemies, False)if hits:bullet.penetration -= 1for enemy in hits:if isinstance(enemy, EliteEnemy):enemy.health -= 2if enemy.health <= 0:enemy.kill()for _ in range(5):exp_orb = ExpOrb(enemy.rect.centerx, enemy.rect.centery)exp_orbs.add(exp_orb)all_sprites.add(exp_orb)else:enemy.kill()exp_orb = ExpOrb(enemy.rect.centerx, enemy.rect.centery)exp_orbs.add(exp_orb)all_sprites.add(exp_orb)for player in players:if math.hypot(player.rect.x - enemy.rect.x, player.rect.y - enemy.rect.y) < 100:player.kills += 1if bullet.penetration <= 0:bullet.kill()for player in players:hits = pygame.sprite.spritecollide(player, exp_orbs, True)if hits:player.exp += 10 * len(hits)if player.exp >= player.max_exp:player.level += 1player.exp -= player.max_expplayer.max_exp = int(player.max_exp * 1.5)player.speed += 0.5if pygame.sprite.spritecollide(player, enemies, True):player.health -= 10bullet_hits = pygame.sprite.spritecollide(player, bullets, True)if bullet_hits:player.health -= 5alive_players = [p for p in players if p.health > 0]if not alive_players or time_left <= 0:running = Falseall_sprites.draw(screen)draw_hud(screen, players, time_left)pygame.display.flip()clock.tick(30)pygame.quit()if __name__ == "__main__":player_count, roles = main_menu()if player_count > 0:game_loop(player_count, roles)
下面是python2的代码。方便不同环境的兄弟们运行:
# -*- coding: utf-8 -*-
import os
import pygame
import random
import math
from datetime import datetime# 初始化路径
BASE_DIR = os.path.dirname(os.path.abspath(__file__))
IMAGE_DIR = os.path.join(BASE_DIR, 'image')RESOURCES = {'role1': os.path.join(IMAGE_DIR, 'role1.png'),'role2': os.path.join(IMAGE_DIR, 'role2.png'),'enemy': os.path.join(IMAGE_DIR, 'enemy.png'),'bullet': os.path.join(IMAGE_DIR, 'bullet.png'),'exp_orb': os.path.join(IMAGE_DIR, 'exp_orb.png'),'background': os.path.join(IMAGE_DIR, 'background.png')
}pygame.init()
WIDTH, HEIGHT = 800, 600
screen = pygame.display.set_mode((WIDTH, HEIGHT))
clock = pygame.time.Clock()# 颜色定义
WHITE = (255, 255, 255)
GRAY = (100, 100, 100)
BUTTON_COLOR = (50, 150, 50)
HOVER_COLOR = (70, 170, 70)
RED = (255, 0, 0)
YELLOW = (255, 255, 0)font = pygame.font.Font(None, 24)def load_image(path, default_size=(30, 30)):try:image = pygame.image.load(path).convert_alpha()return pygame.transform.scale(image, default_size)except:surf = pygame.Surface(default_size)surf.fill(RED)return surfGAME_IMAGES = {'role1': load_image(RESOURCES['role1']),'role2': load_image(RESOURCES['role2']),'enemy': load_image(RESOURCES['enemy'], (20, 20)),'bullet': load_image(RESOURCES['bullet'], (10, 10)),'exp_orb': load_image(RESOURCES['exp_orb'], (10, 10)),'background': load_image(RESOURCES['background'], (WIDTH, HEIGHT))
}class Button:def __init__(self, text, x, y, w, h):self.rect = pygame.Rect(x, y, w, h)self.text = textself.color = BUTTON_COLORself.hover = Falsedef draw(self, surface):color = HOVER_COLOR if self.hover else BUTTON_COLORpygame.draw.rect(surface, color, self.rect, border_radius=5)text_surf = font.render(self.text, True, WHITE)text_rect = text_surf.get_rect(center=self.rect.center)surface.blit(text_surf, text_rect)def check_hover(self, mouse_pos):self.hover = self.rect.collidepoint(mouse_pos)class Player(pygame.sprite.Sprite):def __init__(self, controls, role_type, pos_offset=0, is_player2=False):pygame.sprite.Sprite.__init__(self)self.role_type = role_typeself.image = GAME_IMAGES['role2' if role_type == 2 else 'role1']self.rect = self.image.get_rect(center=(WIDTH//2 + pos_offset, HEIGHT//2))self.speed = 5self.health = 100self.exp = 0self.level = 1self.max_exp = 100self.kills = 0self.controls = controlsdef update(self, keys):if keys[self.controls['up']]:self.rect.y -= self.speedif keys[self.controls['down']]:self.rect.y += self.speedif keys[self.controls['left']]:self.rect.x -= self.speedif keys[self.controls['right']]:self.rect.x += self.speedself.rect.clamp_ip(screen.get_rect())def get_attack_directions(self):if self.role_type == 1:return ["up", "down", "left", "right"]else:return ["up_left", "up_right", "down_left", "down_right"]class Bullet(pygame.sprite.Sprite):def __init__(self, x, y, direction):pygame.sprite.Sprite.__init__(self)self.image = GAME_IMAGES['bullet']self.rect = self.image.get_rect(center=(x, y))self.speed = 8self.start_pos = (x, y)self.max_distance = 300self.penetration = 2self.dx, self.dy = 0, 0dir_mapping = {"up": (0, -1),"down": (0, 1),"left": (-1, 0),"right": (1, 0),"up_left": (-math.sqrt(0.5), -math.sqrt(0.5)),"up_right": (math.sqrt(0.5), -math.sqrt(0.5)),"down_left": (-math.sqrt(0.5), math.sqrt(0.5)),"down_right": (math.sqrt(0.5), math.sqrt(0.5))}dx_mult, dy_mult = dir_mapping[direction]self.dx = dx_mult * self.speedself.dy = dy_mult * self.speeddef update(self):self.rect.x += self.dxself.rect.y += self.dyif math.hypot(self.rect.x - self.start_pos[0], self.rect.y - self.start_pos[1]) > self.max_distance:self.kill()class Enemy(pygame.sprite.Sprite):def __init__(self):pygame.sprite.Sprite.__init__(self)self.image = GAME_IMAGES['enemy']self.rect = self.image.get_rect(center=(random.choice([-100, WIDTH+100]), random.randint(0, HEIGHT)))self.speed = 2def update(self, targets):if not targets: returnnearest = min(targets, key=lambda t: math.hypot(t.rect.x-self.rect.x, t.rect.y-self.rect.y))dx = nearest.rect.x - self.rect.xdy = nearest.rect.y - self.rect.ydist = math.hypot(dx, dy)if dist != 0:self.rect.x += dx / dist * self.speedself.rect.y += dy / dist * self.speedclass ExpOrb(pygame.sprite.Sprite):def __init__(self, x, y):pygame.sprite.Sprite.__init__(self)self.image = GAME_IMAGES['exp_orb']self.rect = self.image.get_rect(center=(x, y))self.float_timer = 0def update(self):self.float_timer += 1self.rect.y += math.sin(self.float_timer * 0.1) * 0.5def draw_hud(surface, players, time_left):time_text = font.render("Time: {:02}:{:02}".format(time_left//60, time_left%60), True, WHITE)surface.blit(time_text, (WIDTH//2 - 60, 10))for i, player in enumerate(players):y_offset = 40 + i*80pygame.draw.rect(surface, GRAY, (10, y_offset, 100, 10))health_width = int((player.health / 100.0) * 100)pygame.draw.rect(surface, RED, (10, y_offset, health_width, 10))pygame.draw.rect(surface, GRAY, (10, y_offset+20, 100, 10))exp_width = int((player.exp / float(player.max_exp)) * 100)pygame.draw.rect(surface, YELLOW, (10, y_offset+20, exp_width, 10))info_text = font.render("P{} Lv{} K{}".format(i+1, player.level, player.kills), True, WHITE)surface.blit(info_text, (10, y_offset+40))def role_selection_menu(player_count):roles = []buttons = []descriptions = ["Role 1: 4-Direction Attack","Role 2: Diagonal Attack"]for i in range(player_count):y_base = 150 + i*150buttons.append([Button("Player{} Role1".format(i+1), WIDTH//2-250, y_base, 200, 50),Button("Player{} Role2".format(i+1), WIDTH//2+50, y_base, 200, 50)])confirm_btn = Button("Start Game", WIDTH//2-100, HEIGHT-100, 200, 50)while True:screen.fill((30, 30, 30))mouse_pos = pygame.mouse.get_pos()for event in pygame.event.get():if event.type == pygame.QUIT:pygame.quit()return []if event.type == pygame.MOUSEBUTTONDOWN:for i, pair in enumerate(buttons):for j, btn in enumerate(pair):if btn.rect.collidepoint(mouse_pos):if len(roles) <= i:roles.append(j+1)else:roles[i] = j+1if confirm_btn.rect.collidepoint(mouse_pos) and len(roles) == player_count:return rolesdesc_y = 100for desc in descriptions:text = font.render(desc, True, WHITE)screen.blit(text, (WIDTH//2 - text.get_width()//2, desc_y))desc_y += 30for i, pair in enumerate(buttons):for j, btn in enumerate(pair):btn.check_hover(mouse_pos)btn.draw(screen)if i < len(roles) and roles[i] == j+1:pygame.draw.rect(screen, YELLOW, btn.rect.inflate(10,10), 3, border_radius=7)confirm_btn.check_hover(mouse_pos)confirm_btn.draw(screen)pygame.display.flip()clock.tick(30)def main_menu():buttons = [Button("1 Player", WIDTH//2-100, HEIGHT//2-50, 200, 50),Button("2 Players", WIDTH//2-100, HEIGHT//2+20, 200, 50)]while True:screen.fill((30, 30, 30))mouse_pos = pygame.mouse.get_pos()for event in pygame.event.get():if event.type == pygame.QUIT:pygame.quit()return (0, [])if event.type == pygame.MOUSEBUTTONDOWN:for i, btn in enumerate(buttons):if btn.rect.collidepoint(mouse_pos):selected = i+1roles = role_selection_menu(selected)if roles:return (selected, roles)title = font.render("Vampire Survivors", True, WHITE)screen.blit(title, (WIDTH//2 - title.get_width()//2, 100))for btn in buttons:btn.check_hover(mouse_pos)btn.draw(screen)pygame.display.flip()clock.tick(30)def game_loop(player_count, roles):background = GAME_IMAGES['background']controls = [{'up': pygame.K_w, 'down': pygame.K_s, 'left': pygame.K_a, 'right': pygame.K_d},{'up': pygame.K_UP, 'down': pygame.K_DOWN, 'left': pygame.K_LEFT, 'right': pygame.K_RIGHT}]players = pygame.sprite.Group()for i in range(player_count):player = Player(controls=controls[i],role_type=roles[i],pos_offset=-50 + i*100,is_player2=(i==1))players.add(player)all_sprites = pygame.sprite.Group(players)enemies = pygame.sprite.Group()bullets = pygame.sprite.Group()exp_orbs = pygame.sprite.Group()enemy_spawn_timer = 0attack_timer = 0start_ticks = pygame.time.get_ticks()time_limit = 300running = Truewhile running:screen.blit(background, (0, 0))keys = pygame.key.get_pressed()elapsed_seconds = (pygame.time.get_ticks() - start_ticks) // 1000time_left = max(time_limit - elapsed_seconds, 0)if time_left <= 0:running = Falsefor event in pygame.event.get():if event.type == pygame.QUIT:running = Falseenemy_spawn_timer += 1if enemy_spawn_timer >= 60 - (sum(p.level for p in players)*2):enemy = Enemy()enemies.add(enemy)all_sprites.add(enemy)enemy_spawn_timer = 0attack_timer += 1if attack_timer >= 30:for player in players:directions = player.get_attack_directions()for direction in directions:bullet = Bullet(player.rect.centerx, player.rect.centery, direction)bullets.add(bullet)all_sprites.add(bullet)attack_timer = 0for player in players:player.update(keys)enemies.update(players)bullets.update()exp_orbs.update()for bullet in bullets:hits = pygame.sprite.spritecollide(bullet, enemies, False)if hits:bullet.penetration -= 1for enemy in hits:enemy.kill()exp_orb = ExpOrb(enemy.rect.centerx, enemy.rect.centery)exp_orbs.add(exp_orb)all_sprites.add(exp_orb)for player in players:if math.hypot(player.rect.x-enemy.rect.x, player.rect.y-enemy.rect.y) < 100:player.kills += 1if bullet.penetration <= 0:bullet.kill()for player in players:hits = pygame.sprite.spritecollide(player, exp_orbs, True)if hits:player.exp += 10 * len(hits)if player.exp >= player.max_exp:player.level += 1player.exp -= player.max_expplayer.max_exp = int(player.max_exp * 1.5)player.speed += 0.5alive_players = [p for p in players if p.health > 0]for player in alive_players:if pygame.sprite.spritecollide(player, enemies, True):player.health -= 10if len(alive_players) == 0 or time_left <= 0:running = Falseall_sprites.draw(screen)draw_hud(screen, players, time_left)pygame.display.flip()clock.tick(30)pygame.quit()if __name__ == "__main__":player_count, roles = main_menu()if player_count > 0:game_loop(player_count, roles)
互动引导
-
投票:
"如果让你添加新功能,你会选择?
A) 联机对战 B) 技能组合 C) BOSS战 D) 自定义角色" -
讨论:
"你在用Python开发游戏时遇到过哪些难题?欢迎评论区交流!"
版权声明
"本项目为开源学习作品,遵循MIT协议,欢迎二次开发但需保留原作者信息"
相关文章:
《用Python+PyGame开发双人生存游戏!源码解析+完整开发思路分享》
导语 "你是否想过用Python开发一款可玩性高的双人合作游戏?本文将分享如何从零开始实现一款类《吸血鬼幸存者》的生存射击游戏!包含完整源码解析、角色系统设计、敌人AI逻辑等核心技术点,文末提供完整代码包下载!" 哈…...
优选算法系列(1. 双指针_上)
目录 双指针 一:移动零(easy) 题目链接:移动零 解法: 代码: 二:复写零(easy) 题目链接:复写零 编辑 解法: 代码: 三:快乐…...
永洪科技深度分析实战,零售企业的销量预测
随着人工智能技术的不断发展,智能预测已经成为各个领域的重要应用之一。现在,智能预测技术已经广泛应用于金融、零售、医疗、能源等领域,为企业和个人提供决策支持。 智能预测技术通过分析大量的数据,利用机器学习和深度学习算法…...
c语言笔记 函数参数的等价(上)
这三种写法在 C 语言中是等价的,因为它们都用于声明一个指向二维数组的指针,或者用于声明一个二维数组作为函数参数。它们的等价性源于 C 语言中数组和指针之间的密切关系。让我们逐一分析这三种写法: 在C语言中,当数组作为函数参…...
hive面试题--left join的坑
student 表: 课程表course: 1、key为null, 不关联 select * from student s left join course c on s.id c.s_id;2、on中过滤条件 与 where 过滤条件区别 on and c.id<>‘1001’ 先过滤右表数据,然后与左表关联 select * from student s le…...
CEH与OSCP:网络安全认证对比分析
在网络安全领域,渗透测试被视为至关重要的一环,帮助企业检测和修复系统漏洞。为提升行业标准,许多认证应运而生,其中CEH和OSCP作为行业认可度较高的认证,广泛被网络安全从业者选择。尽管这两者都涉及渗透测试领域&…...
HTML 属性详解:为网页元素赋予更多功能
在构建网页的过程中,HTML 是基础的标记语言,而 HTML 属性则是为 HTML 元素提供附加信息的重要组成部分。 一、属性的基本概念与使用 属性通常出现在 HTML 标签的开始标签内,以 “name"value"” 的形式存在。这里的 “name” 是属…...
Ceph(2):Ceph简介
1 Ceph简介 Ceph使用C语言开发,遵循LGPL协议开源。Sage Weil(Ceph论文发表者)于2011年创立了以Inktank公司主导Ceph的开发和社区维护。2014年Redhat收购inktank公司,并发布Inktank Ceph企业版(ICE)软件,业务场景聚焦云…...
国产编辑器EverEdit - 设置文件类型关联为EverEdit
1 设置-文件关联 1.1 应用场景 文件关联是指在文件管理器中双击某类型的文件,操作系统自动调用可以打开该文件的应用程序,比如:用户双击XXXX.txt文件,系统默认会使用记事本打开该文件。 由于各行各业都会定义特有的文件类型&…...
2025网络安全工程师:软考新挑战与职业发展探析
网络安全工程师的崛起 随着信息技术的迅猛发展,网络安全问题日益凸显,网络安全工程师这一职业逐渐受到社会各界的广泛关注。特别是在2025年,随着各项网络安全法规的完善和实施,网络安全工程师的角色愈发重要。他们不仅是企业信息…...
设计模式之建造者模式:原理、实现与应用
引言 建造者模式(Builder Pattern)是一种创建型设计模式,它通过将复杂对象的构建过程分解为多个简单的步骤,使得对象的创建更加灵活和可维护。建造者模式特别适用于构建具有多个组成部分的复杂对象。本文将深入探讨建造者模式的原…...
【Leetcode 每日一题 - 补卡】2070. 每一个查询的最大美丽值
问题背景 给你一个二维整数数组 i t e m s items items,其中 i t e m s [ i ] [ p r i c e i , b e a u t y i ] items[i] [price_i, beauty_i] items[i][pricei,beautyi] 分别表示每一个物品的 价格 和 美丽值 。 同时给你一个下标从 0 0 0 开始的整数数…...
雪藏HsFreezer(游戏冻结工具) v2.21
HsFreezer 是一款让你可以随心冻结游戏的软件(游戏暂停软件、系统优化软件、进程管理软件),想玩就玩,想停就停,快捷键随心瞬发,单锁模式极致的丝滑切换,当然,不止适用游戏。更有丰富的特色系统优化功能。 PC主机,win掌机,笔记本--无脑装就对了,超大按键超大列表,触控盲操,非常巴…...
2019年蓝桥杯第十届CC++大学B组真题及代码
目录 1A:组队(填空5分_手算) 2B:年号字符(填空5分_进制) 3C:数列求值(填空10分_枚举) 4D:数的分解(填空10分) 5E:迷宫…...
前端安全面试题汇总及参考答案
目录 简述 XSS 攻击的原理及三种常见类型(存储型、反射型、DOM 型) 如何在前端防御 XSS 攻击?列举编码、过滤、CSP 策略的具体实现方式 富文本编辑器场景下如何安全处理用户输入的 HTML 内容? 如何通过 HttpOnly 属性增强 Cookie 安全性?它与 XSS 防御的关系是什么? …...
修复ubuntu下找不到音频设备的问题
出现问题的状态: ALSA 已正确识别到 ZOOM H2n 设备(card 1)sounddevice 库(依赖 PortAudio)未能正确枚举设备 修复方法: 1. 强制 sounddevice 使用 ALSA 后端 默认情况下,sounddevice 可能尝…...
⭐LeetCode周赛 3468. 可行数组的数目——暴力与数学⭐
⭐LeetCode周赛 3468. 可行数组的数目——暴力与数学⭐ 示例 1: 输入:original [1,2,3,4], bounds [[1,2],[2,3],[3,4],[4,5]] 输出:2 解释: 可能的数组为: [1, 2, 3, 4] [2, 3, 4, 5] 示例 2: 输入&…...
在线json转ArkTs-harmonyos
轻松将 JSON 数据转换为类型安全的 ArkTs 接口。快速准确地生成代码,提升开发效率,告别手动编写,让您的开发流程更加流畅! gotool...
Vue 实现AI对话和AI绘图(AIGC)人工智能
我司是主要是负责AIGC人工智能化平台的项目,俗称内容创作及智能工具平台。 授人以鱼不如授人以渔 首先我们要明白AIGC中前端需要做什么 会用到哪些技术栈 。 AIGC前端需要用到的技术栈:Vue,Markdown,SSE。就这个三件套。 前沿:有人觉得AI对…...
Visual Studio Code 基本使用指南
Visual Studio Code(简称 VSCode)是一款由微软开发的免费、开源、跨平台的代码编辑器,凭借其轻量级设计、丰富的插件生态和强大的功能,成为全球开发者的首选工具。本文将从安装配置到核心功能,全面解析 VSCode 的基本使…...
【根据当天日期输出明天的日期(需对闰年做判定)。】2022-5-15
缘由根据当天日期输出明天的日期(需对闰年做判定)。日期类型结构体如下: struct data{ int year; int month; int day;};-编程语言-CSDN问答 struct mdata{ int year; int month; int day; }mdata; int 天数(int year, int month) {switch (month){case 1: case 3:…...
SkyWalking 10.2.0 SWCK 配置过程
SkyWalking 10.2.0 & SWCK 配置过程 skywalking oap-server & ui 使用Docker安装在K8S集群以外,K8S集群中的微服务使用initContainer按命名空间将skywalking-java-agent注入到业务容器中。 SWCK有整套的解决方案,全安装在K8S群集中。 具体可参…...
Unity3D中Gfx.WaitForPresent优化方案
前言 在Unity中,Gfx.WaitForPresent占用CPU过高通常表示主线程在等待GPU完成渲染(即CPU被阻塞),这表明存在GPU瓶颈或垂直同步/帧率设置问题。以下是系统的优化方案: 对惹,这里有一个游戏开发交流小组&…...
Java如何权衡是使用无序的数组还是有序的数组
在 Java 中,选择有序数组还是无序数组取决于具体场景的性能需求与操作特点。以下是关键权衡因素及决策指南: ⚖️ 核心权衡维度 维度有序数组无序数组查询性能二分查找 O(log n) ✅线性扫描 O(n) ❌插入/删除需移位维护顺序 O(n) ❌直接操作尾部 O(1) ✅内存开销与无序数组相…...
Golang dig框架与GraphQL的完美结合
将 Go 的 Dig 依赖注入框架与 GraphQL 结合使用,可以显著提升应用程序的可维护性、可测试性以及灵活性。 Dig 是一个强大的依赖注入容器,能够帮助开发者更好地管理复杂的依赖关系,而 GraphQL 则是一种用于 API 的查询语言,能够提…...
Java面试专项一-准备篇
一、企业简历筛选规则 一般企业的简历筛选流程:首先由HR先筛选一部分简历后,在将简历给到对应的项目负责人后再进行下一步的操作。 HR如何筛选简历 例如:Boss直聘(招聘方平台) 直接按照条件进行筛选 例如:…...
嵌入式学习笔记DAY33(网络编程——TCP)
一、网络架构 C/S (client/server 客户端/服务器):由客户端和服务器端两个部分组成。客户端通常是用户使用的应用程序,负责提供用户界面和交互逻辑 ,接收用户输入,向服务器发送请求,并展示服务…...
【分享】推荐一些办公小工具
1、PDF 在线转换 https://smallpdf.com/cn/pdf-tools 推荐理由:大部分的转换软件需要收费,要么功能不齐全,而开会员又用不了几次浪费钱,借用别人的又不安全。 这个网站它不需要登录或下载安装。而且提供的免费功能就能满足日常…...
tauri项目,如何在rust端读取电脑环境变量
如果想在前端通过调用来获取环境变量的值,可以通过标准的依赖: std::env::var(name).ok() 想在前端通过调用来获取,可以写一个command函数: #[tauri::command] pub fn get_env_var(name: String) -> Result<String, Stri…...
区块链技术概述
区块链技术是一种去中心化、分布式账本技术,通过密码学、共识机制和智能合约等核心组件,实现数据不可篡改、透明可追溯的系统。 一、核心技术 1. 去中心化 特点:数据存储在网络中的多个节点(计算机),而非…...

