游戏引擎学习第71天

回顾

我们目前正在进行一个类似于整体构建的过程,目标是将所有的部分整合在一起。我们希望确保每个组成部分能够顺利地适配到整体结构中。尽管目前每个具体的部分还有很多工作要做,但我们在思考如何将各个部分连接成一个完整的系统。我们在上周五做了一个待办清单,目的是梳理出完成这项工作的所有必要步骤。

现在,至少我们处于一个可以检查进度和下一步任务的状态。当我们完成一个阶段性的工作时,可以停下来审视当前的进展,看看还有哪些待完成的任务,并选择下一个合适的任务继续进行。这样的方法可以帮助我们更好地推进整体进展。

这个过程为我们提供了一个很好的工作流程,可以在忘记某些步骤时快速回顾和执行下一步任务。

人们如何使用他们的TODO笔记本的巧合

在编程时,我们通常会保持一本笔记本,记录下我们在编程过程中需要记住的事项。我们会把这些事项列成一个清单,并在完成某些任务或者决定不再做某些事情后,从清单中移除它们。当页面上的内容填满时,我们会换一页,并将清单上的内容复制到新的一页,再次过滤掉那些不再需要或已经完成的任务。

这种方法似乎非常流行,虽然我们无法确切解释为什么它比使用任务管理工具更受欢迎。重要的是,这种手写和抄写的过程似乎对我们有某种特殊的意义,尽管我们并不完全理解其中的原因。

今天的任务是继续解决我们待办事项上的问题。为了跟进这个过程,如果你想一起操作,你可以下载并解压第70天的源代码文件。接下来,我们将打开这些源代码并继续进行我们的工作。

修复bug:移除我们关于familiar留下的代码

在编程过程中,曾有人在论坛上提到一个问题,指的是我们并没有完全完成某些代码的更新,特别是在处理敌人代码时,我们没有移动代码到新的模块。这意味着我们仍在使用旧的更新方式,导致某些元素的距离计算不准确,尤其是敌人与英雄之间的距离。我们曾经做过一个临时修改,使英雄与熟悉的敌人之间的距离显得更近,目的是让敌人更容易追随英雄,但这导致了不正确的行为。

现在,我们决定清理这些不再需要的代码,恢复到原本的状态,以确保距离计算更准确。虽然这个更改并不大,但它有助于避免未来可能的错误,尤其是在最终版本中不小心使用了这些过时的代码。

接下来,我们需要集中精力解决我们目前的待办事项,而不是开始新的任务。虽然我们之前在周五进行了头脑风暴并提出了一些新的任务,但我们决定先完成现有的工作。这是为了确保我们在生产工作正式开始前,将所有积累的事项处理完毕,以便更好地专注于接下来的任务。

因此,我们决定暂时搁置碰撞检测的工作,转而处理一些更直接、更紧急的任务,确保能够在未来几天内完成当前的工作,特别是在我们即将进入一周的休假之前。

今天决定做什么:通过使用v3来清理事物

我们决定清理代码中有关z轴使用的问题。为了确保代码一致性,我们计划快速检查一遍代码,整理出可以优化的部分。

具体来说,我们注意到有些地方本可以使用向量3来处理z轴相关的数据,但实际上没有这样做。我们希望通过这次清理,统一处理方式,减少因z轴数据的不一致而可能导致的问题。

在清理的过程中,我们会对代码进行进一步的收紧和优化,确保代码结构更加清晰明了,同时减少z轴相关数据在不同模块之间的混乱情况。通过这些改进,我们希望代码能够更加规范化,为后续开发奠定良好的基础。

去除结构体 "world_difference",并修复我们使用它的其他地方

我们计划优化代码中关于"世界差异"的实现。最初的实现使用了一个单独的类来表示三维坐标之间的差异,但随着代码的演进,我们已经引入了向量的支持。因此,我们决定废除这个独立的类,直接使用向量3来统一处理三维坐标。

在实现过程中,我们将逐一修复代码中所有引用旧类的地方,确保它们可以直接使用v3。通过这种方式,可以减少重复计算和代码冗余,提高代码的可维护性和一致性。

同时,我们意识到在处理z轴数据时,需要特别考虑"块"的高度(或深度)是否对称。z轴上的不对称性可能对某些计算有影响,因此我们将调整代码,使其能够正确处理这些不对称情况。为此,我们需要优化计算逻辑,确保当z轴的数据参与运算时,能够充分考虑其实际高度。

此外,我们考虑进一步优化代码结构。例如,将"块的大小(以米为单位)"重构为一个向量的形式,而不是分别独立定义宽度、高度和深度。这种方法可以更直观地表示块的三维尺寸,同时减少多个变量定义带来的复杂性。

