
本教程详细介绍了如何在 Python 中使用嵌套列表构建 2D 游戏地图,并实现以玩家为中心的局部视图渲染系统。文章涵盖了地图数据结构、环境元素表示、视口计算、地图初始化与边界处理,并提供了示例代码,帮助开发者在终端环境中高效地展示游戏世界。
在 Python 中,通常使用“列表的列表”(nested lists)来模拟二维数组或矩阵,这非常适合表示游戏地图。每个内部列表代表地图的一行,而列表中的元素则代表地图上的一个瓦片(tile)。
我们可以用整数值来代表不同的环境元素。例如:
为了在终端中显示这些元素,我们需要一个映射表,将整数值转换为可打印的字符。
立即学习“Python免费学习笔记(深入)”;
# 示例地图数据
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 2D 游戏地图与局部渲染教程的详细内容,更多请关注php中文网其它相关文章!
每个人都需要一台速度更快、更稳定的 PC。随着时间的推移,垃圾文件、旧注册表数据和不必要的后台进程会占用资源并降低性能。幸运的是,许多工具可以让 Windows 保持平稳运行。
Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号