本教程详细介绍了如何在 Python 中使用嵌套列表构建 2D 游戏地图,并实现以玩家为中心的局部视图渲染系统。文章涵盖了地图数据结构、环境元素表示、视口计算、地图初始化与边界处理,并提供了示例代码,帮助开发者在终端环境中高效地展示游戏世界。
在 Python 中,通常使用“列表的列表”(nested lists)来模拟二维数组或矩阵,这非常适合表示游戏地图。每个内部列表代表地图的一行,而列表中的元素则代表地图上的一个瓦片(tile)。
我们可以用整数值来代表不同的环境元素。例如:
为了在终端中显示这些元素,我们需要一个映射表,将整数值转换为可打印的字符。
# 示例地图数据
game_map = [
[1, 1, 1, 1, 1, 1, 1, 1, 1, 1],
[1, 0, 0, 0, 0, 0, 0, 0, 0, 1],
[1, 0, 0, 0, 0, 0, 0, 0, 0, 1],
[1, 0, 0, 0, 0, 0, 0, 0, 0, 1],
[1, 0, 0, 0, 0, 0, 0, 0, 0, 1],
[1, 0, 0, 0, 0, 0, 0, 0, 0, 1],
[1, 0, 0, 0, 0, 0, 0, 0, 0, 1],
[1, 1, 1, 1, 1, 1, 1, 1, 1, 1],
]
# 元素到字符的映射
tile_textures = {
0: ' ', # 空地
1: '#', # 墙壁
2: 'P', # 玩家
3: 'X', # 道具
-1: '.', # 边界外的“无”区域
}为了只渲染玩家周围的区域,我们需要定义一个“视口”(viewport),即屏幕上可见的区域大小。玩家通常位于视口的中心。
a. 定义视口大小与玩家位置
假设我们的终端显示区域(视口)有固定的宽度和高度。玩家的坐标是 (player_x, player_y)。
VIEWPORT_WIDTH = 15 VIEWPORT_HEIGHT = 7 player_x = 5 player_y = 3
b. 计算视口边界
视口的左上角和右下角坐标需要根据玩家位置和视口大小来计算。
def calculate_viewport_bounds(player_x, player_y, viewport_width, viewport_height, map_width, map_height):
# 计算视口左上角坐标
view_left = player_x - viewport_width // 2
view_top = player_y - viewport_height // 2
# 调整视口,确保它不会超出地图边界
# 如果视口左边超出地图左边,则向右平移
if view_left < 0:
view_left = 0
# 如果视口右边超出地图右边,则向左平移
elif view_left + viewport_width > map_width:
view_left = map_width - viewport_width
# 如果视口上边超出地图上边,则向下平移
if view_top < 0:
view_top = 0
# 如果视口下边超出地图下边,则向上平移
elif view_top + viewport_height > map_height:
view_top = map_height - viewport_height
# 确保视口至少为0
view_left = max(0, view_left)
view_top = max(0, view_top)
# 计算视口右下角坐标
view_right = view_left + viewport_width
view_bottom = view_top + viewport_height
return view_left, view_top, view_right, view_bottomc. 地图初始化与边界填充
为了简化渲染逻辑,尤其是当玩家靠近地图边缘时,我们可以将实际地图嵌入到一个更大的、用“无”区域(例如,用 -1 表示)填充的画布中。这样,即使玩家的视口超出了实际地图的范围,我们也能从这个更大的画布中获取到值,避免索引越界错误。
def create_padded_map(original_map, pad_value=-1):
map_height = len(original_map)
map_width = len(original_map[0]) if map_height > 0 else 0
# 确定需要的额外填充量(通常是视口的一半)
# 这里我们简化,直接在地图四周添加一圈pad_value
# 更精确的做法是根据VIEWPORT_WIDTH/HEIGHT来确定
padding = max(VIEWPORT_WIDTH // 2, VIEWPORT_HEIGHT // 2) + 1
padded_height = map_height + 2 * padding
padded_width = map_width + 2 * padding
padded_map = [[pad_value for _ in range(padded_width)] for _ in range(padded_height)]
# 将原始地图复制到填充地图的中心
for y in range(map_height):
for x in range(map_width):
padded_map[y + padding][x + padding] = original_map[y][x]
return padded_map, padding
# 假设原始地图和填充后的地图以及偏移量
# padded_game_map, map_offset = create_padded_map(game_map)
# player_x_padded = player_x + map_offset
# player_y_padded = player_y + map_offset然而,对于终端渲染,更常见且简单的做法是直接在渲染循环中检查边界,而不是预先填充一个巨大的地图。上述 calculate_viewport_bounds 函数已经处理了视口不会超出实际地图边界的情况。如果需要渲染“地图外”区域,则在渲染时判断坐标是否有效,无效则显示默认的“无”字符。
d. 渲染逻辑
遍历视口内的每个单元格,获取其在地图中的值,并根据 tile_textures 字典打印相应的字符。
def render_viewport(game_map, player_x, player_y, viewport_width, viewport_height, tile_textures):
map_height = len(game_map)
map_width = len(game_map[0])
# 计算视口左上角在地图上的起始坐标
start_col = player_x - viewport_width // 2
start_row = player_y - viewport_height // 2
output_lines = []
for r in range(viewport_height):
current_row_chars = []
for c in range(viewport_width):
map_row = start_row + r
map_col = start_col + c
# 检查是否是玩家当前位置
if map_row == player_y and map_col == player_x:
current_row_chars.append(tile_textures[2]) # 玩家纹理
# 检查坐标是否在地图范围内
elif 0 <= map_row < map_height and 0 <= map_col < map_width:
tile_value = game_map[map_row][map_col]
current_row_chars.append(tile_textures.get(tile_value, '?')) # 获取瓦片纹理
else:
# 超出地图范围,显示“无”区域纹理
current_row_chars.append(tile_textures.get(-1, ' '))
output_lines.append("".join(current_row_chars))
# 打印渲染结果
print("\n".join(output_lines))玩家移动时,需要更新 player_x 和 player_y。同时,必须确保玩家不能移动到无效区域(例如墙壁或地图边界外)。
def move_player(game_map, player_x, player_y, dx, dy):
map_height = len(game_map)
map_width = len(game_map[0])
new_x = player_x + dx
new_y = player_y + dy
# 检查新位置是否在地图范围内
if 0 <= new_x < map_width and 0 <= new_y < map_height:
# 检查新位置是否可通行(例如,不是墙壁)
if game_map[new_y][new_x] != 1: # 假设1是墙壁
return new_x, new_y
return player_x,
player_y # 无法移动,返回原位置结合上述组件,我们可以构建一个简单的终端游戏循环:
import os
import time
# 游戏地图数据
game_map = [
[1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1],
[1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 1],
[1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 1],
[1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 1],
[1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 1],
[1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 1],
[1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 1],
[1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 1],
[1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 1],
[1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1],
]
# 元素到字符的映射
tile_textures = {
0: ' ', # 空地
1: '#', # 墙壁
2: 'P', # 玩家
3: 'X', # 道具
-1: '.', # 边界外的“无”区域
}
# 视口大小
VIEWPORT_WIDTH = 21 # 奇数方便居中
VIEWPORT_HEIGHT = 11 # 奇数方便居中
# 玩家初始位置
player_x = 1
player_y = 1
def clear_terminal():
os.system('cls' if os.name == 'nt' else 'clear')
def render_viewport(game_map, player_x, player_y, viewport_width, viewport_height, tile_textures):
map_height = len(game_map)
map_width = len(game_map[0])
# 计算视口左上角在地图上的起始坐标
start_col = player_x - viewport_width // 2
start_row = player_y - viewport_height // 2
output_lines = []
for r in range(viewport_height):
current_row_chars = []
for c in range(viewport_width):
map_row = start_row + r
map_col = start_col + c
# 检查是否是玩家当前位置
if map_row == player_y and map_col == player_x:
current_row_chars.append(tile_textures[2]) # 玩家纹理
# 检查坐标是否在地图范围内
elif 0 <= map_row < map_height and 0 <= map_col < map_width:
tile_value = game_map[map_row][map_col]
current_row_chars.append(tile_textures.get(tile_value, '?')) # 获取瓦片纹理
else:
# 超出地图范围,显示“无”区域纹理
current_row_chars.append(tile_textures.get(-1, ' '))
output_lines.append("".join(current_row_chars))
# 打印渲染结果
print("\n".join(output_lines))
def move_player(game_map, player_x, player_y, dx, dy):
map_height = len(game_map)
map_width = len(game_map[0])
new_x = player_x + dx
new_y = player_y + dy
# 检查新位置是否在地图范围内
if 0 <= new_x < map_width and 0 <= new_y < map_height:
# 检查新位置是否可通行(例如,不是墙壁)
if game_map[new_y][new_x] != 1: # 假设1是墙壁
return new_x, new_y
return player_x, player_y # 无法移动,返回原位置
# 游戏主循环
def game_loop():
global player_x, player_y # 允许修改全局玩家位置
# 模拟简单的输入(例如,每秒向右移动一次)
# 实际游戏中会监听键盘输入
moves = [(1, 0), (0, 1), (-1, 0), (0, -1)] # 右, 下, 左, 上
move_index = 0
running = True
while running:
clear_terminal()
render_viewport(game_map, player_x, player_y, VIEWPORT_WIDTH, VIEWPORT_HEIGHT, tile_textures)
print(f"Player Pos: ({player_x}, {player_y})")
print("Press Ctrl+C to exit. (Simulating movement every second)")
# 模拟移动
dx, dy = moves[move_index % len(moves)]
player_x, player_y = move_player(game_map, player_x, player_y, dx, dy)
move_index += 1
time.sleep(0.5) # 暂停0.5秒
# 运行游戏
if __name__ == "__main__":
try:
game_loop()
except KeyboardInterrupt:
print("\nGame Over!")
通过本文的指导,您应该能够理解并实现一个基本的 Python 2D 游戏地图系统,并掌握以玩家为中心的局部渲染技术,为构建更复杂的终端游戏打下基础。
# python
# windows
# 计算机
# app
# ai
# unix
# win
# 键值对
# 端游
# elif
相关文章:
导航网站建站方案与优化指南:一站式高效搭建技巧解析
打鱼网站制作软件,波克捕鱼官方号怎么注册?
如何快速搭建二级域名独立网站?
c++怎么实现高并发下的无锁队列_c++ std::atomic原子变量与CAS操作【详解】
学校建站服务器如何选型才能满足性能需求?
如何通过免费商城建站系统源码自定义网站主题与功能?
如何在Windows虚拟主机上快速搭建网站?
西安大型网站制作公司,西安招聘网站最好的是哪个?
建站之星安装路径如何正确选择及配置?
如何通过建站之星自助学习解决操作问题?
建站之星安装后如何配置SEO及设计样式?
如何通过商城自助建站源码实现零基础高效建站?
如何快速搭建安全的FTP站点?
无锡营销型网站制作公司,无锡网选车牌流程?
网站制作话术技巧,网站推广做的好怎么话术?
详解jQuery中基本的动画方法
已有域名和空间如何搭建网站?
高端企业智能建站程序:SEO优化与响应式模板定制开发
如何在宝塔面板中创建新站点?
网站制作公司哪里好做,成都网站制作公司哪家做得比较好,更正规?
云南网站制作公司有哪些,云南最好的招聘网站是哪个?
小捣蛋自助建站系统:数据分析与安全设置双核驱动网站优化
电商网站制作价格怎么算,网上拍卖流程以及规则?
网站设计制作企业有哪些,抖音官网主页怎么设置?
在线ppt制作网站有哪些,请推荐几个好的课件下载的网站?
天河区网站制作公司,广州天河区如何办理身份证?需要什么资料有预约的网站吗?
建站之星2.7模板快速切换与批量管理功能操作指南
如何高效生成建站之星成品网站源码?
网站制作难吗安全吗,做一个网站需要多久时间?
如何配置IIS站点权限与局域网访问?
如何在Golang中实现微服务服务拆分_Golang微服务拆分与接口管理方法
香港服务器网站生成指南:免费资源整合与高速稳定配置方案
高端建站三要素:定制模板、企业官网与响应式设计优化
Dapper的Execute方法的返回值是什么意思 Dapper Execute返回值详解
网站制作外包价格怎么算,招聘网站上写的“外包”是什么意思?
标准网站视频模板制作软件,现在有哪个网站的视频编辑素材最齐全的,背景音乐、音效等?
视频网站app制作软件,有什么好的视频聊天网站或者软件?
专业商城网站制作公司有哪些,pi商城官网是哪个?
为什么Go需要go mod文件_Go go mod文件作用说明
如何在阿里云完成域名注册与建站?
如何彻底卸载建站之星软件?
如何快速选择适合个人网站的云服务器配置?
阿里云网站制作公司,阿里云快速搭建网站好用吗?
C#怎么创建控制台应用 C# Console App项目创建方法
建站主机如何选?性能与价格怎样平衡?
北京网站制作网页,网站升级改版需要多久?
自助网站制作软件,个人如何自助建网站?
Python文件管理规范_工程实践说明【指导】
网站制作哪家好,cc、.co、.cm哪个域名更适合做网站?
如何挑选优质建站一级代理提升网站排名?
*请认真填写需求信息,我们会在24小时内与您取得联系。