总体目标是通过这些调整,提升代码的规范性、可读性和扩展性,为后续开发奠定更加稳定的基础。

game_world.h: 将 ChunkSideInMeters 改为 v3

我们正在优化代码,主要目标是统一表示块的维度,并通过调整数据结构和算法提升代码的可维护性和功能性。具体如下:

  1. 块的维度统一表示

    将块的维度定义为一个三维向量 ChunkDimInMeters,包含 X、Y 和 Z 三个方向的数值。这种做法可以让块的维度更加灵活,便于我们根据需求自由调整各个方向的大小。

  2. 优化计算方式

    我们通过将块维度与dTile差异向量相乘的方式计算结果,这样可以避免逐一处理每个方向的计算。这种方法提高了代码的可读性和效率。

  3. 提高数学库的通用性

    计划向数学库中添加新的工具函数,例如向量的乘积操作。这些工具函数能够简化处理逻辑,同时减少重复代码,为后续开发提供便利。

  4. 偏移量的调整

    偏移量的处理也将从单独的分量处理转为三维向量表示。这种改进使偏移量的使用更加一致,降低出错的可能性。

  5. 代码结构改进

    在实现上述功能的同时,也逐步调整和重构代码,使其更加模块化和清晰化。例如,将所有三维相关的操作集中处理,减少冗余代码的分散。

总的来说,这些改进目标是为了统一维度表示,简化计算逻辑,提高代码的通用性和易读性,为后续复杂功能的实现奠定基础。

game_math.h: 实现Hadamard积

我们决定移除一些不必要的部分,因为在引入了新的点积和积元素功能后,这些部分已经显得多余。我们在代码中实现了点积(dot product)或内积(inner product)的功能,尽管名称看起来复杂,其实本质上就是将对应分量相乘然后求和。比如对于二维向量,这涉及到 x 1 × x 2 x_1 \times x_2 x1×x2 和 y 1 × y 2 y_1 \times y_2 y1×y2 的操作。

在实现积元素(component-wise multiplication)时,我们保留了各个分量的独立性,而不是将结果求和。这使得计算结果可以以向量的形式返回,方便进一步操作。这个实现对三维向量 v 3 v3 v3 同样适用,我们计划在未来扩展这些功能。

目前我们注意到 v 3 v3 v3 类的功能较为基础,很多实现仍然缺失,仅有一些简单的操作。这种状态让我们意识到需要重新审视 v 3 v3 v3 的实现,确保它的功能能满足当前需求并支持未来扩展。

通过这样的改动,我们的数学库将更加一致和高效,能够支持更多的操作和优化,为后续的开发打下基础。

game_math.h: 克隆 v2 操作到 v3

我们开始实现一个更加灵活的方法,尽管这种方式有时不被接受。选择不采用模板的原因是,模板可能导致较差的错误信息、更长的编译时间以及额外的复杂性。通过直接手动实现,我们能够更快地完成调试并减少潜在的问题。

模板相关的考虑:

  • 模板确实能减少一些代码重复,但它引入了额外的复杂性,例如必须处理模板错误和更长的编译时间。
  • 对于一次性调试的代码(例如数学库),手动实现可能是更直接的选择。
  • 如果需要,也可以选择模板实现,但需承担复杂性和维护成本。

当前实现的细节:

  1. 矢量运算操作

    • 对于矢量操作,使用逐分量的实现方法。例如,点乘和阿达玛积在逻辑上相似,但阿达玛积保留了分量操作结果。
    • 执行加法、减法和标量乘法时,分别对 xyz 分量进行独立计算,并将结果存入对应位置。
  2. 标量乘法

    • 实现了标量与矢量的乘法,即对每个分量进行标量乘法,并返回新的矢量。
    • 此外,还包括复合赋值操作(如 +=-=),简化了多步运算。
  3. 矢量基本功能

    • 实现了常见的矢量函数,例如计算矢量的长度和长度平方,基于分量的平方和。
    • 提供了内积和阿达玛积的实现,用于更复杂的数学运算。
  4. 大块处理

    • 计算两点之间的距离并乘以块大小,将结果转换为实际米的单位。
    • 此外,还扩展了对三维矢量的支持。

总结:

  • 通过非模板方式实现了基本矢量操作,确保了代码的可读性和易维护性。
  • 在实现过程中,权衡了模板与手动实现的优劣,选择最适合当前场景的方式。
  • 最终目标是提供一个灵活而高效的数学库,减少未来开发和调试的复杂性。

game_world.h: Offset_ 现在是 v3

