Go:使用 Ebiten 在 2D 视频游戏中进行图像渲染

共 4060字,需浏览 9分钟

 ·

2021-04-06 19:39

点击上方蓝色“Go语言中文网”关注,每天一起学 Go

插图由创作原始 Go Gopher 作品的 Renee French 为“ Go 的旅程”创作。

本文基于 Ebiten 1.10。

Ebiten[1] 是由 Hajime Hosh[2] 用 Go 语言编写的成熟的 2D 游戏库。它是 Apple Store 上一些手机游戏如 Bear's Restaurant[3] 或桌面游戏如 OpenDiablo2[4] 的引擎,OpenDiablo2 是 Go 版本暗黑 2 的开源实现。现在,让我们深入了解电子游戏中的一些基本概念以及它们在 Ebiten 中的实现。

动画制作

在电子游戏世界中,Ebiten 通过分离的静态图像来渲染动画。这些图像集合被组合成一个更大的图像,通常称为“纹理图集[5]”,也称为“精灵图”。这是 网站[6] 上提供的示例:

image1

然后,经过简单的数学运算,这张图被加载到内存中,然后被一部分一部分地渲染。在前面的示例中,每个部分的宽度为 32 个像素。渲染每个图像非常简单,只需将 X 坐标移动 32 个像素。这是第一步:

image2
image3

渲染动画时,只需要当前图像的值即可渲染精灵图的正确部分。Ebiten 提供了所有的 API 来轻松地渲染它,下面是之前第一步中精灵图的示例:

screen.DrawImage(
   // 在坐标上绘制子画面的子图像
   // x=0 to x=32 and y=32 to y=64
   runnerImg.SubImage(image.Rect(0323264)).(*ebiten.Image),
   // 在该位置之前声明的变量定义了屏幕上的位置
   op,
)

Ebiten 的创建者 Hajime Hosh 还开发了“ file2byteslice[7]”,该工具可将任何图像转换为字符串,并允许将任何文件嵌入到 Go 中。它可以通过注释 go:generate 与 Go 工具轻松集成,自动将图像转储到 Go 文件中。下面是一个例子:

//go:generate file2byteslice
  -package=img
  -input=./images/sprite.png
  -output=./images/sprite.go -var=Runner_png

现在可以从以下代码的 img.Runner_png 变量中获取精灵图:

image4

然后,必须先对图像进行解码,然后再由 Ebiten 加载并在游戏中进行渲染。让我们转到另一种渲染较大背景图像的方法,该图像由循环元素组成。

瓷砖背景

渲染背景使用相同的技术,将主要地图集分为许多小图像,称为“瓷砖”。这是网站上提供的纹理图集的示例:

tiles1

该图集可以被 16 像素的图块分割 ,这是使用 Tiled[8] 软件创建的图块集合:

tiles2

每个图块将分配一个数字,从 0 开始,逐次加 1。由于每行都有相同数量的图块,因此可以通过除法和取模来获取图块的坐标。这是蓝色花朵的示例:

tiles3

蓝色花朵的编号为 303,这意味着它们位于第 4 列(303 以每行的图块数为模,例如 303%25 = 3)和第 13 行(303 除以每行的图块数,例如,303/25 = 12)。

现在,我们可以使用索引数组来构建地图:

map_array

从这张地图上绘制图像可以得到主要装饰:

bg1

但是缺少背景。我们必须构建一个表示背景的类似数组。结果如下:

bg2

现在,这些图层已经准备好了,我们只需要迭代两个图层就可以得到最终结果:

bg3

此示例中的代码可在 Ebiten 网站[9] 上找到。

生成图像后,Ebiten 必须管理屏幕更新并将指令发送到图形卡片。

屏幕更新

Ebiten 提供了 iOS(使用 Metal)和 Android(使用 OpenGL ES)使用的驱动程序的抽象,这使开发更加容易。它还允许您定义一个函数,该函数可以更新屏幕并绘制所有更改。但是,出于性能原因,该库会将源打包在一起,并将这些更改存储在缓冲区中,然后再发送给驱动程序:

screen_update1

该缓冲区还能够合并绘图指令,以减少对 GPU 的调用次数。在上图中,这三个指令现在可以合并为一个:

screen_update2

这项改进很重要,因为它可以减少发送指令时的开销。这是合并指令的性能:

perf1

通过逐个发送指令,性能会大大降低:

perf2

Ebiten 还提供了对屏幕刷新的控制,这有助于调整性能。

TPS 管理

Ebiten 的默认 TPS 是 60。但是,这可以用 Ebiten 提供的 API ebiten.SetMaxTPS() 来轻松配置。它有助于减少机器上的压力。这是具有 25 TPS 的简单程序:

tps1

TPS(每秒 tick 数)与 FPS(每秒帧数)不同。Hajime Hosh[10] 很好地描述了这些差异:

帧代表图形更新。这取决于用户显示屏上的刷新率。那么 FPS 可能是 60、70、120,依此类推。这个数字基本上是不可控制的。Ebiten 可以打开或关闭 vsync。如果关闭了 vsync,则 Ebiten 会尝试尽可能多地更新图形,那么 FPS 可以为 1000 左右。tick 表示逻辑更新。TPS 表示每秒调用更新功能的次数。默认情况下固定为 60。游戏开发人员可以通过 SetMaxTPS 配置 TPS。如果设置了 UncappedTPS,则 Ebiten 会尝试尽可能多地调用更新函数。

当窗口在后台运行时,Ebiten 为渲染带来了另一种优化。失去焦点时,游戏将进入休眠状态,直到重新获得焦点。它实际上 sleep 了 1/60 秒,然后再次检查焦点。这是保存的资源的示例:

tps2

尽管可以关闭此优化,但是运行的系统可能会限制游戏的运行。例如,在 Firefox[11]Chrome[12] 或其他浏览器上的后台标签中运行时,在浏览器中运行的游戏会受到限制。

Ebiten 在 Github[13] 上接受赞助。如果您希望看到更多 Go 编写的游戏,请随时贡献力量。


via: https://medium.com/a-journey-with-go/go-image-rendering-in-2d-video-games-with-ebiten-912cc2360c4f

作者:Vincent Blanchon[14]译者:alandtsang[15]校对:polaris1119[16]

本文由 GCTT[17] 原创编译,Go 中文网[18] 荣誉推出

参考资料

[1]

Ebiten: https://ebiten.org/

[2]

Hajime Hosh: https://github.com/hajimehoshi

[3]

Bear's Restaurant: https://daigostudio.com/bearsrestaurant/en/

[4]

OpenDiablo2: https://github.com/OpenDiablo2

[5]

纹理图集: https://en.wikipedia.org/wiki/Texture_atlas

[6]

网站: https://ebiten.org/examples/animation.html

[7]

file2byteslice: https://github.com/hajimehoshi/file2byteslice

[8]

Tiled: https://www.mapeditor.org/

[9]

Ebiten 网站: https://ebiten.org/examples/tiles.html

[10]

Hajime Hosh: https://github.com/hajimehoshi

[11]

Firefox: https://hacks.mozilla.org/2018/01/firefox-58-the-quantum-era-continues/

[12]

Chrome: https://developers.google.com/web/updates/2017/03/background_tabs

[13]

Github: https://github.com/sponsors/hajimehoshi

[14]

Vincent Blanchon: https://medium.com/@blanchon.vincent

[15]

alandtsang: https://github.com/alandtsang

[16]

polaris1119: https://github.com/polaris1119

[17]

GCTT: https://github.com/studygolang/GCTT

[18]

Go 中文网: https://studygolang.com/



推荐阅读


福利

我为大家整理了一份从入门到进阶的Go学习资料礼包,包含学习建议:入门看什么,进阶看什么。关注公众号 「polarisxu」,回复 ebook 获取;还可以回复「进群」,和数万 Gopher 交流学习。

浏览 49
点赞
评论
收藏
分享

手机扫一扫分享

分享
举报
评论
图片
表情
推荐
点赞
评论
收藏
分享

手机扫一扫分享

分享
举报