《游戏编程模式》一1.1 什么是软件架构

1.1 什么是软件架构

如果你从头到尾阅读了这本书,那么你并不会了解到3D图形背后的线性代数或者游戏物理背后的演算。这本书也不会告诉你如何一步步改进你的AI搜索树或者模拟音频播放中的房间混响。

哇,此段简直为这本书打了一个糟糕的广告。

相反,这本书是关于上面这一切要使用的代码的组织方式。这里少谈代码,多谈代码组织。每个程序都具有一定的组织性,即使它只是“把所有东西扔到main()函数里然后看看会发生什么”,所以我认为讨论如何形成好的组织性会更有趣些。我们如何分辨一个架构的好坏呢?

我大概有5年时间一直在思索这个问题。当然,像你一样,我对好的设计有着一种直觉。我们都遇见过非常糟糕的代码库,最希望做的就是剔除它们,结束自己的痛苦。

不得不承认,我们大多数人只接触到一部分这样的工作。

少数幸运儿有相反的经验,他们有机会与设计精美的代码共事。那种代码库,感觉就像在一个完美的豪华酒店里站了很多礼宾在翘首等待你的光临。两者之间有什么区别呢?

1.1.1 什么是好的软件架构

对于我来说,好的设计意味着当我做出一个改动时,就好像整个程序都在期待它一样。我可以调用少量可选的函数来完美地解决一个问题,而不会为软件带来副作用。

这听起来不错,但还不够切实。“只管写你的代码,架构会为你收拾一切。”没错!。

让我解释下。第一个关键部分是,架构意味着变化。人们不得不修改代码库。如果没人接触代码(不管是因为代码非常完美,又或者糟糕到人人都懒得打开文本编辑器来编辑它),那么它的设计就是无法体现其意义的。衡量一个设计好坏的方法就是看它应对变化的灵活性。如果没有变化,那么这就像一个跑步者从来没有离开过起跑线一样。

1.1.2 你如何做出改变

在你打开编辑器添加新功能,修复bug或者由于其他原因要修改代码之前,你必须要明白现有的代码在做什么。当然,你不必知道整个程序,但是你需要将所有相关的代码加载到你的大脑中。

这在字面上是一个OCR1过程,不过这个想法有些奇怪。

我们倾向于略过这一步,但它往往是编程中最耗时的部分。如果你认为从磁盘加载一些数据到RAM很慢的话,试着通过视觉神经将这些数据加载到你的大脑里。

一旦你的大脑有了一个全面正确的认识,则只需稍微思考一下就能提出解决方案。这观点值得反复斟酌,但通常这是比较明确的。一旦你理解了这个问题和它涉及的代码,则实际的编码有时是微不足道的。

你的手指游走于键盘间,直到右侧的彩色灯光在屏幕上闪烁时,你就大功告成了,是吗?还没有!在你编写测试,并将它发送给代码审查之前,你通常有一些清理工作要做。

我说“测试”了吗?哦,是的,我说了。为一些游戏代码编写单元测试比较难,但是大部分代码是可以完全测试的。

我这里不是要慷慨陈词,不过,如果你之前没有考虑过多做自动化测试的话,我希望你多做一些。难道没有比一遍一遍手动验证东西更好的事情要做吗?
你在游戏中加入了一些代码,但是你不想后面处理代码的人花大量时间理解或修改你的代码。除非变动很小,通常都会做些重新组织工作来让你新加的代码无缝集成到程序中。如果你做得很好,那么下一个人在添加代码的时候就不会察觉到你的代码变动。

简而言之,编程的流程图如图1-1所示。

现在想想,流程图的环路中没有出口有点小惊悚。

1.1.3 我们如何从解耦中受益

虽然不是很明显,但我认为很多软件架构师还处于学习阶段。将代码加载到脑中如此痛苦缓慢,得自己寻找策略来减少装载代码的体积。这本书有一整章(第5章)是关于解耦模式的,许多的设计模式也有相同的思想。

你可以用一堆方式来定义“解耦”,但我认为如果两块代码耦合,意味着你必须同时了解这两块代码。如果你让它们解耦,那么你只需了解其一。这很棒,因为如果只有一块代码和你的问题相关,则你只需要将这块代码装载到你的脑袋中,而不用把另外一块也装载进去。

对我来说,这是软件架构的一个关键目标:在你前进前,最小化你脑海中的知识储存量。

当然,对解耦的另一个定义就是当改变了一块代码时不必更改另外一块代码。很明显,我们需要更改一些东西,但是耦合得越低,更改所波及的范围就会越小。

时间: 2024-09-19 23:44:14

《游戏编程模式》一1.1 什么是软件架构的相关文章

《游戏编程模式》一导读

前 言 游戏编程模式在五年级的时候,我和我的小伙伴们获准使用一个放置着几台非常破旧的TRS-80s[1]的闲置教室.为了激励我们,一位老师找到了一份印有一些简单BASIC程序的打印文档给我们. 当时,计算机上的音频磁带驱动器是坏掉的,所以每次我们想要运行一些代码的时候,都不得不仔细地从头开始键入代码.这使得我们更喜欢那些只有几行代码的程序: 如果计算机打印足够多的次数,或许它会神奇的变成现实哦[2]. 10 PRINT "BOBBY IS RADICAL!!!" 20 GOTO 10