在实现过程中,我们对当前的方法进行了反思,决定将一些功能简化或去除。这些变动虽然看似不那么重要,但实际上有助于减少冗余,提升代码的清晰度。通过这些调整,我们能够更好地理解和追踪数据的位置,进而推动后续工作的推进。

目前的目标是通过减少不必要的复杂性,来使得项目变得更加高效。尽管我们可能会放弃一些过去的实现,最终的结果可能更简洁,也更能适应当前的需求。这些更改并非全无意义,反而是基于对当前状态的重新评估,意在通过更直接和高效的方式实现目标。

总的来说,尽管这些调整可能看起来像是一些小的改变,但它们将帮助我们在未来减少不必要的工作量并提高代码的维护性。

指出 world_position 中的难题可以解决,因为现在我们不再操作那些没有带入模拟区域的实体

在这个过程中,我们发现可能已经通过一些架构设计工作,不自觉地解决了一个困惑我们的问题。我们之前在操作实体时,需要将它们带入同一区域并遍历瓦片块。然而,现在我们发现不再需要这样做,因为我们不再依赖于将实体放入同一区域,而是通过世界地位来定位实体。这意味着,我们不再需要存储特定的瓦片位置,而是通过计算x、y和z值来确定它们的位置。尽管我们可能仍然会用到一些信息来计算位置,但我们不再需要存储这些信息。这个变化似乎解决了之前困扰我们的难题,给我们带来了意想不到的解决方案,这也让我们感到非常惊喜。

回到解决编译错误

在这个过程中,我们讨论了如何通过规范化的方式处理世界的大小和维度。我们发现,通过使用世界的位置,我们不再需要处理复杂的偏移量,而是能够通过简化的方式进行计算。随着我们调整代码,将块的边界转换为米并处理每个维度,我们逐步解决了之前的复杂问题。

首先,我们需要确保当初始化世界时,所有的块大小和坐标都设置得当,尤其是在米为单位的情况下。随后,针对z轴,我们做了调整,认为z轴的深度会小于其他维度,因此我们决定将其设置为单一瓦片的高度。

在这个过程中,考虑到代码的简化,我们逐步淘汰了不再需要的变量,并优化了数据结构。最终,所有的变动都确保了代码流畅,并且数据能够正确地处理和返回。

这个过程虽然复杂,但随着不断地优化和调整,我们发现许多困扰我们的问题实际上得到了意外的解决。我们将继续推进这个方向,确保所有数据的流动和处理都能顺利进行。

地面平面不应该是 Tiles 的负底面,因为我们想要防止物体在 Z 轴边界附近加速

在讨论中,我们考虑了如何处理z轴的基准点。决定将人站立的位置设定为z=0,而不是使用负的底部值,这样可以避免振荡问题。目的是使站立在地面上的物体保持在合理的z范围内,从而避免不必要的计算和误差。

我们进一步探讨了是否应该将地图映射到块空间中,使用零作为映射中心,以此产生一致的结果。尽管最初并未使用这种方法,但考虑到后期可能需要调整,采用零中心映射似乎是一个更合适的选择。这种方式可以确保计算的稳定性,尤其是在处理较大的数据时。

我们还讨论了是否应该先进行除法运算的问题。考虑到计算的复杂性,这一决定取决于未来可能的需求,但目前的方案似乎并不急于改变。

game_world.h: 整理 ChunkPositionFromTilePosition 函数

在讨论中,我们决定继续按照现有方式进行,先进行必要的乘法运算,以便最终得到所需的结果。我们确定了将BasePos设置为零,并将世界位置映射到块空间的方案。所有这些运算将通过先前计算的偏移量进行调整。

尽管这一方式简洁且有效,但我们意识到可能会遇到精度问题,特别是在处理较大的数据时。考虑到这一点,未来在构建世界时,可能会考虑采用更智能的方案,尤其是在更大范围或更复杂区域的情况下。随着模拟的进行,我们还将定期更新位置,并通过已有的代码继续推动开发。

总体来看,当前的解决方案简单易行,能够为我们提供预期的结果。接下来,将继续导入相关代码,并进一步测试和优化这一方法。

game_math.h: 使用匿名结构体在联合体中获取我们感兴趣的 v3 部分

在讨论中,首先提到了一个结构体,包含 x 和 y 坐标。目标是将其扩展为包含 x、y 和 z 坐标的结构,并逐步实现这一目标,而不是一次性进行太多改动。通过第二轮修改,添加更多维度使结构体能够支持三维坐标。为了便于操作,想要通过简单的方式直接获取 x 和 y 坐标,不需要复杂的转换。

接着,讨论了如何通过堆叠多个结构体来实现这一目标,使得可以通过简单的输入直接访问 x 和 y。这种设计使得代码更加直观,能够直接获取坐标,提升开发效率。

