Python战舰游戏开发:核心循环与命中实现
2025-09-12 10:11:14
0浏览
收藏
IT行业相对于一般传统行业,发展更新速度更快,一旦停止了学习,很快就会被行业所淘汰。所以我们需要踏踏实实的不断学习,精进自己的技术,尤其是初学者。今天golang学习网给大家整理了《Python战舰游戏开发:核心循环与命中逻辑实现》,聊聊,我们一起来看看吧!

1. 游戏基础结构回顾
在构建战舰游戏时,首先需要定义一些基本元素,如地图大小、船只类型以及用于显示和操作游戏板的函数。原始代码已经提供了这些基础:
- ship_initial 和 ship_names: 定义了船只的缩写和全称。
- get_username(): 获取玩家名称。
- create_battlefield(map_size): 创建一个指定大小的空地图。
- display_battlefield(board): 打印地图的当前状态。
- player_ship_coordinate(player_board, occupied): 引导玩家放置船只。
- comp_ship_coordinate(comp_board): 随机放置电脑的船只。
这些函数构成了游戏的骨架,但在实现完整的游戏流程(即玩家和电脑轮流攻击直到一方所有船只被击沉)时,还需要进一步完善。
2. 核心挑战:游戏循环与命中判定
原始代码的主要挑战在于缺乏一个持续的游戏循环来管理回合制攻击,以及有效的机制来跟踪船只的命中状态和判断游戏结束。为了解决这些问题,我们需要引入以下概念:
- 虚拟板 (Dummy Board): 玩家在攻击电脑时,不应看到电脑船只的真实位置。虚拟板作为玩家视角下的电脑地图,只显示玩家攻击的结果(命中或未命中)。
- 命中计数器 (Hit Counters): 分别为玩家和电脑维护一个计数器,当击中对方一艘船时,计数器增加。达到预设值(例如,所有船只数量)时,游戏结束。
- 主游戏循环 (Main Game Loop): 一个 while True 循环,用于交替执行玩家和电脑的回合,直到满足游戏结束条件。
3. 改进的命中判定函数
为了支持上述逻辑,我们需要修改 check_player_hit 和 check_comp_hit 函数,并引入一个虚拟板。
3.1 玩家攻击判定 (check_player_hit)
此函数将接收电脑的真实船只板 (comp_board) 和玩家视角的虚拟板 (dummy_board)。
def check_player_hit(comp_board, dummy_board, user):
"""
函数用于玩家攻击敌方船只的命中或未命中判定。
同时更新玩家视角的虚拟板。
"""
print(f"\n{user},请选择攻击坐标:")
while True:
try:
row = int(input("输入行 (0-9): "))
col = int(input("输入列 (0-9): "))
if 0 <= row < map_size and 0 <= col < map_size:
# 检查是否重复攻击
if dummy_board[row][col] != '_' and dummy_board[row][col] != '*':
print("你已经攻击过这个位置了,请选择其他位置。")
continue
break
else:
print("无效坐标,请重新输入。")
except ValueError:
print("输入无效,请输入整数。")
hit = 0 # 默认未命中
if comp_board[row][col] in ship_initial: # 如果是船只字母
# 击中船只,将电脑板上的船只标记为小写表示已击中
# 虚拟板上标记为'X'表示命中
original_ship_char = comp_board[row][col]
comp_board[row][col] = original_ship_char.lower()
dummy_board[row][col] = "X"
print(f"电脑:{ship_names[ship_initial.index(original_ship_char)]} 被击中了!")
hit = 1 # 命中
else:
dummy_board[row][col] = "*" # 未命中,虚拟板上标记为'*'
print("未命中!")
return hit关键改进点:
- 参数增加 dummy_board: 用于更新玩家对电脑板的视图。
- 命中标记: 电脑板上的船只被击中后,其字符变为小写(例如 'B' -> 'b'),这样可以区分未被击中的船只和已被击中的船只。
- 虚拟板更新: 命中时 dummy_board[row][col] 变为 'X',未命中时变为 '*'。
- 返回命中状态: 函数返回 1 表示命中,0 表示未命中,以便更新总命中计数。
- 输入验证: 增加了对重复攻击同一位置的检查。
3.2 电脑攻击判定 (check_comp_hit)
电脑的攻击需要随机性,并且不能重复攻击已经攻击过的位置。
def check_comp_hit(player_board):
"""
函数用于电脑攻击玩家船只的命中或未命中判定。
"""
hit = 0 # 默认未命中
while True: # 随机选择一个尚未攻击过的位置
row = randrange(0, map_size)
col = randrange(0, map_size)
# 检查该位置是否已被电脑攻击过(标记为小写船只或'*')
if player_board[row][col] not in ['*', 'a', 'b', 'c', 'f', 's']:
break
print(f"电脑选择了坐标 ({row}, {col}) 进行攻击。")
if player_board[row][col] in ship_initial: # 如果是船只字母
original_ship_char = player_board[row][col]
player_board[row][col] = original_ship_char.lower() # 标记为小写表示已击中
print(f"玩家:你的 {ship_names[ship_initial.index(original_ship_char)]} 被击中了!")
hit = 1 # 命中
else:
player_board[row][col] = "*" # 未命中
print("电脑未命中!")
return hit关键改进点:
- 随机不重复攻击: while True 循环确保电脑选择一个尚未被攻击过的有效位置。
- 命中标记: 玩家板上的船只被击中后,同样将其字符变为小写。
- 返回命中状态: 函数返回 1 表示命中,0 表示未命中。
4. 构建主游戏循环
现在,我们可以将所有组件整合到 if __name__ == "__main__": 块中,构建游戏的完整流程。
from random import randrange
ship_initial = ["B", "C", "F", "A", "S"]
ship_names = ["Battleship", "Cruiser", "Frigate", "Aircraft Carrier", "Submarine"] # 修正Sub为Submarine
map_size = 10
def get_username():
"""
获取用户名的函数。
"""
while True:
user_name = input("请输入你的名字: ")
if user_name:
print(f"欢迎来到战舰游戏,{user_name}!")
return user_name
else:
print("请不要留空,请输入你的名字。")
def create_battlefield(map_size):
"""
根据指定大小创建地图的函数。
"""
return [["_"] * map_size for _ in range(map_size)]
def display_battlefield(board, title=""):
"""
显示当前地图状态的函数。
"""
if title:
print(f"\n--- {title} ---")
# 打印列索引
print(" " + " ".join(str(i) for i in range(map_size)))
for i, row in enumerate(board):
print(f"{i} " + " ".join(row))
def player_ship_coordinate(player_board, occupied):
"""
玩家放置船只的函数。
"""
print("\n请放置你的船只。")
for i, ship_name in enumerate(ship_names):
while True:
try:
row = int(input(f"请输入 {ship_name} 的行坐标 (0-9): "))
col = int(input(f"请输入 {ship_name} 的列坐标 (0-9): "))
if 0 <= row < map_size and 0 <= col < map_size and (row, col) not in occupied:
player_board[row][col] = ship_initial[i]
occupied.add((row, col))
break
else:
print("无效坐标或该位置已被占用。请输入正确的值。")
except ValueError:
print("输入无效。请输入一个有效的整数。")
return player_board, occupied
def comp_ship_coordinate(comp_board):
"""
电脑放置船只的函数。
"""
for ship in ship_initial:
while True:
row = randrange(0, map_size)
col = randrange(0, map_size)
if comp_board[row][col] == "_":
comp_board[row][col] = ship
break
return comp_board
def check_player_hit(comp_board, dummy_board, user):
"""
函数用于玩家攻击敌方船只的命中或未命中判定。
同时更新玩家视角的虚拟板。
"""
print(f"\n{user},请选择攻击坐标:")
while True:
try:
row = int(input("输入行 (0-9): "))
col = int(input("输入列 (0-9): "))
if 0 <= row < map_size and 0 <= col < map_size:
if dummy_board[row][col] != '_': # 检查是否重复攻击
print("你已经攻击过这个位置了,请选择其他位置。")
continue
break
else:
print("无效坐标,请重新输入。")
except ValueError:
print("输入无效,请输入整数。")
hit = 0
if comp_board[row][col] in ship_initial:
original_ship_char = comp_board[row][col]
comp_board[row][col] = original_ship_char.lower()
dummy_board[row][col] = "X"
print(f"电脑:{ship_names[ship_initial.index(original_ship_char)]} 被击中了!")
hit = 1
else:
dummy_board[row][col] = "*"
print("未命中!")
return hit
def check_comp_hit(player_board):
"""
函数用于电脑攻击玩家船只的命中或未命中判定。
"""
hit = 0
while True:
row = randrange(0, map_size)
col = randrange(0, map_size)
# 检查该位置是否已被电脑攻击过(标记为小写船只或'*')
if player_board[row][col] not in ['*', 'a', 'b', 'c', 'f', 's']:
break
print(f"\n电脑选择了坐标 ({row}, {col}) 进行攻击。")
if player_board[row][col] in ship_initial:
original_ship_char = player_board[row][col]
player_board[row][col] = original_ship_char.lower()
print(f"玩家:你的 {ship_names[ship_initial.index(original_ship_char)]} 被击中了!")
hit = 1
else:
player_board[row][col] = "*"
print("电脑未命中!")
return hit
if __name__ == "__main__":
user = get_username()
player_board = create_battlefield(map_size)
comp_board = create_battlefield(map_size)
dummy_board = create_battlefield(map_size) # 玩家视角下的电脑板
occupied = set()
# 玩家放置船只
player_ship_coordinate(player_board, occupied)
display_battlefield(player_board, f"{user} 的船只部署")
# 电脑放置船只
comp_ship_coordinate(comp_board)
# display_battlefield(comp_board, "电脑的船只部署 (仅供测试)") # 实际游戏中不显示
display_battlefield(dummy_board, f"{user} 的攻击目标板") # 显示玩家的攻击目标板
player_hits = 0 # 玩家击中电脑船只的数量
comp_hits = 0 # 电脑击中玩家船只的数量
total_ships = len(ship_initial) # 总船只数量
print("\n--- 游戏开始! ---")
while True:
# 玩家回合
print(f"\n--- {user} 的回合 ---")
display_battlefield(dummy_board, f"{user} 的攻击目标板")
player_hits += check_player_hit(comp_board, dummy_board, user)
if player_hits == total_ships:
print(f"\n恭喜你,{user}!你击沉了电脑所有船只,赢得了游戏!")
break
# 电脑回合
print("\n--- 电脑的回合 ---")
comp_hits += check_comp_hit(player_board)
display_battlefield(player_board, f"{user} 的船只板") # 显示玩家自己的板,查看电脑攻击结果
if comp_hits == total_ships:
print("\n很遗憾,电脑击沉了你所有船只,电脑赢得了游戏!")
break
# 每回合结束显示双方板状态(玩家只看自己的船只板和攻击目标板)
display_battlefield(player_board, f"{user} 的船只板")
display_battlefield(dummy_board, f"{user} 的攻击目标板")5. 运行与测试
运行上述代码,你将体验到一个完整的战舰游戏流程:
- 输入你的名字。
- 在提示下输入船只的行和列坐标来部署你的船只。
- 电脑会自动部署其船只。
- 游戏进入循环,你和电脑轮流输入攻击坐标。
- 每次攻击后,会显示命中或未命中信息,并更新相应的地图。
- 当一方的所有船只都被击沉时,游戏结束,并宣布获胜者。
6. 注意事项与潜在改进
- 船只大小与方向: 当前实现中,每艘船只占用一个格子。更真实的战舰游戏会包含不同长度的船只,并允许玩家选择水平或垂直放置。这需要更复杂的船只放置逻辑和命中判定。
- 输入验证: 尽管已添加了基本的坐标和重复攻击验证,但可以进一步增强,例如防止输入非数字字符时程序崩溃。
- 游戏界面: 这是一个命令行游戏。可以通过使用库(如 pygame 或 curses)来创建更具吸引力的图形用户界面。
- 电脑AI: 当前电脑只是随机攻击未命中/未被击中的位置。可以实现更复杂的AI策略,例如“搜索-摧毁”模式(当击中一艘船后,在其周围格子进行搜索)。
- 船只沉没判定: 当前游戏只计数总命中次数。更精细的实现会跟踪每艘船的命中情况,当一艘船的所有部分都被击中时,宣布该船沉没。
通过本教程,你已经掌握了构建一个基本战舰游戏的核心逻辑,包括游戏循环、命中判定和游戏状态管理。在此基础上,你可以继续探索更高级的功能和改进,使你的游戏更加完善和有趣。
到这里,我们也就讲完了《Python战舰游戏开发:核心循环与命中实现》的内容了。个人认为,基础知识的学习和巩固,是为了更好的将其运用到项目中,欢迎关注golang学习网公众号,带你了解更多关于的知识点!
Excel边框怎么去掉?简单删除方法
- 上一篇
- Excel边框怎么去掉?简单删除方法
- 下一篇
- Node.js对象合并内存优化方法
查看更多
最新文章
-
- 文章 · python教程 | 40分钟前 | sdk 邮件API requests库 smtplib Python邮件发送
- Python发送邮件API调用方法详解
- 165浏览 收藏
-
- 文章 · python教程 | 47分钟前 |
- Pandasmerge_asof快速匹配最近时间数据
- 254浏览 收藏
-
- 文章 · python教程 | 1小时前 |
- 列表推导式与生成器表达式区别解析
- 427浏览 收藏
-
- 文章 · python教程 | 1小时前 |
- Pythonopen函数使用技巧详解
- 149浏览 收藏
-
- 文章 · python教程 | 1小时前 |
- Python合并多个列表的几种方法
- 190浏览 收藏
-
- 文章 · python教程 | 1小时前 |
- Python嵌套if语句使用方法详解
- 264浏览 收藏
-
- 文章 · python教程 | 1小时前 |
- Python队列判空安全方法详解
- 293浏览 收藏
-
- 文章 · python教程 | 1小时前 |
- RuffFormatter尾随逗号设置方法
- 450浏览 收藏
-
- 文章 · python教程 | 2小时前 |
- Python读取二进制文件的缓冲方法
- 354浏览 收藏
-
- 文章 · python教程 | 3小时前 | Python 数据结构 namedtuple 扑克牌 Card
- Pythonnamedtuple打造扑克牌玩法详解
- 291浏览 收藏
-
- 文章 · python教程 | 3小时前 |
- PythonIQR方法检测异常值详解
- 478浏览 收藏
查看更多
课程推荐
-
- 前端进阶之JavaScript设计模式
- 设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
- 543次学习
-
- GO语言核心编程课程
- 本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
- 516次学习
-
- 简单聊聊mysql8与网络通信
- 如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
- 500次学习
-
- JavaScript正则表达式基础与实战
- 在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
- 487次学习
-
- 从零制作响应式网站—Grid布局
- 本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
- 485次学习
查看更多
AI推荐
-
- ChatExcel酷表
- ChatExcel酷表是由北京大学团队打造的Excel聊天机器人,用自然语言操控表格,简化数据处理,告别繁琐操作,提升工作效率!适用于学生、上班族及政府人员。
- 3186次使用
-
- Any绘本
- 探索Any绘本(anypicturebook.com/zh),一款开源免费的AI绘本创作工具,基于Google Gemini与Flux AI模型,让您轻松创作个性化绘本。适用于家庭、教育、创作等多种场景,零门槛,高自由度,技术透明,本地可控。
- 3398次使用
-
- 可赞AI
- 可赞AI,AI驱动的办公可视化智能工具,助您轻松实现文本与可视化元素高效转化。无论是智能文档生成、多格式文本解析,还是一键生成专业图表、脑图、知识卡片,可赞AI都能让信息处理更清晰高效。覆盖数据汇报、会议纪要、内容营销等全场景,大幅提升办公效率,降低专业门槛,是您提升工作效率的得力助手。
- 3429次使用
-
- 星月写作
- 星月写作是国内首款聚焦中文网络小说创作的AI辅助工具,解决网文作者从构思到变现的全流程痛点。AI扫榜、专属模板、全链路适配,助力新人快速上手,资深作者效率倍增。
- 4535次使用
-
- MagicLight
- MagicLight.ai是全球首款叙事驱动型AI动画视频创作平台,专注于解决从故事想法到完整动画的全流程痛点。它通过自研AI模型,保障角色、风格、场景高度一致性,让零动画经验者也能高效产出专业级叙事内容。广泛适用于独立创作者、动画工作室、教育机构及企业营销,助您轻松实现创意落地与商业化。
- 3807次使用
查看更多
相关文章
-
- Flask框架安装技巧:让你的开发更高效
- 2024-01-03 501浏览
-
- Django框架中的并发处理技巧
- 2024-01-22 501浏览
-
- 提升Python包下载速度的方法——正确配置pip的国内源
- 2024-01-17 501浏览
-
- Python与C++:哪个编程语言更适合初学者?
- 2024-03-25 501浏览
-
- 品牌建设技巧
- 2024-04-06 501浏览