《游戏编程模式》一第1章 架构,性能和游戏

第1章 架构,性能和游戏 游戏编程模式在我们一头扎进一堆模式之前,我想为你介绍一些关于我如何看待软件架构以及它是如何应用到游戏的一些背景,这可能会帮助你更好地理解这本书的其余部分.至少,当你陷入关于设计模式和软件架构是多么糟糕(或者很棒)的一场争论中时,它会给你一些论据来使用. 请注意,我没有假设你站在争论中的哪一方.就像任何军火商一样,我为所有战斗方提供武器.

《游戏编程模式》一第7章 状态模式

第7章 状态模式 "允许一个对象在其内部状态改变时改变自身的行为.对象看起来好像是在修改自身类." 交代一下:我写的有些过头了,我在本章里面添加了太多东西.表面上这一章是介绍状态模式[1]的,但是我不能抛开游戏里面的有限状态机(finite state machines,FSM)而单独只谈"状态模式".不过,当我讲到FSM的时候,我发觉我还有必要再介绍一下层次状态机(hierarchical state machine)和下推自动机(pushdown automat

《游戏编程模式》一1.3 性能和速度

1.3 性能和速度 你有时候会听到关于软件架构和相关概念的批评声,尤其在游戏开发中:它会影响到游戏的性能.许多模式让你的代码更加灵活,但是它依赖于虚函数派发.接口.指针.消息以及其他至少有一些运行成本的机制. 一个有趣的范例是C++模板.模板元编程有时可以让你获得抽象接口而没有任何运行时开销. 对灵活的定义,不同人有不同的看法,当你在某些类中调用一个具体方法时,你相当于将这个类固定(很难做出改变).当你使用一个虚方法或者接口时,被调用的类将直到真正运行起来才能被追踪到,这样的程序更具灵活性但是会

《游戏编程模式》一7.2 救星:有限状态机

7.2 救星:有限状态机 为了消除你心中的疑惑,你可以准备一张纸和一支笔,让我们一起来画一张流程图.对于女主角能够进行的动作画一个"矩形":站立.跳跃.躲避和俯冲.当你可以按下一个键让主角从一个状态切换到另一个状态的时候,我们画一个箭头,让它从一个矩形指向另一个矩形.同时在箭头上面添加文本,表示我们按下的按钮. 恭喜,你刚刚已经成功创建了一个有限状态机.有限状态机借鉴了计算机科学里的自动机理论(automata theory)中的一种数据结构(图灵机)思想.有限状态机(FSMs)可以看

《游戏编程模式》一7.5 状态对象应该放在哪里呢

7.5 状态对象应该放在哪里呢 我这里忽略了一些细节.为了修改一个状态,我们需要给state_指针赋值为一个新的状态,但是这个新的状态对象要从哪里来呢?我们之前的枚举方法是定义一些数字.但是,现在我们的状态是类,我们需要获取这些类的实例.通常来说,有两种实现方法. 7.5.1 静态状态 如果一个状态对象没有任何数据成员,那么它的唯一数据成员便是虚表指针了.那样的话,我们就没有必要创建此状态的多个实例了,因为它们的每一个实例都是相同的. 在那种情况下,我们可以定义一个静态实例.即使你有一系列的FS

《游戏编程模式》一7.1 我们曾经相遇过

7.1 我们曾经相遇过 假设我们现在正在开发一款横版游戏.我们的任务是实现女主角--游戏世界中玩家的图像.我们需要根据玩家的输入来控制主角的行为.当按下B键的时候,她应该跳跃.我们可以这样实现: void Heroine::handleInput(Input input) { if (input == PRESS_B) { yVelocity_ = JUMP_VELOCITY; setGraphics(IMAGE_JUMP); } } 找找看,bug在哪里? 这里应该还有如果主角着地将isJum

《游戏编程模式》一7.3 枚举和分支

7.3 枚举和分支 一个问题是,Heroine类有一些布尔类型的成员变量:isJumping_和isDucking_,但是这两个变量不应该同时为true.当你有一系列的标记成员变量,而它们只能有且仅有一个为true时,这表明我们需要把它们定义成枚举(enum). 在这个例子当中,我们的有限状态机的每一个状态可以用一个枚举来表示,所以,让我们定义以下枚举: enum State { STATE_STANDING, STATE_JUMPING, STATE_DUCKING, STATE_DIVING

《游戏编程模式》一7.4 状态模式

7.4 状态模式 对于熟知面向对象方法的人来说,每一个条件分支都可以用动态分发来解决(换句话说,都可以用C++里面的虚函数来解决).但是,如果这样做,你可能会把简单问题复杂化.有时候,一个简单的if语句就足够了. 状态模式的由来也有一些历史原因.许多面向对象设计的拥护者-- GoF和重构的作者Martin Fowler都是Smalltalk出身.在那里,如果有一个ifThen语句,我们便可以用一个表示true和false的对象来操作.但是,在我们这个例子当中,我们发现面对对象设计也就是状态模式更