之后,提到了在处理空间时必须考虑 z 坐标,目前并没有明确的三维矩形概念。尽管对三维矩形的需求还不明确,但还是决定继续开发,逐步进行三维坐标的扩展。

总结来说,主要围绕结构体的扩展,如何通过简化操作使得坐标获取更加直接,并讨论了如何逐步实现三维空间的处理,尤其是对 z 坐标的考虑。

game_math.h: 创建一个 v3 "构造函数",将 v2 和一个 real32 值拼接

讨论中提到了一些构造函数的概念,特别是如何构造一个三维向量。通过传递 x 和 y 值,可以方便地扩展为三维向量。这是一种便于操作的方式,能够轻松构建向量扩展,进而提高开发效率。

接着,讨论了实体类的概念。实体类通常用于表示一些类型的对象,如果更倾向于使用类的结构,实际的类会更好。在这个上下文中,涉及到二维结构的使用,其中可能包括仅有 x 和 y 坐标的二维空间。讨论中也提到,这种结构在实践中能帮助理解如何在开发中通过概念化来管理对象。

总结来说,重点放在了通过构造函数来简化三维向量的创建,以及在类的使用上如何更加方便地管理和扩展数据结构,提升代码的可维护性和扩展性。

game_sim_region.h: 将 sim_entity 中的 ddP 升级为 v3

我们正在尝试在一个三维环境中恢复一个原本在二维的模型。尽管这听起来像是一个巨大的变动,但实际上这应该不会太复杂。我们并不清楚为什么某些特定的参数存在,尤其是ChunkZ的部分,甚至不明白为什么某些数据还保留在那里。我们直接设置了这些数据,但在某些情况下,这并没有太大的意义。虽然我们不完全理解现有的设置,但为了推进进程,我们打算继续进行下去,并尝试让这个过程看起来是合乎逻辑的。

继续解决编译错误

我们正在调整无效的参数,并进行必要的提升和改进。接下来,我们需要处理矢量和编译器错误,通过逐步修改代码并推进开发过程来解决这些问题。我们现在可以进行之前没有做过的改变,并且对结果进行调整,以确保它符合预期。

在进行空间计算时,虽然某些参数与调用的词相同,但我们不期望它们返回相同的结果。接下来,我们会对添加实体的部分进行升级,确保它能够处理新的改动。

尽管在调整过程中我们并没有完全按照矩形轨道的方式进行操作,但我们还是希望能够逐步实现这些修改,并确保最终结果符合要求。我们决定继续处理矩形三的相关内容,甚至不确定是否需要矩形二,但我们会根据需要决定是否保留这些部分。

接下来,我们会检查矩形的相关内容,并在计算时逐步填充必要的数据。

game_math.h: rectangle3

我们正在处理一个矩形的相关代码,并将其移到适当的位置,因为我们已经在代码的顶部进行了定义。该代码与之前的结构非常相似,几乎没有太大变化。我们讨论了使用模板的选项,虽然我倾向于采用现有方法,但最终选择取决于个人经验和优先事项。这一过程本身并不复杂,代码的结构也相对简单明了,修改起来应该比较容易。

无论如何,关键在于根据需求和上下文灵活处理,并做出最合适的选择。

game_math.h: AddRadiusTo 应该接受一个向量作为参数

我们正在考虑是否直接使用一个矢量来进行操作,这样似乎会更简洁、更智能。我不能想到不这么做的理由,尤其是当我们需要加上半径时。接下来,我们决定添加一个安全边缘,特别是在Z轴上的安全边距,虽然目前不完全清楚具体的数值应该是什么,但我们依然继续按照这个思路进行。

我们打算在后续处理中更新这些安全边距,尽管目前对Z轴的具体边距并没有完全的把握,最终目标是确保这一切都符合要求。我们将继续进行测试,纠正错误并调整代码,看看能否解决问题并优化结果。

game_sim_region.h: 升级 MoveEntity 例程为 v3

我们正在处理三维加速,并将所有向量转换为三维,这样可以确保计算的准确性。我们在实现过程中考虑了加入重力的加速度,特别是-9.8的加速度常数,这将自动影响物体的运动。随着进展,我们不再需要手动计算重力的影响,因为重力效果已经集成并自动执行。

接着,我们讨论了与Z轴的碰撞检测,计划在后续进一步优化与Z轴相关的内容,尽管目前的处理方式已经能够有效运作。为了简化计算,我们选择了一些基本的处理方式,例如假设物体的深度值类似于瓦片高度,方便后续的碰撞检测和处理。

在处理直径时,我们发现需要根据物体的宽度和高度来调整碰撞的逻辑,同时考虑到深度的影响,尽管这一部分尚未完全明确。我们通过设置直径和更新碰撞检测来调整实体的行为,确保一切运作正常。

