这家80岁的游戏厂商,给了入行新人一份开发培训教材
更为普适的一本游戏开发书籍。文/龙之心半个月前,虚幻5引擎的亮相在游戏行业引发了广泛讨论,同时也点燃了一部分人的游戏开发热情。对于初学者而言,学习游戏开发难免会陷入一种无从下手的困局,怎么去建立起一套知识体系来更好地掌握这门技能?从最近出版的一本书中,葡萄君一定程度上找到了这个参考答案。这本书名为《游戏开发:世嘉新人培训教材》,作者是平山尚,曾在世嘉参与《电脑战机》(PS2)、《超级网球大奖赛3》(AC、PS3)的开发。世嘉在游戏行业的地位无需赘述,它曾与任天堂、索尼、微软并列四大家用游戏机制造商,是业内知名的游戏开发公司。本书推出后很快受到世嘉内部新人欢迎,并荣获游戏开发者奥斯卡CEDEC AWARDS 2009著作奖,称得上一本经典书。
2012年夏天,本书译者、拥有15年游戏开发经验的资深工程师罗水东在东京的书店里发现此书,并被其吸引,不过因为版权的问题,翻译一事被搁浅。随着图灵引进该书中文版权,翻译工作正式启动。值得一提的是,为了确保翻译的精准,罗水东将原著反复读了3次。要知道,这本书多达27个章节、676页,能全篇通读下来就已经很不容易了,而在翻译出来的同时,他还将本书的可读性尽可能提高了不少。这本书写了什么?《游戏开发:世嘉新人培训教材》一书可分为三部分:2D游戏、3D游戏和通往商业游戏之路,而这种层层铺垫、由浅及深的知识体系也比较适合新人来阅读(文末附目录)。在2D游戏的开始,作者率先开发了经典的推箱子游戏。在他看来,这款仅200行代码就能运行的游戏,做起来并不是想象中的小菜一碟,需要考虑的东西是相当多的。
拿“移动”来举例,游戏中不存在玩家和箱子分开移动的情况,因此移动处理就是判断玩家朝上下左右哪个方向移动。而在此之前,首先要判断移动位置是否处于允许范围之内;编写代码时,开头第一个if语句检测要移动的位置是否为空,如果为空则移动到该处。
部分代码展示当然,这个所谓的空可能有“空白”和“目的地”两种情况,所以代码会有些复杂。如果将“是否是目的地”的信息存储在别处,因为该数据固定而且不会被修改,所以代码写起来可能会更简单。接下来则是对代码的进一步改善,如数据容量、处理速度、代码可读性和可拓展性等等。完成核心模块后,作者下一步开始引导游戏更好地朝商业化发展,从像素到图片素材,再到动画、文本绘制、声音等角度深入开发,来输出成品。而在3D游戏部分,你可以学着自制动作游戏《机甲大战》,过程中涵盖计算机运算(3D碰撞处理)、计算机图形学(角色动画)、编程等知识细节。
举一个《机甲大战》自动跟踪的例子。更好的设计应当是玩家可以在导弹即将击中自己的瞬间,通过跳跃或者横向闪避来躲开导弹,而且此时的导弹不允许突然掉头袭击,不然就算作弊了。因此,代码只要让导弹飞行过程中每帧都朝着敌人的方向改变一定角度,让导弹沿着头部的朝向移动,然后逐渐改变它的方向。“角度的调整通过上下角(绕x轴旋转)与方向角(绕y轴旋转)实现。绕z轴旋转意味着导弹沿着其自身的轴心旋转,这只起到一个装饰的作用,所以绘制时再去思考就可以了。每帧计算出当前敌人所处的方位,并设置以上两个角度变量使其接近该值。这种方法虽然不完美,但在具备重力的环境近距离作战时并不会有什么问题。接下来用角度的补间方法使其绕x轴旋转即可。不过,绕x轴的旋转角的范围要控制在-90°至90°,超过90°意味着导弹的上下两面发生了反转。这一点并不难理解。我们抬起头一直往后仰,最终就会看到地面出现在我们的“上方”。任何朝向都可以通过一个-90°到90°的绕x轴的旋转角与一个-180°到180°的绕y轴的旋转角来表示,所以这种设计是没问题的。就好比我们想欣赏身后的风景时不必仰头反转,只要旋转身体就可以了。”
当掌握了3D游戏开发的基础知识,能够独立地开发一款产品后,接下来的考验就集中在商业化之前的这段优化过程上,比如如何缩短加载时间?如何预防和处理bug?以代码为例,书中提到,在项目中要求统一风格的情况下,如果有程序员不愿意改变自己的风格,可能就会起冲突。但事实上,每个人都应该接受好的代码规范,做到“不复杂、不繁琐、有必要”。
在本书最后笔者坦言,“虽然写了这样厚厚的一本书,但初入游戏公司时连书的十分之一都没未掌握”。在他看来,以前只要有兴趣,一个人也能开发一款游戏,但参考当下游戏规模的技术复杂程度,一己之力很难开发像样的游戏了。因此,这样一本教材或许能体现出它的价值。更为普适的一本开发书籍其实,进入游戏行业的新人往往存在对专业书籍的诉求,而市面上多数游戏开发书籍很难兼顾专业性和可读性,又以可读性的缺失最为明显。这便导致读者阅读门槛提高,有可能无法坚持看完一本书。《游戏开发:世嘉新人培训教材》则针对性地解决了这个问题。它以开发游戏demo、梳理设计知识为轴,尝试穿插了不少私货,让读者知道“怎么做”的同时,明白“为什么”这么做,甚至跟策划扯皮的过程也夹带进来,显得更接地气。另一方面,虽然书名提到“新人培训教材”,但受众并不限于新人。Unity User Group意见领袖、UVP价值专家宣雨松MOMO评价称:“它远远超过了新人培训,甚至适合工作多年的老手。现在市面上有很多好用的商业游戏引擎,它们将大量原理部分的知识隐藏起来,暴露给程序员的都是简单易用的接口。时间久了很多程序员虽然能开发游戏,但已经浑然不知背后的原理知识了。如果对原理不了解,就意味着无法更好地和硬件打交道,无法更好地优化自己的游戏。这本书从无到有一点点揭秘游戏开发的原理,由浅入深,很容易理解,确实是一本不可不读的好书。”
《游戏开发:世嘉新人培训教材》开头提到了这样一句话,“本书就像一条从山脚蜿蜒而上的山路,虽漫长但不陡峭,让读者一路欣赏风景直至顶峰,不会半途而弃。这也是导致本书页数超级多的原因。这些风景就是当下游戏编程中最常见的内容。”在众多游戏开发书籍中,这或许就是本书的最大竞争力。(目前该书已在葡萄书房上架,长按下方图片扫码进入小程序即可购买。欢迎各位读者在评论区留言,葡萄君将随机抽取3位读者赠送一本《游戏开发:世嘉新人培训教材》)
长按此图扫码进入小程序即可购买附本书目录:向上滑动查看第 1部分 2D 游戏.1第 1章 第 一个游戏21.1 开发一个益智游戏 .31.2 示例代码解说 71.3 添加读取场景数据的功能 161.4 C 课堂 211.5 补充内容:标志位和位运算261.6 补充内容:指针和内存 341.7 补充内容:引用 411.8 本章小结 46第 2章 从像素开始学习2D 图形处理 .472.1 什么是2D 图形处理 482.2 准备工作 502.3 打印一个点 542.4 移植《箱子搬运工》 .552.5 补充内容:结束处理 .582.6 本章小结 60第3章 使用图片素材 613.1 读取图片文件 .623.2 带图片的《箱子搬运工》的示例代码723.3 使用透明通道 .753.4 头文件包含关系的组织策略 .823.5 补充内容:透明混合的性能优化 863.6 补充内容:加法混合 .883.7 本章小结 90第4章 实时游戏 914.1 什么是实时游戏 924.2 运行动画 954.3 带动画的《箱子搬运工》.964.4 获得游戏的帧率 .1014.5 解决帧率差异 1034.6 补充内容:根据帧率变化动态改变游戏运行速度 1054.7 补充内容:影像撕裂现象 1104.8 本章小结 .111第5章 简单的状态迁移 1135.1 往类库追加功能 .1145.2 相对直接的做法 .1175.3 试着增加状态 1195.4 代码审查 .1225.5 示例代码解说 1285.6 本章小结 .136第6章 文本绘制方法 .1376.1 字体图片 .1386.2 文本绘制函数 1386.3 一些改进 .1406.4 成果验证 .1426.5 示例代码解说 1436.6 注意著作权 1476.7 示例类库的功能 .1476.8 本章小结 .148第7章 动作游戏初体验 1497.1 用到的类库 1507.2 开发《炸弹人》 .1517.3 示例代码解说 1527.4 添加背景显示 1577.5 配置移动的对象 .1647.6 游戏的改进方向 .1697.7 本章小结 .170第8章 2D 平面内的碰撞处理 1718.1 碰撞检测 .1728.2 碰撞响应 .1748.3 发生多个碰撞时的问题 .1798.4 碰撞响应与操作性 1808.5 移动的物体相互碰撞 1848.6 《炸弹人》的碰撞处理 .1848.7 本章小结 .186第9章 各种输入设备 .1879.1 获取输入设备实例 1889.2 键盘 1899.3 鼠标 1899.4 手柄 1909.5 在《炸弹人》游戏中使用手柄 .1919.6 本章小结 .193第 10章 状态迁移详解 19410.1 问题定位 19510.2 使用继承 19510.3 实际运用 19810.4 补充内容:简化状态迁移的代码 .20110.5 补充内容:跨层级的状态迁移处理的改进 20410.6 补充内容:继承的原理 20810.7 本章小结 213第 11章 播放声音 .21511.1 关于音频类库 21611.2 补充内容:计算机如何播放声音 .21811.3 补充内容:音高和音量 21911.4 补充内容:音色 22011.5 补充内容:声音的叠加 22111.6 补充内容:do、re、mi 的原理 22211.7 补充内容:演奏乐谱 22311.8 补充内容:读取WAV 音频文件 22511.9 补充内容:使用Sound 模块来合成声波 .22611.10 本章小结 226第 12章 旋转、缩放与平移 22712.1 旋转 .22812.2 引入向量和矩阵 24112.3 利用顶点来实现 24612.4 缩放 .25312.5 在缩放的同时进行旋转.25512.6 矩阵的力量 .25612.7 补充内容:旋转公式的由来 26312.8 补充内容:更为实用的旋转处理方法 26612.9 补充内容:数学中的矩阵 .26712.10 本章小结 271第 13章 显卡的力量 27213.1 关于使用的类库 27313.2 使用显卡绘制三角形 27313.3 将图像贴到三角形中 27513.4 混合模式 27813.5 旋转、缩放和移动 .27813.6 移植《炸弹人》 28113.7 本章小结 284第 2部分 3D 游戏285第 14章 绘制立体物体 28614.1 关于类库 28714.2 开始制作3D 动作游戏《机甲大战》 .28714.3 绘制三角形 .28914.4 按位置前后绘制物体 29014.5 将远处的物体绘制得小一些 29414.6 坐标变换 30314.7 用矩阵表示透视变换 31614.8 开始制作《机甲大战》 32214.9 补充内容:Z 缓存的精度问题 33514.10 本章小结 338第 15章 类库的封装方法 .34015.1 整体设计 34115.2 资源的详细内容 34215.3 试运行 35015.4 从文件载入 .35215.5 补充内容:将类库从游戏中分离 .35415.6 本章小结 363第 16章 伪XML 文件的读取 36416.1 确定文件格式 36516.2 创建前的准备 36616.3 处理流程 36816.4 字符串解析 .37116.5 编写代码 37316.6 运用 .37516.7 示例代码解说 37816.8 补充内容:生成数据文件38216.9 本章小结 385第 17章 编写高性能的代码 38617.1 算法与时间复杂度 .38717.2 数据结构基础 39017.3 吞吐量与延迟 40217.4 并行处理 40417.5 内存问题 40417.6 STL 和数据结构 41117.7 性能瓶颈分析 41617.8 补充内容:函数调用的开销 41717.9 补充内容:高效运算与低效运算 .42017.10 本章小结 422第 18章 3D 碰撞处理 .42318.1 长方体的碰撞处理 .42418.2 使用浮点数的碰撞检测 42718.3 三角形和线段的相交检测 .43618.4 实用性 44518.5 其他问题 44718.6 本章小结 448第 19章 《机甲大战》的设计 45019.1 状态迁移 45119.2 操作 .45119.3 发射导弹 45419.4 将功能整合到一起 .45719.5 前端展现 46119.6 不足之处 46319.7 本章小结 464第 20章 光照 46520.1 看见物体的过程 46620.2 光的衰减过程 46920.3 尝试计算 47320.4 整合到《机甲大战》中.47920.5 补充内容:性能优化 48020.6 补充内容:更好的绘制效果 48320.7 本章小结 484第 21章 角色动画 .48521.1 相对运动 48621.2 层级Model 类 49221.3 自动构建树结构 49521.4 将动画数据化 50021.5 补间方法 50621.6 引入到《机甲大战》中.51821.7 补充内容:联立方程组vs 斜率指定法 51921.8 补充内容:不足之处 52021.9 本章小结 523第3部分 通往商业游戏之路525第 22章 高效的碰撞检测 .52622.1 低效的循环判断方法 52722.2 性能改善的基本思路 52822.3 基于排序的方法 53122.4 依靠分割实现的方法 53922.5 补充内容:改进等分切割法54522.6 补充内容:空间分割的高级技巧 .54822.7 本章小结 552第 23章 数据加载 .55423.1 为何加载时间越来越长.55523.2 文件加载类 .55623.3 通过合并文件提升性能.56223.4 通过压缩提升性能 .57123.5 补充内容:多线程异步处理57923.6 补充内容:编码技术 59023.7 补充内容:用于加密的合并与压缩 59123.8 本章小结 592第 24章 float 的用法 59324.1 位数限制 59424.2 float 的实现 59424.3 位数截断带来的误差 59624.4 误差程度 59724.5 减小误差的方法 60024.6 特别的数 60624.7 本章小结 608第 25章 随书类库概要 60925.1 类库中的类 .61025.2 启动设定 61125.3 Framework 模块 61325.4 WindowCreator 模块 61425.5 FileIO 模块 61425.6 Base 模块 61525.7 Math 模块 61625.8 Threading 模块 61825.9 Input 模块 .62025.10 Sound 模块 62025.11 PseudoXml 模块 62125.12 Graphics 模块 62125.13 Scene 模块 62825.14 抗锯齿处理 63025.15 将《机甲大战》改用最终版类库实现 63125.16 X 文件 .63325.17 本章小结 634第 26章 bug 的应对方法 .63526.1 防火与灭火 .63626.2 bug 的种类 .63626.3 bug 的预防 .64126.4 bug 的处理 .65426.5 《机甲大战》的处理 65626.6 补充内容:如何检测内存溢出 65726.7 本章小结 660第 27章 进阶方向 .66227.1 应该学习什么 66327.2 工具开发 66427.3 AI 66727.4 网络 .66727.5 shader 66827.6 参考文献 66927.7 一些期待的书 672后记 .675致谢 .676