《用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逻辑等核心技术点,文末提供完整代码包下载!" 哈…...
【ES6】在ES6中自定义数组
在ES6中是允许自定义类扩展基础类型的,因为这些基础类型是有构造函数的,在JS中类就是函数。 // 自定义数组 class myArray extends Array {constructor() {super();} }let arr new myArray();arr.push(1);console.log(arr);重写Array的原生方法 ES6的…...
软件开发项目有哪些风险
软件开发项目风险主要包括 需求不明确、技术实现难度大、进度延误、成本超支、质量问题。其中,需求不明确可能导致功能设计反复修改;技术实现难度大会使开发过程中不断遇到未知挑战;进度延误常常因资源配置不足或变更频繁而发生;成…...
47.HarmonyOS NEXT 登录模块开发教程(二):一键登录页面实现
温馨提示:本篇博客的详细代码已发布到 git : https://gitcode.com/nutpi/HarmonyosNext 可以下载运行哦! HarmonyOS NEXT 登录模块开发教程(二):一键登录页面实现 文章目录 HarmonyOS NEXT 登录模块开发教程࿰…...
RAGFlow版本升级-Win10系统Docker
下载源码压缩包 https://github.com/infiniflow/ragflow.git 删除旧版本代码文件夹,把下载的代码解压到原先目录 更新一下env文件:ragflow/docker/.env 把值改为最新版本即可 RAGFLOW_IMAGEinfiniflow/ragflow:v0.17.1 更新一下docker docker compose -…...
dns劫持是什么?常见的劫持类型有哪些?如何预防?
DNS劫持的定义 DNS劫持(Domain Name System Hijacking)是一种网络攻击手段,攻击者通过篡改域名解析的过程,将用户对某个域名的访问请求重定向到错误或恶意的IP地址。这种攻击可能导致用户访问到钓鱼网站、恶意广告页面࿰…...
Python精进系列: isinstance 函数
Python isinstance函数:类型检查的得力助手 目录 Python isinstance函数:类型检查的得力助手引言一、isinstance函数基础语法结构简单示例 二、isinstance函数的应用场景函数参数类型检查数据处理与类型转换面向对象编程中的类型判断 三、isinstance函数…...
【基础知识】回头看Maven基础
版本日期修订人描述V1.02025/3/7nick huang创建文档 背景 项目过程中,对于Maven的pom.xml文件,很多时候,我通过各种参考、仿写,最终做出想要的效果。 但实际心里有些迷糊,不清楚具体哪个基础的配置所实现的效果。 今…...
练习题:81
目录 Python题目 题目 题目分析 需求理解 关键知识点 实现思路分析 代码实现 代码解释 运行思路 结束语 Python题目 题目 使用字典推导式创建一个字典,键为 1 到 10 的整数,值为键的平方。 题目分析 需求理解 本题要求使用 Python 的字典…...
三角函数:从宇宙法则到AI革命的数学密钥
——跨越三千年的数学语言与现代科技全景透视 一、数学本质:宇宙的波动密码 1.1 拓扑学视角下的三角函数 三角函数本质是单位圆上点的坐标参数化,其数学表达可抽象为: { x cos θ ℜ ( e i θ ) y sin θ ℑ ( e i θ ) \begin…...
【论文笔记】Best Practices and Lessons Learned on Synthetic Data for Language Models
论文信息 论文标题:Best Practices and Lessons Learned on Synthetic Data for Language Models 作者信息: Ruibo Liu, Jerry Wei, Fangyu Liu, Chenglei Si, Yanzhe Zhang, Jinmeng Rao, Steven Zheng, Daiyi Peng, Diyi Yang, Denny Zhou1 and Andre…...
Java高频面试之集合-10
hello啊,各位观众姥爷们!!!本baby今天来报道了!哈哈哈哈哈嗝🐶 面试官:详解红黑树?HashMap为什么不用二叉树/平衡树呢? 一、红黑树(Red-Black Treeÿ…...
Keil 5 环境下STM32F4 HAL库版本MDK工程创建详细步骤(适合小白,附工程源码)
一、前期准备 1.安装好keil Keil(MDK) 5 软件安装教程-CSDN博客https://blog.csdn.net/qq_42748213/article/details/90485750 2.安装好STM32F4的芯片包 Keil5中STM32F4xx芯片包下载安装_stm32f4芯片包-CSDN博客https://blog.csdn.net/weixin_45783141/article/details/131…...
【微服务】Nacos 配置动态刷新(简易版)(附配置)
文章目录 1、实现方法2、配置依赖 yaml3、验证效果 1、实现方法 环境:Nacos、Java、SpringBoot等 主要是在boostrap.yaml中的data-id属性下配置refresh:true来实现动态更新 2、配置依赖 yaml 具体的版本参考官方的说明:官方版本说明 <!--读取boo…...
LabVIEW cRIO中CSV文件的读取
在LabVIEW cRIO中读取CSV文件,需通过文件传输、路径配置、数据解析等步骤实现。本文详细说明如何通过代码读取本地存储的CSV文件,并探讨直接通过对话框选择文件的可行性及替代方案。 一、CSV文件传输至cRIO本地存储 1. 使用NI MAX文件管理 步骤…...
双周报Vol.67: 模式匹配支持守卫、LLVM 后端发布、支持 Attribute 语法...多项核心技术更新!
2025-03-10 语言更新 模式匹配支持守卫(Pattern Guard) 模式守卫可以通过在模式后追加 if ... 的语法结构来指定。有模式守卫的分支只有在被模式匹配的值满足对应模式,并且模式守卫为真的情况下才会执行。如果模式守卫为假,则会…...
从青铜到王者:六大排序算法实战解析
前言 在编程的世界里,排序算法如同一颗璀璨的明珠,闪耀着智慧的光芒。它不仅是计算机科学的基础知识点,更是每一位程序员必备的技能。今天,就让我们一同走进排序算法的世界,深入探究冒泡排序、选择排序、插入排序、快速排序、归并排序、堆排序这六大经典算法的精髓所在,…...
011-base64
base64 编码 以下是C实现的Base64字符串加密算法及其原理说明,综合了多个技术文档的核心要点: 一、Base64编码原理 Base64是一种将二进制数据转换为ASCII字符的编码方式,核心原理基于 3字节转4字符 的转换规则: 分组规则&…...
汽车NVH诊断案例 | 纯电车急加速过大弯底盘异响
引言 失去发动机的掩蔽效应后,新能源电车的NVH问题,成为了困扰维修技师新难点。风噪、胎噪、电机高频啸叫等问题更容易车主识别,根源却难以被有效分辨。如何更精准且高效地识别电车NVH问题根源?今天分享的这个案例,内…...
springcloud gateway通过数据库获取路由信息
在 Spring Cloud Gateway 中结合 MyBatis 动态从数据库加载路由配置,可以实现灵活的路由管理。以下是详细实现步骤: 1. 数据库表设计 创建路由配置表 gateway_route: CREATE TABLE gateway_route (id varchar(50) NOT NULL COMMENT 路由唯一…...
QtDataVisualization使用
Qt Data Visualization 是一个开源的第三方库,它为Qt框架提供了高级的数据可视化功能。这个库允许开发者创建复杂的3D和2D图表,包括但不限于散点图、曲面图、条形图等。它基于Qt 3D模块,因此可以充分利用Qt 3D引擎的强大功能来呈现三维数据。…...
【Go每日一练】实现简单的控制台计算器
👻创作者:丶重明 👻创作时间:2025年3月7日 👻擅长领域:运维 目录 1.😶🌫️题目:简单的控制台计算器2.😶🌫️代码输出3.😶&#…...
TDengine 数据对接 EXCEL
简介 通过配置使用 ODBC 连接器,Excel 可以快速访问 TDengine 的数据。用户可以将标签数据、原始时序数据或按时间聚合后的时序数据从 TDengine 导入到 Excel,用以制作报表整个过程不需要任何代码编写过程。 前置条件 准备以下环境: TDen…...
1.8 双指针专题:四数之和
1.题目链接 18. 四数之和 - 力扣(LeetCode)18. 四数之和 - 给你一个由 n 个整数组成的数组 nums ,和一个目标值 target 。请你找出并返回满足下述全部条件且不重复的四元组 [nums[a], nums[b], nums[c], nums[d]] (若两个四元组元…...
基于用户标签和协同过滤混合算法的商城推荐系统设计与实现
一、研究背景 随着电子商务的快速发展,用户面对海量商品时往往面临“信息过载”问题。传统的推荐算法(如协同过滤)在用户行为数据稀疏或新用户场景下存在冷启动、推荐多样性不足等缺陷。 现状与挑战: 协同过滤:依赖用…...
软件版本号设计
软件版本号的设计是软件开发中的重要环节,它不仅帮助开发团队管理代码,还能让用户清楚地了解软件的更新状态。以下是常见的版本号设计方法和最佳实践,供你参考: 1. 常见的版本号设计规范 语义化版本控制(Semantic Ver…...
ESMFold对决AlphaFold:蛋白质-肽相互作用预测的新进展
今天向大家介绍的这篇文章题目为:“Protein−Peptide Docking with ESMFold Language Model”,近期发表在JCTC上。 本文主要研究 ESMFold 语言模型在蛋白质-肽对接中的应用。通过探索多种对接策略,评估其在预测蛋白质-肽相互作用方面的性能&a…...
【项目】负载均衡式在线OJ
负载均衡式在线OJ 目录 负载均衡式在线OJ 1.项目介绍: 2.comm 2.1 log.hpp 日志等级 开放式日志 时间戳工具 2.2 util.hpp TimeUtil类 PathUtil类 FileUtil类 StringUtil类 3.Compile_server 3.1compile_run.hpp RemoveTempFile CodeToDesc Start 3.…...
Android启动速度优化
Android启动速度优化 一、应用启动基础知识 1.1 启动类型 Android应用的启动类型主要分为三种: 冷启动(Cold Start):应用进程不存在,系统需要创建新的进程,加载并启动应用。这是最耗时的启动方式。 温启动(Warm Start):应用进程存在,但Activity可能被销毁,需要重新创…...
python爬虫碰到IP被封的情况,如何解决?
在数据抓取和爬虫开发的实践中,Python作为一种功能强大且易于上手的编程语言,被广泛应用于网络数据的采集。然而,随着网络环境的日益复杂,爬虫活动也面临着越来越多的挑战,其中IP被封便是常见且棘手的问题。IP被封不仅…...
Web网页制作(静态网页):千年之恋
一、是用的PyCharm来写的代码 二、代码中所用到的知识点(无 js) 这段HTML代码展示了一个简单的注册页面,包含了多个HTML元素和CSS样式的应用。 这段HTML代码展示了一个典型的注册页面,包含了常见的HTML元素和表单控件。通过CSS样…...
mac安装mysql之后报错zsh: command not found: mysql !
在Mac上安装MySQL后,如果终端中找不到mysql命令,通常是 因为MySQL的命令行工具(如mysql客户端)没有被正确地添加到你的环境变量中。 检查 MySQL 是否已安装 ps -ef|grep mysql查看到路径在 /usr/local/mysql/bin 查看 .bash_pro…...
Spring Boot 启动失败:Failed to start bean ‘documentationPluginsBootstrapper’ 解决方案
文章目录 1. 问题描述 🎯2. 可能原因分析 🔍原因 1:SpringFox 版本与 Spring Boot 版本不兼容 ❌✅ 解决方案:添加兼容性配置(首选!!!!) 原因 2:S…...
Python Cookbook-3.16 查看汇率
任务 想周期性地(用 crontab 或者 Windows计划任务来运行某 Python 脚本)从 Web 获取数据,监视某两种货币之间的兑换比例,并在两者之间的汇率达到某个值时发送提醒邮件。 解决方案 这个任务和一系列的从 Web 获取数据的监控任务很类似,它们…...
Manus(一种AI代理或自动化工具)与DeepSeek(一种强大的语言模型或AI能力)结合使用任务自动化和智能决策
一、Manus与DeepSeek差异 十分好奇DeepSeek和Manus究竟谁更厉害些,DeepSeek是知识型大脑,Manus则是全能型执行者。即DeepSeek专注于语言处理、知识整合与专业文本生成。其核心优势在于海量参数支持的深度学习和知识推理能力,例如撰写论文、润…...
Redis存数据就像存钱:RDB定期存款 vs AOF实时记账
Redis持久化 ◆ 核心概念1. ◆ 持久化全景图2. ◆ 生产环境黄金法则 ◆ RDB深度优化1. ◆ 生产配置精要2. ◆ 高级触发场景3. ◆ 故障应急方案 ◆ AOF深度解析1. ◆ 7.0版本革命性改进2. ◆ 同步策略深度测试3. ◆ 重写过程优化 ◆ 混合持久化实战1. ◆ 配置示例2. ◆ 数据恢复…...
【从零开始学习计算机科学】编译原理(一)编译过程概述
【从零开始学习计算机科学】编译原理(一)编译过程概述 绪论编译过程概述词法分析语法分析代码优化代码生成其他功能编译器的前端和后端绪论 什么叫编译程序?为什么我们需要编译程序?编译程序就是一个程序,将便于人编写、阅读、维护的高级计算机语言所写作的源代码程序,翻…...
第十八:go 并发 goroutine
channel 可以让多个goroutine 之间实现通信 Add方法调用时机:必须在goroutine 启动之前调用Add方法来增加计数器的值。 如果在goroutine已经启动之后再调用Add,可能会导致Wait方法提前返回,因为计数器没有正确反映正在运行的goroutine的数量…...
基于QGIS的二次开发(四):矢量编辑与属性表操作
一、实验目的 本次实验续接上一次的实验内容,旨在通过设计与开发地理信息系统的过程,加深学生对地理信息系统的理解,并掌握相关的设计与开发技能,包括熟悉地理信息系统的设计与开发流程,加强对 MVC 软件设计模式的理解…...
AI日报 - 2025年3月13日
🌟 今日概览(60秒速览) ▎🤖 AGI突破 | Reka开源21B参数推理模型Flash 3,推出企业智能平台Nexus 🔬 模型采用RLOO方法结合模型与规则基础奖励,实现高效推理 ▎💼 商业动向 | Waymo在…...
lua C语言api学习1 编译第一个程序
本文开始进行lua C语言api的学习 1 简介 lua语言与C语言使用还是很紧密,以前我只是学习lua语言比较多,C语言api部分了解比较少,最近在学习tcc编译器的使用进一步学习一下lua C语言api的使用。 2 配置编译环境 首先需配置好tcc编译器环境[参考],再配置好lua源码路径[参考],新…...
【物联网-WIFI】
物联网-WIFI ■ ESP32-C3-模块简介■ ESP32-C3-■ ESP32-C3-■ WIFI-模组■ WIFI-■ WIFI- ■ ESP32-C3-模块简介 ■ ESP32-C3- ■ ESP32-C3- ■ WIFI-模组 ■ WIFI- ■ WIFI-...
在MATLAB中实现PID控制仿真
在MATLAB中实现PID控制仿真可以通过代码编程或Simulink图形化建模两种方式完成。以下是两种方法的详细操作步骤和示例: 方法1:使用MATLAB脚本编程(基于控制系统工具箱) 步骤1:定义被控对象的数学模型 假设被控对象是…...
C#实现本地Deepseek模型及其他模型的对话v1.4
前言 系 统:Window11 开发工具:Visual Studio 2022 相关技术:C# 、WPF .Net 8.0 1、C#实现本地AI聊天功能 WPFOllamaSharpe实现本地聊天功能,可以选择使用Deepseek 及其他模型。 新增根据聊天记录回复的功能。 优化了部分ViewModelÿ…...
用sphinx-doc整理文档#2
上一篇博客:用sphinx-doc整理文档 回头看,上一篇博客已经是18年的事情了。最近我又开始维护起18年的项目了。最近策划同事提了一些需求。我又改进了一波,所以有本文。 sphinx支持导出pdf sphinx本身是支持导出pdf的,命令如下&am…...
DBeaver部分操作指南(数据库连接,构造ERD图,格式化SQL)
详细步骤指导如何使用DBeaver来连接到数据库: 步骤 1: 下载并安装 DBeaver 如果还没有安装DBeaver,请访问DBeaver官网下载适合操作系统的版本,并按照指示完成安装。 步骤 2: 启动 DBeaver 安装完成后,启动DBeaver应用程序。 …...
十种处理权重矩阵的方法及数学公式
1. 权重归一化(Weight Normalization) 目的:通过分离权重向量的范数和方向来加速训练。公式:对于权重向量 w \mathbf{w} w,归一化后的权重 w ′ \mathbf{w} w′ 为: w ′ w ∥ w ∥ \mathbf{w} \frac{…...
姚安娜新剧瘦了一圈,《仁心俱乐部》急诊医生顾诗宜在线上岗
《仁心俱乐部》在芒果 TV 播出,湖南卫视金鹰独播剧场也随之播出,这一剧集受到了不少观众的关注。姚安娜在剧中饰演的急诊科医生顾诗宜,她为患者检查身体时动作娴熟,与患者沟通时展现出的耐心和专注,都展现出很高的专业…...
postgresql源码安装
步骤 1: 安装依赖 在开始之前,请确保您的系统上安装了编译 PostgreSQL 所需的依赖包。使用以下命令安装必要的软件包: 对于 Debian/Ubuntu 系统: sudo apt update sudo apt install build-essential libreadline-dev zlib1g-dev flex biso…...
【51单片机】程序实验15.DS18B20温度传感器
主要参考学习资料:B站【普中官方】51单片机手把手教学视频 开发资料下载链接:http://www.prechin.cn/gongsixinwen/208.html 单片机套装:普中STC51单片机开发板A4标准版套餐7 目录 DS18B20介绍主要特性内部结构控制时序初始化时序写时序读时序…...