整个过程中的碰撞检测仍然主要是在二维空间内进行,Z轴的处理将会在以后进一步完善。当前的重点是确保所有基本功能能够有效执行,同时为未来的扩展和优化打下基础。

去除 GetCameraSpaceP

继续清理

我们正在调整相机空间的速度计算,并开始模拟与相机视角相关的碰撞检测。首先,我们使用了一个矩形表示相机的范围,并基于米为单位进行计算。虽然目前还不清楚该如何精确处理,但我们简单地设定了一些范围值来确保基本效果。

我们还讨论了速度与加速度的计算,特别是关于Z轴的部分。虽然目前并不打算允许Z轴上的加速,但我们通过直接修改速度的方式,使得对象可以跳跃并在当前状态下保持稳定。这样做虽然能实现一定的功能,但并没有涉及复杂的加速度计算。

此外,我们还调整了实体的定位和空间,尤其是与Z轴相关的部分,确保物体的位置能正确地反映在三维空间中。通过这种方式,我们确保了碰撞检测和模拟效果的基本正确性,并为将来可能的优化和功能扩展留下了空间。

总体来说,这些更改使得当前的模拟和碰撞检测更加简洁有效,虽然还存在一些待处理的细节问题,但整体上进展顺利。

程序现在可以编译

现在,程序已经成功编译,尽管做了很多修改,可能会出现问题,但编译过程已经完成。进行这么大的改变时,通常需要确保每一部分都能正确整合,以避免新的错误。

运行游戏,发现只有我们的英雄,其他什么都没有

现在,虽然成功完成了编译,这是成功的第一步,但目前并没有看到预期的结果。这个问题其实是预料到的,尤其是在做出那次大的改变之后。调试可能需要等到明天,因为时间已经超出了约30分钟,且因为开始时的延迟,调试工作可能需要推迟。

确保矩形边界设置正确

在进行矩形包含测试时,需要考虑到一个问题,即确保矩形的边界是居中排列的。关键是判断矩形是否被正确包含在内。因此,在进行调试之前,首先要确保边界设置正确,以免在调试过程中遇到不必要的麻烦。

黑板:更详细地解释我们要确保的内容

在进行调试时,需要确保正确处理区域和实体的边界。首先,需要确保在添加实体时,正确地使用了它们的位置,并进行适当的计算,尤其是在瓦片位置和边界的处理上。通过检查最小和最大边界值,以及实体是否正确被包含在相应的区域内,确保没有任何偏差或错误。此外,在进行模拟区域和块访问时,特别要注意实体是否被正确捕捉和检查。调试过程中发现,问题可能出在检查实体的过程没有返回任何实际的实体,这通常是因为在填充块时出现了错误。因此,需要进一步调试并检查为什么块未能正确处理或返回实体。

这个bug可能是因为在存储内容到块位置时搞砸了

目前需要做的是检查修改后的世界是否出现了问题,特别是在存储和放置实体的过程中的潜在问题。问题可能出在块空间的映射过程中,或者是某些未更新的内容。尽管应该是直接的操作,但仍然存在一些不确定性,可能遗漏了某些明显的细节。处理这些问题并不复杂,预计会在短时间内解决。接下来,计划在明天继续进行调试,解决剩下的问题并进一步处理。

你考虑过使用宏来传入一个名称和运算符,以便插入简单的向量函数吗?

正在考虑使用宏时,虽然宏本身并不好,但它们在某些情况下可能会有用,特别是在简单的向量函数中。然而,由于宏在处理较长的函数时显得笨拙且不易维护,因此不喜欢广泛使用宏。虽然C语言中的宏支持有限,并且容易出现问题,但它们仍然是一个有用的工具。推荐尝试使用宏处理器来改善宏的使用,因为宏语言本身有许多局限,尤其是在需要更复杂转换的场景中,宏的效果并不理想。

为什么要写 1.0f 而不是 1.0?

过去,编译器比较简单,只能接受单一的数值类型,尤其是浮点数。在早期,如果输入一个双精度浮点数(double)并加上一个小数点,编译器会抱怨精度损失的问题,要求显式地指定浮点数类型(例如加上"f"标记)。这种习惯来源于早期的编译器限制,它们只能处理单一类型的浮点数,并且会警告从双精度到单精度转换的精度损失。

然而,随着编译器的智能化,现在的编译器能够智能地识别类型,知道数字可以是双精度或单精度浮点数,并能够处理类型转换而不造成精度丢失。因此,现在不再需要像以前那样总是加上"f"标记,编译器能够自动处理转换。

