新闻中心
Python 2D 游戏地图与局部渲染教程

本教程详细介绍了如何在 Python 中使用嵌套列表构建 2D 游戏地图,并实现以玩家为中心的局部视图渲染系统。文章涵盖了地图数据结构、环境元素表示、视口计算、地图初始化与边界处理,并提供了示例代码,帮助开发者在终端环境中高效地展示游戏世界。
构建 Python 2D 游戏地图
在 Python 中,通常使用“列表的列表”(nested lists)来模拟二维数组或矩阵,这非常适合表示游戏地图。每个内部列表代表地图的一行,而列表中的元素则代表地图上的一个瓦片(tile)。
1. 地图数据结构与元素表示
我们可以用整数值来代表不同的环境元素。例如:
- 0:空地
- 1:墙壁
- 2:玩家
- 3:道具
为了在终端中显示这些元素,我们需要一个映射表,将整数值转换为可打印的字符。
# 示例地图数据
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: '.', # 边界外的“无”区域
}2. 实现以玩家为中心的局部渲染(视口系统)
为了只渲染玩家周围的区域,我们需要定义一个“视口”(viewport),即屏幕上可见的区域大小。玩家通常位于视口的中心。
a. 定义视口大小与玩家位置
假设我们的终端显示区域(视口)有固定的宽度和高度。玩家的坐标是 (player_x, player_y)。
VIEWPORT_WIDTH = 15 VIEWPORT_HEIGHT = 7 player_x = 5 player_y = 3
b. 计算视口边界
视口的左上角和右下角坐标需要根据玩家位置和视口大小来计算。
GemDesign
AI高保真原型设计工具
652
查看详情
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(v
iewport_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))3. 玩家移动与边界检查
玩家移动时,需要更新 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!")
注意事项与总结
- 性能优化: 对于大型地图,每次渲染都遍历整个视口是可行的。如果地图和视口都非常巨大,可以考虑更高级的渲染技术,但对于终端游戏,当前方法已足够。
- 输入处理: 上述示例使用简单的 time.sleep 模拟移动。实际游戏中,需要使用 curses (Unix-like) 或 msvcrt (Windows) 等库来捕获实时键盘输入,实现交互式移动。
- 地图设计: 地图的边界处理至关重要。本教程中的 render_viewport 函数通过检查 0
- 可扩展性: tile_textures 字典使得添加新的环境元素变得非常容易,只需增加新的键值对即可。
- 坐标系: 在大多数2D游戏和计算机图形中,通常约定 (0,0) 为左上角,X轴向右增加,Y轴向下增加。本教程也遵循此约定。
通过本文的指导,您应该能够理解并实现一个基本的 Python 2D 游戏地图系统,并掌握以玩家为中心的局部渲染技术,为构建更复杂的终端游戏打下基础。
以上就是Python 2D 游戏地图与局部渲染教程的详细内容,更多请关注其它相关文章!
# 更大
# 广西本地网络推广营销
# 杨浦抖音营销推广地址
# 桂林网站优化外包
# 宁波做网站seo费用
# 开发区关键词排名如何调整
# seo企业站排名
# 抖音卖粉网站排名优化
# 5000元撬动营销推广
# 8seo网站改了
# 短信seo
# 如何用
# 多线程
# 重启
# 我们可以
# python
# 遍历
# 图上
# 键值
# 在地
# 数据结构
# elif
# 端游
# 键值对
# win
# unix
# ai
# app
# 计算机
# windows
相关栏目:
【
科技资讯46185 】
【
网络学院92790 】
相关推荐:
Golang如何实现状态模式管理对象状态_Golang State模式实现技巧
电脑屏幕颜色不舒服怎么办_Windows夜间模式与色彩校准教程【护眼技巧】
如何在低配置电脑上搭建轻量级J*a环境_占用更小的环境选择技巧
蛙漫画网页版全站入口 蛙漫热门作品免费浏览
Web Components中自定义开关组件状态同步的常见陷阱与解决方案
QQ邮箱网页版入口页面 QQ邮箱在线登录入口官网
poki免费入口快捷访问 poki人气小游戏直接玩站点
百度浏览器字体显示异常偏小_百度浏览器字体渲染修复方案
qq邮箱日历功能怎么用_创建日程与会议邀请的技巧
AI泡沫首次被“刺破”:GPU十年都无法存活!
构建轻量级网站内部消息系统:Formspree 集成指南
J*a最大堆Heapify方法修复:索引计算与边界条件深度解析
C++ map遍历方法大全_C++ map迭代器使用总结
mysql通配符支持数字匹配吗_mysql通配符能否用于数字匹配的解析
魅族20怎样在浏览器开无图省流_iPhone魅族20浏览器开无图省流【流量节省】
windows10怎么查看硬盘序列号_windows10硬盘id查询命令
win11 Snap Layouts怎么用 Win11窗口布局与分屏多任务高效指南【必学】
PHP中高效并行检查多链接状态的教程
邮政编码查询不到怎么办_邮政编码查询不到的常见原因与对策
微信语音通话掉线如何解决 微信语音通话稳定优化方法
AO3中文官网链接_AO3网页版稳定镜像站
C++如何解决segmentation fault_C++段错误调试与原因分析
邮政快递包裹最新位置 邮政快递实时追踪入口
Go语言中高效处理x-www-form-urlencoded表单数据
汽水音乐车机版横屏版7.1 汽水音乐车机版横屏版下载入口
Win11如何使用Windows Sandbox Win11沙盒功能开启与使用教程【详解】
TikTok网页版直接登录 TikTok网页端官方平台入口
4399体育竞技小游戏_4399小游戏赛事入口
如何使用Node.js csv 包按条件移除含空字段的CSV记录
怎么在mac上运行html代码_mac运行html代码方法【指南】
HTML元素状态管理:根据DIV内容动态启用/禁用按钮
抖音商城签到领现金是真的吗_抖音商城签到奖励与提现说明
mc.js官网登录入口 mc.js官方登录入口最新版
C++如何实现单例模式_C++设计模式之线程安全的单例写法
sublime怎么覆盖插件的默认快捷键_sublime快捷键优先级与设置
高德地图公交到站提醒失败如何解决 高德提醒权限设置
消息称三星明年 2 月正式发布 HBM4,与 SK 海力士同台竞技
windows10怎么关闭系统提示音_windows10彻底静音设置方法
sublime怎么进行远程开发编辑_配置rsub/rmate实现sublime编辑服务器文件
J*aScriptWebpack优化_J*aScript构建工具实战
c++ 命名空间怎么用 c++ namespace使用指南
Python异步编程实践:使用Binance API构建实时交易数据流
解决macOS Tkinter应用双击启动崩溃:PyInstaller打包指南
c++中的std::basic_string的SSO优化_c++短字符串优化深度解析
b站赚钱渠道_b站收益来源
韩剧圈正版入口页面_韩剧圈官网登录链接
zookeeper 都有哪些功能?
基于动态规划的房屋花卉种植最小成本算法详解
LINUX怎么设置定时任务_LINUX crontab配置教程
聚水潭ERP登录页面入口 聚水潭ERP官网登录界面


2025-12-09
浏览次数:次
返回列表
iewport_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))