这种习惯来自于早期对编译器行为的适应,现在大部分编译器已经解决了这个问题,因此不再需要担心精度丢失警告。

你考虑过写测试吗?

在编写代码时,测试是非常必要的,但并不是在所有阶段都需要花费大量时间进行测试。在代码的早期阶段,特别是当代码尚未准备好发布时,编写测试可能并不值得花费大量时间。因为如果编写和调试测试的时间超过了调试没有测试的代码所需的时间,那么测试反而会浪费时间。

测试变得更加合理和必要的时机是当代码接近发布时,或者当开发者遇到很多问题,花费大量时间调试时。这时候编写测试能够帮助验证代码的正确性,减少未来调试的时间。尽管如此,编写测试并非总是最优先的选择,尤其是在一些阶段,编写和调试测试可能不会带来立竿见影的好处。

对于游戏编程来说,编写有效的单元测试特别困难,因为许多涉及游戏逻辑和行为的部分无法通过传统的单元测试来验证。游戏中的测试往往复杂且难以全面覆盖,尤其是当涉及到随机化行为和复杂的互动时。与其他学科(如文本处理或网络编程)不同,游戏编程中的测试效果相对较少,因为许多行为无法被简单地隔离和验证。

因此,在游戏开发中,测试的收益可能并不像在其他领域那样明显,除非在某些特定的阶段或者问题出现时,测试才变得更加必要。

你能解释一下 Z 轴究竟是什么意思吗?如果是用于分离层次,那么高墙是否应该能穿透上层的 Z 层?

在目前的游戏引擎中,z轴并没有特殊的意义,它几乎被当作一个"第三坐标"来处理。虽然引擎本身的结构可以支持三维,但实际上这个引擎仍然在某些地方回到了二维的实现,特别是为了方便处理一些功能。因此,z轴目前只是作为一个额外的坐标值来使用,而没有体现出传统三维空间中的深度或复杂性。

在未来,可能会在某些特定的情况下赋予z轴更多的意义,尤其是在游戏的不同部分需要表现出真正的三维空间时,z轴可能会变得更有语义性。然而,目前来看,z轴仍然只是一个第三坐标,并没有特别复杂的处理逻辑。

怪物在两层楼之间战斗,爬楼梯上下战斗

当前的引擎设计允许实现跨越多个楼层的实体。例如,玩家可以与怪物的顶部战斗,然后再从地板上爬下来,继续与怪物的底部进行战斗。这种设计让游戏场景变得更加动态和多层次,类似于在两层楼之间进行战斗。

尽管目前还没有看到类似的设计应用于其他游戏,但这种设计显得非常有趣和创新。引擎还没有进行特殊处理来确保这一设计顺利进行,当前的结构尚未做出作弊性的修改或调整。

当引擎进一步发展,可能会开始对某些特定场景进行处理,比如让怪物可以同时在两个楼层间互动,允许玩家在楼梯之间或顶部和底部之间自由战斗。这种设计在2D游戏中比较少见,因此它可能会带来新颖的游戏体验。

你是如何在这么短的时间内快速切换任务的?

快速在不同任务之间切换是通过使用缓冲区实现的。通过在多个窗口中打开内容,可以更加高效地管理和组织任务。例如,利用Visual Studio的功能,可以在同一个缓冲区中处理不同的任务,这使得切换变得更快速。可以将特定功能绑定到快捷键上,这样在进行文件管理时会更加流畅。如果使用时速度较慢,可以通过查找并设置这些快捷键来加快工作流程。这种方法使得处理任务时能够更迅速地获取和切换信息,从而提升工作效率。

需要什么样的电脑才能顺利编码?

编码本身并不需要特别强大的计算机,普通的设备就足够支持开发。很多机器即使已经有几年历史,也能轻松进行编程工作。然而,所开发的游戏类型和复杂度才是决定计算机需求的关键。如果是开发需要大量3D图形处理的游戏,那么就需要更强的硬件支持,尤其是在调试过程中,强大的计算机能够帮助更顺畅地运行和测试游戏。总的来说,虽然编码不要求高性能的计算机,但如果游戏设计复杂、需要高端硬件才能运行流畅,就需要更强的计算能力来支持开发和优化。

将 Python 的知识转到 C++ 里,过渡顺利吗?

将Python知识转移到C++时,最大的挑战在于对内存和指针的直接操作。在C++中,开发者需要更深入地了解如何处理内存,如何管理内存的位置以及如何直接操作内存,这比Python中的内存管理要复杂。Python有更高级的内存管理机制,开发者无需过多关注底层细节,而C++要求开发者更多地考虑内存的位置和使用方式。除此之外,代码本身的结构和逻辑并没有太大差别。对于有Python经验的开发者,转到C++时,最大的差异就在于对内存管理的要求。

你有想过 Z 轴实现后会是什么样子吗?

关于最终实现时z轴的设计,尚未完全确定。对于z轴的具体作用,主要想让它处理上下的运动,例如下沉或上升等,但并不想让z轴过于复杂。当前更关注的重点是让世界保持平坦,同时考虑到与怪物的战斗,可能需要让z轴表现出一些上下的互动。当考虑到战斗时,可能需要观察在顶部和底部的情况,可能会涉及到看到位于自己下方的一些东西。因此,z轴的设计可能会随着实际的尝试和实验进行调整,仍需要进一步的探索和调整。

当你做游戏时,你会专注在一个项目上还是会被合同约束做多个项目?

在游戏开发中,是否同时参与多个项目取决于具体情况。大多数情况下,人们倾向于专注于开发一个单一的产品,而不是同时处理多个项目。这种情况是情境性的,会根据不同的项目需求而有所变化。

你是否经常考虑数据局部性,以减少缓存未命中的情况?

在编程过程中,通常对数据局部性(data locality)的关注程度可能较低,这种情况部分源于数据局部性的重要性在过去几十年间逐渐增加。在早期,数据局部性并不是一个特别重要的问题,但在过去的十年里,它的重要性开始显著增长,成为一个更具主导性的优化方向。

在日常编码时,优化数据局部性并不是首要考虑的内容,而更多是在需要优化性能时才会开始关注。这种方式的好处在于可以将更多精力集中在架构设计上,而不被优化问题过早干扰。然而,设计良好的架构也需要避免做出阻碍后期优化的决定。因此,在某些情况下,对数据局部性的关注可能比当前实际付出的精力更多。

总体而言,编码的目标通常是使代码"相当快",而不是追求极致的速度。与某些专注于优化性能的开发者相比,关注点可能更多地放在代码结构清晰、功能完善上,而非极限优化。即使如此,在设计和优化阶段,确保代码具备良好的优化潜力仍然是重要的一环。

你真的已经不再工作了吗------赚够了钱,现在做这个只是为了乐趣/慈善目的?

编程的时间已经有三十一年了,这是一个漫长而深入的历程。在这一过程中,编程不仅是一项兴趣,也是一份职业工作,每天都需要投入工作,为了生计而努力。虽然并不是出于独立财富的自由状态,而是需要依赖工作来维持生活,并为未来的财务状况而担忧。这种生活方式既包含对职业的热爱,也带有实际生活中的责任和现实考量。

你是如何确定 Z 轴中一个块的内容的?你是否会从块的所有六个面抓取实体?

在处理包含 Z 轴访问的块时,聚集数据需要从块的所有六个方向(上、下、左、右、前、后)进行实体的抓取。这是因为块是一个体积而非平面,需要全面考虑其三维空间中的实体。

为了解决这个问题,需要开发一种方法,能够有效地从所有六个方向收集数据。然而,当前的系统中某些部分被破坏,因此需要进一步开发和修复这一功能。这种修复和完善的任务预计会在接下来的时间中进行,以确保块中的数据能够从所有相关方向正确收集并整合。

黑板:"轴对齐包围盒"

在处理具有 Z 轴访问的世界数据时,当前的做法是通过 Z 切片来收集实体。在过去,整个世界被划分为多个切片,每个切片中包含许多 16x16 的瓦片块。然而,瓦片的具体划分被优化掉了,现在每个块更像是一个 20 米 x 20 米的区域。所有世界数据都以稀疏存储的方式存放在这些块中。

在处理时,系统会遍历所有与摄像机区域重叠的块,提取相关数据,进行处理后再存回去。当前的扩展是在切片概念的基础上,将块的形状改为更接近三维的长方体。这些块通常被称为直角棱柱或者轴对齐边界框(AABB)。这种块的特点是,与 X、Y 和 Z 轴对齐,不能自由旋转,方便进行空间计算和优化。

具体操作上,系统会从这些块中提取任何与所需区域重叠的实体数据。目前在实现上有一些需要改进的地方,例如在提取数据时,矩形判断逻辑需要进一步优化。这些改进属于可以简单实现的部分,因此计划将这些优化加入后续的任务清单中以便及时处理。

Minkowski 包含测试用于模拟区域的开始/更新边界

在当前的系统中,处理模拟区域时,对于实体的包含测试仍以点测试为主。具体来说,当前的实现是检查实体的某个参考点是否位于目标体积内。然而,这种方法忽略了实体本身的体积,可能导致不准确或遗漏的情况。

更理想的处理方式是将测试范围扩展为体积测试。具体步骤包括:首先,根据实体的维度扩展收集区域,然后测试实体的整个体积是否与目标区域有重叠。这种方法可以更准确地确定实体与目标区域的交集,而不仅仅依赖点测试。

由于扩展测试范围对当前的性能影响很小,因此建议将其纳入优化计划中。这种改进不仅能够提高包含测试的精确度,还能为后续的区域处理提供更可靠的基础数据。

你认为 RK4 积分对于像 game Hero 这样的 2D 游戏来说是"过度"的吗?

在讨论游戏中是否适合使用四阶Runge-Kutta方法(RK4)时,指出这种数值积分技术在游戏开发中通常并不适用。RK4是一种数值积分方法,主要用于处理复杂的物理模拟。然而,对于大多数游戏场景而言,RK4可能会引入额外的动量,从而导致系统的不稳定,如数值震荡、动量积累过多或其他异常行为。

更适合游戏的积分方法通常是那些能够抑制多余动量、保持系统稳定的保守方法。例如,反向欧拉法(Backward Euler)或其他减动量的积分技术,这些方法有助于防止动量积累导致的系统振荡或失控,并能更可靠地模拟物理系统的逐渐收敛。

历史经验表明,RK4的主要缺点包括其计算复杂度高、对资源消耗较大,以及不适用于需要平稳收敛的游戏场景。从实际开发的角度来看,优先选择更简单、更高效的积分方案(如反向欧拉法)往往能够在性能和稳定性之间实现更好的平衡。

总结而言,游戏中选择数值积分方法时,应结合具体需求,避免使用不必要的复杂技术,除非对其有深刻的理解,并且项目需求确实要求实现更精确的复杂模拟。

你考虑过有两个区域吗------一个外部区域和一个内部区域,包含进入内部的实体,排除进入外部的实体------这样就有一个"边缘",可以在不立即触发进出线的情况下进行一些移动?

在讨论双区域系统时,提到了一种方法:通过内外两个区域的设置,允许实体从内部区域移动到外部区域,同时避免实体直接从外部进入内部。这种机制旨在避免跨越明确边界时出现问题。其核心理念是引入一定的缓冲区域,从而实现更平滑的实体移动和更高效的区域管理。

当前的实现已经具备这种机制。具体来说,设置了一个"更新区域"(update zone)和一个"非可用区域"(non-available zone)。在外部区域内,允许实体从内部移动到外部,但不允许外部的实体进入。这种设置的主要目的是优化系统性能,而不是仅为了处理边界问题。

该优化策略的核心在于减少需要检查的实体集合数量。通过调整区域对齐,可以确保每次计算时只处理必要的范围,避免额外的计算和重复的检查。同时,这种区域划分也减少了涉及的块(tile chunk)数量,进一步优化了性能。

在具体实现中,强调了实体的大小对区域划分的重要性。为了避免实体跨越边界,这种对齐方式将边界调整到合适的位置,从而减少对多个区域的检查需求。这不仅提升了效率,还确保了边界管理的稳定性。

总的来说,这种双区域机制结合缓冲区域的设计,是为了在性能和准确性之间找到平衡。通过减少多余的计算区域和明确边界对齐,可以更高效地管理实体的移动和区域划分。

仓库: https://gitee.com/mrxiao_com/2d_game

相关推荐
PyAIGCMaster3 小时前
docker学习记录:本地部署mongodb
学习·mongodb·docker
架构文摘JGWZ3 小时前
一键完成!!网页打包成桌面应用
开发语言·学习·开源软件·工具
灵魂画师向阳4 小时前
【CSDN首发】Stable Diffusion从零到精通学习路线分享
人工智能·学习·计算机视觉·ai作画·stable diffusion·midjourney
菠菠萝宝4 小时前
【Go学习】-01-1-入门及变量常量指针
开发语言·学习·golang·go·软件工程·web·go1.19
跳河轻生的鱼6 小时前
海思Linux(一)-Hi3516CV610的开发-ubuntu22_04环境创建
linux·单片机·学习·华为
跳跳的向阳花6 小时前
02、Docker学习,理论知识,第二天:基础概念与常用命令
学习·docker·容器
PyAIGCMaster6 小时前
Docker学习记录:安装nginx
学习·nginx·docker
Lumos_yuan7 小时前
Lumos学习王佩丰Excel二十四讲系列完结
学习·excel·教程总结
东京老树根7 小时前
Excel 技巧02 - 如何批量输入百分号 (★),如何输入百分号并指定小数位数,如何批量删除百分号,如何批量删除小数最后的0?
笔记·学习·excel·vba
don't_be_bald7 小时前
数据结构与算法-顺序表
c语言·开发语言·数据结构·学习·链表