我们上次讨论的是模拟区域
我们刚刚完成了代码更新,采用了模拟区域。虽然完成了更新,但我们还没有机会彻底调试它。实际上,有几个功能仍然需要添加,以确保它能够完整运行并完成所有所需的操作。因此,今天我们需要进行一些清理工作,确保新代码正常运行,并添加一两个小功能,以确保它可以实现我们需要的所有功能。
模拟区域的限制概述 - 实体当前必须有位置
目前,我们有一个模拟区域,但它并不支持实体在某个位置"不存在"。换句话说,游戏中已经有了一个类似的概念,比如英雄携带的剑,它在世界中并不占据特定位置,只有在使用或摆动时才会"出现"。这个概念虽然还没有完全实现,但它非常有潜力,尤其是在渲染方面,能够让实体根据需要在世界中"出现"或"消失"。
为了在模拟区域中支持这个操作,我们希望能有一种标志,或者某种机制,让我们在将实体加载到模拟区域时,能够平稳地处理它在空间上的不存在性。具体来说,这些实体只在逻辑上存在,不参与空间模拟。我们需要确保在模拟区域的更新过程中,能够正确地跟踪这些实体的状态,确保它们不会参与空间上的碰撞或其他模拟,但在需要时能再次被加载和更新。这是一个我们尚未深入探讨的概念,因此我们对如何实现它的要求还不完全明确。
添加 sim_entity_flags
我们已经有了碰撞的概念,因此现在我们考虑是否可以引入实体标志(flags)的概念。标志是用于标记实体特定条件的一种方式,它们提供了一种简单的小存储空间,可以方便地跟踪实体的状态。我们可以使用这些标志来检查实体是否符合某些特殊条件,比如是否处于某种状态。
我们计划为实体创建一个枚举类型,用于保存这些标志。尽管可以使用宏定义(#define)或者数字值来实现,但是使用枚举能够更清晰地组织标志。每个标志代表一个布尔条件,这样我们就可以通过检查这些条件来确定实体的状态。
例如,实体可能会被标记为"非空间的",意味着它不参与任何空间操作。在这种情况下,实体的位置值将变得没有意义。为了提高代码的可读性,可以采用将标志逐渐增加的方式,而不是按位移位的方式,这样可以让代码更加易于理解。
此外,还需要处理一些清理工作,比如处理Z轴的相关问题,虽然目前尚未详细考虑。未来的工作可能还包括加入像楼梯这样的元素,用于处理实体上下移动的情况。这些任务将有助于更好地管理实体及其空间状态。
更新代码,将移动碰撞成员变量移入标志中
首先,通过编译程序,可以开始实施更改并检查碰撞的发生位置。接下来,可以决定如何处理位置的计算。为此,可以通过检查每个标志来处理不同的方式。一种方法是使用一个检查函数来查看某个标志是否设置,也可以采用更简单的方式,通过传递标志值来直接检查实体的标志状态。
在处理这些标志时,考虑到标志值可能是多个标志的组合,需要检查类型的正确性。为了避免将标志合并为整数值,在传递标志时使用单独的设置函数进行检查。这个设置函数接收实体和标志值,并返回对应的结果。
在实际实现中,通过与标志值进行按位与运算,可以确定标志是否已设置。如果标志值为0,则表示未设置,非零值则表示标志已设置。这样,功能简单且易于理解。
此外,在一些情况下,可能会为实体添加新的标志,如添加碰撞标志。在这种情况下,可以通过设置标志来检查实体是否参与碰撞操作。可以使用标志设置函数,在设置时避免手动指定每个标志值,这样可以提高代码的可读性。
在标志设置过程中,通常标志从零开始,只需要在需要时打开特定标志。在代码中使用这种方法时,添加标志值通过按位或操作符来进行。
最终,通过确保标志的正确设置和清晰的代码结构,可以更容易地管理和控制实体的不同状态。
使剑变为非空间性的
在处理剑的情况时,由于剑是非空间实体,因此我们需要为其设置一个非空间标志。通过在低级实体的模拟(sim)中添加 EntityFlag_NonSpatial
标志,可以确保它被标记为非空间的实体,这意味着它并不存在于当前世界中。
我们知道,实体的位置是通过改变其位置来决定是否为空间实体或非空间实体的。如果没有新的位置,我们将会添加非空间标志;如果有新的位置,则会清除该标志。这样可以确保标志始终根据实体的位置正确设置。
该机制主要用于区分实体是否应该被视为空间实体,或者它是否处于非空间状态。这个过程简单而直观,确保在流式传输实体进出时,标志状态会被正确处理。
在 MoveEntity 中处理非空间实体
在模拟区域中,我们可以开始处理这些问题。首先,考虑到移动实体的情况,显然我们需要确保没有人试图移动一个已经设置了非空间标志的实体,因为这没有意义。因此,我们需要在此时做一个断言,确保实体没有设置非空间标志。这样做是为了避免发生不合理的操作。
类似地,我们应该假设碰撞(collides)是权威的标志,但我们也可以进一步确保这两个标志都被正确设置。具体来说,我们希望确保非空间标志没有被设置,碰撞标志也应该被设置。如果这两个条件不符合,我们可以设置断言,确保这些标志的状态符合预期。虽然我们可以假设编译器足够智能,能够处理这些情况,但在测试实体时,我们可能需要考虑这些细节,确保标志状态始终符合要求。
对于一些不太明确的标志设置问题,我们可以考虑是否需要自己实现一种"崩溃检测",简化这种标志状态的测试,或者就交由编译器来处理。这些细节可以在未来的工作中进行优化和调整。
在 BeginSim 中处理非空间实体
在处理实体时,我们需要检查是否存在非空间实体的情况。如果一个实体被标记为"非空间",那么它不应该被包含在矩形检查中,因为它不可能位于矩形内。因此,在进行矩形检查时,只需考虑那些没有设置"非空间"标志的实体。只有在实体没有设置该标志时,我们才会继续进行相关的处理。
在 EndSim 中处理非空间实体
在处理实体时,对于非空间实体,不应执行块空间的映射。相反,当映射这些实体时,应该将它们的位置设置为"空位置"。这种做法确保了在将实体数据流入或流出时,不会错误地将其位置映射到空间坐标。这样做是为了保证实体状态的准确性,尤其是在需要对空间和非空间实体进行区别处理时。
在 AddEntity/LoadEntityReference/GetSimSpaceP 中处理非空间实体
在处理实体时,问题出在当实体被引用时,系统会将其加载并引入,即使它是非空间的实体。例如,如果某个角色持有一把剑,那么剑也会被加载,这种情况可能会导致一些问题。
具体来说,当前的实体引用加载机制存在一个问题。特别是当加载实体时,系统调用了一个版本的"AddEntity"方法,这个方法并没有传递位置(position)。这种情况下,实体并没有被正确地处理,导致潜在的错误。
解决方案是,在加载实体时,需要使用正确的"AddEntity"版本,它应该接收位置参数,而不是当前的版本(即不传递位置)。这样可以确保在加载实体时,能够正确地设置位置。如果实体是非空间的(即它不需要位置),那么位置可以被设置为NullPosition
或其他标识性的无效位置。
在调试模式下,可以考虑将实体的位置设置为一个特殊值,如NaN
,来检测是否有误用非空间实体的位置值。这种方法可以帮助开发人员在调试过程中及时发现问题。
通过这样做,实体的加载过程可以确保正确处理空间实体和非空间实体的位置,避免错误发生。同时,采用特殊值来标识无效位置,有助于在开发过程中进行调试和验证。
在 UpdateSword 中处理非空间实体
当前的任务是调试并完善一些实体更新的逻辑。具体而言,在更新剑的逻辑中,发现之前没有正确处理非空间实体。对于更新剑的过程,如果实体处于非空间状态,那么实际上不需要做任何处理。此时只需检查是否设置了"非空间"标志。如果该标志设置了,就不需要进行位置更新,否则就按正常流程更新剑的位置。
另外,需要处理一些关于实体存在与否的情况,确保实体的"空间状态"能够正确设置。如果实体已经完成了所有的动作或移动,可以将其标记为非空间实体,以便它不再占用空间。在这过程中,还可以考虑在调试模式下通过设置一个特殊值(例如信号 NAN)来标记非空间实体的位置,以确保它们不会被误用。
接下来,可能需要考虑如何确保其他的实体和更新操作也符合相应的逻辑,例如检查速度是否需要重置,以及如何正确传递实体的起始位置等。
最后,更新逻辑中还需要考虑如何处理位置无效的情况,可以使用预定义的无效位置值来标记实体的无效状态。通过这种方式,能够确保不需要额外的处理或错误,保持代码的清晰性和稳定性。
改进 ChangeEntityLocation
在处理实体的空间与非空间状态时,我们注意到在某些情况下,更新实体的位置时出现了断言问题。具体来说,当剑的状态变为非空间时,它的位置应当保持有效,而不是变成无效块。我们希望确保这种更新可以顺利进行,但在实现过程中,我们遇到了不必要的复杂性,例如必须检查是否空间化,并处理位置为空的情况。
为了简化这个过程,我们决定重新设计这一逻辑:当更新实体的位置时,可以直接传递一个新的有效位置,而不需要使用复杂的指针传递。通过这种方式,如果位置有效,我们将其设置为新的位置;如果无效,则保留为空。同时,更新实体的空间状态时,只有在标志未设置时,才将其位置设为零,否则将使用现有的位置。这种改进将使得代码更加简洁且易于理解。
此外,更新后的代码设计不再需要额外的忙碌工作,例如反复检查实体是否空间化或不空间化。通过这种方式,我们能够更加高效地处理实体的位置更新,并减少冗余代码。
调试 GetWorldChunk 断言 - 存储无效位置和非空间标志的问题
修复相机跟随逻辑
我们目前的任务是清理和修复相机跟踪实体的问题。首先,我们发现相机没有正确跟随实体,这可能是因为在添加玩家时,尽管将相机跟随索引设置为玩家,但没有正确更新相机位置。
在代码中,我们发现了用于更新新相机位置的逻辑。然而,这部分代码并未触发实际效果。具体原因在于,CameraP
(相机位置)没有被正确写回。之前我们有一个方法调用负责更改这一值,但现在这部分逻辑似乎缺失了。因此,我们尝试直接将新的相机位置设置为 CameraP
,以确保更新逻辑能够生效。
此外,在测试过程中,我们注意到实体在跳跃时出现了问题:一旦开始跳跃,实体会一直漂浮在空中,无法停止。经过检查,发现这是由于 DZ
(垂直方向速度)未被正确更新所导致的。我们需要进一步检查这一问题,并修复相关逻辑,以确保实体的运动能够正常进行。
通过以上分析和初步修复,我们解决了部分问题,但仍需进一步验证代码是否按预期运行,特别是相机位置更新和实体运动的同步逻辑是否完全正确。接下来,我们将继续完善这些功能,确保整体系统运行顺畅。
一些修改
修复玩家的 dZ
当前相机的DZ值出现了问题。尽管之前没有真正处理过与Z相关的内容,但仍然希望确保其正常工作。通过分析,我们注意到DZ值应该通过某些路径被更新。然而,观察代码时发现,控制器的相关输入并未被清理,这显然导致了问题。
我们正在存储控制器的输入,以便后续进行一些操作。然而,因为输入未被清理,这造成了DZ值无法正确更新的问题。接下来,我们尝试对控制器的DZ值和其他相关的D值进行清零操作。这一步是为了确保没有残留的旧值干扰当前的处理逻辑。
在实现清零后,还需确保只有当控制器的DZ值被明确设置时,实体的DZ值才会被更新。换句话说,仅当控制器的DZ值不为零时,我们才会将其用于更新实体的DZ值。这种方式修复了当前的跳跃控制问题。
尽管上述修复方法有些粗糙,但暂时恢复了跳跃控制功能。接下来可以在未来进一步优化这些逻辑,使其更加优雅和稳健。
玩家的运动状态一直没有更新
恢复剑的功能
在修复上述功能后,计划重新实现"剑"的功能。目前"剑"仅仅是一个简单的占位对象(如石块),但为了进一步优化,我们需要对"剑"进行空间化(spatial
)处理。这意味着在逻辑中需要为剑设置位置(Entity->P
)和移动方向(DP
),从而完成"剑"与其他实体的交互逻辑。
未来还有更多需要优化的部分,例如提升排序算法,以及逐步完善游戏中的核心战斗功能。这些将作为后续开发的重点方向。
为什么剑的阴影那么暗? - 多把剑
我们正在处理一个需要将实体空间化的问题,以确保其正确参与模拟系统。分析后发现,虽然看起来逻辑实现没问题,但阴影的显示似乎不对。阴影显得过暗,或者可能被多次绘制。我们怀疑阴影的重复显示可能是由于实体被多次加载或添加造成的。
为了解决问题,我们决定设置一个循环点来追踪相关行为。我们暂时注释掉了与阴影相关的代码,发现剑实体似乎确实被重复绘制了。为了验证这一点,我们尝试通过添加实体索引作为偏移量来定位多个重复绘制的剑。果然,出现了多个实体索引,说明确实存在重复加载的情况。
接下来,我们将深入检查实体系统,特别是模拟区域中实体的加载和引用逻辑。初步判断可能是由于引用系统在加载过程中存在问题,导致相同的实体被多次写入或绘制。
我们准备到手动实现的模拟区域代码中,追踪和分析多个实体的问题。尽管最初的设计包括一个哈希表来避免重复加载,但实际运行中仍然出现了重复问题。我们需要验证加载逻辑,以确保每个实体只被加载一次,同时对相关代码进行全面的检查和修复。
目前确认的是,我们在整个过程中只创建了一把剑,这意味着问题很可能出现在模拟区域的引用系统或加载逻辑中。接下来的重点是找出根本原因并解决问题。
防止模拟区域中有多个实体具有相同的存储索引
我们进行了一些调整来改进 Sim 区域代码,目的是防止多个实体在 Sim 区域中表示相同的存储索引。以下是具体的改进细节:
-
增加调试标志 :
在实体结构中添加了一个调试标志(例如
Simming
),用于标记实体是否已经加载到 Sim 区域。- 在
BeginSim
操作中,当加载某个实体时,我们会设置该标志以表明该实体已加载。 - 在加载之前,我们添加断言,确保该标志尚未设置,这样可以捕捉重复加载的情况。
- 在卸载实体时,我们会清除该标志,并在清除前断言标志是否正确设置。
- 在
-
修复重复加载问题 :
发现问题的原因在于,当使用空间查询和引用同时加载实体时,可能会导致重复加载。
- 原本仅防止引用加载已经通过其他引用加载的实体,但未防止空间查询加载已经被引用加载的实体。
- 为了解决这个问题,我们在加载前增加检查,通过哈希表确保实体未被重复加载。
-
优化哈希表操作:
- 在查找和加载实体时,将所有的哈希表检查整合到一个统一的流程中。
- 在加载时,若哈希表中已存在目标实体,则直接使用,无需重复查找。
-
清理无效代码:
- 清除了调试过程中为验证实体加载状态而引入的临时代码,确保最终逻辑保持清晰。
这些改进使得 Sim 区域的实体加载更为可靠,避免了重复加载导致的潜在问题。同时,通过断言和调试标志可以更早地发现和定位错误,从而提高开发效率。
防止在按住键时剑被重置
我们继续解决了一个问题,确保当按住键时,剑不会被重复重置。我们通过检查剑的状态来实现这一点。如果剑已经存在,我们就不再重新创建它,这样用户就无法重复触发剑的生成。为了实现这一点,我们检查了剑是否处于非空间状态,如果是这样,就表示剑当前并不存在,允许它再次被激活。
之后,修复了其他小问题,确保模拟区域能够正确工作,并且能够处理实体引用和加载。现在模拟区域运行得很好,支持引用和拉取实体。
接下来,我们计划处理围裙的相关工作,这将在明天完成。我们目前认为其他的更新都已经完成,没有什么其他需要特别处理的事情。
当前的碰撞代码是否支持非90度的墙壁?比如如果墙壁是45度角,碰撞代码是否仍然有效?
目前的碰撞代码支持检测墙壁与实体的碰撞,但仅限于水平或垂直的墙壁。如果墙壁的角度为45度,当前的代码无法处理这种情况。为了支持不同角度的墙壁,必须修改代码来计算射线与墙壁的交点时间。这需要通过编写额外的代码来处理,尤其是在处理与45度角或其他角度的墙壁碰撞时。实现这一功能的数学原理很简单,但目前还没有完成这一工作。
碰撞代码的基本原理是检测实体是否与墙壁发生碰撞,并防止实体穿透墙壁。然而,这段代码在处理数值精度时存在问题,尤其是在计算交点时。如果计算结果非常接近墙壁的边缘,实体可能会在下一帧进入墙壁内部。这是因为在数值精度计算时,可能会出现精度误差。
总的来说,这段代码的逻辑是有效的,可以用于检测实体与墙壁的碰撞,不论墙壁的角度如何。然而,要完全支持多角度墙壁碰撞,并确保数值精度处理得当,仍然需要做一些改进。
为什么不设置一个布尔值"待删除"?将位置设置为任意位置似乎不如设置一个标志明确表示该实体应在下一个更新帧删除。
在处理实体是否被移除时,当前的做法是通过设置一个标志位来表示实体是否应该在下一次更新时被移除。虽然位置被设置为非空间状态(non-spatial
),但实体并没有被删除,而是仍然存在,只是其空间位置被设置为无效。这样做的原因是为了避免实体仍然在空间中存在,但没有正确的空间位置。
通过将实体标记为非空间,实际上是将其从空间计算中移除,而不是删除实体。这个标志位的作用是确保实体在下一次更新时不会被处理。为了进一步调试,可以设置一个无效的P
值,确保不会有其他代码继续使用这个无效的实体位置,防止在它变为非空间状态后仍然被处理。这样做可以更容易地发现问题,确保没有错误的实体被处理或更新。
因此,虽然实体并没有被删除,而是设置为非空间状态,但通过这种方式可以有效地管理实体的状态,防止它们在不再需要时继续占用空间。
代码的重新排列是否引入了一些错误?这在日常编码中是正常的吗?
在进行代码重排时,出现错误是正常的,尤其是在复杂系统中。代码重组时,常常会引入一些小错误,但这也是开发过程的一部分。在日常的编码过程中,尤其是当系统变得复杂时,移动代码和重新组织代码时,出现错误是不可避免的。关键是要学会如何处理这些错误,避免对重构代码产生过多的恐惧。
在这种情况下,虽然错误发生了,但通过一些调试方法和检查措施,可以有效地避免错误的蔓延。比如在系统中添加断言和调试检查,有助于验证假设并及时发现问题,这比单元测试更为适用,因为单元测试在游戏开发中难以涵盖所有的情况。
此外,学习编写能够在出错时显而易见的代码是非常重要的。通过设置检查点或断言,可以在代码发生微妙错误时迅速发现并解决问题。这些方法能够帮助开发者在面对复杂和高性能的游戏机制时,有效管理复杂性,确保代码的健壮性和可维护性。
总的来说,虽然重排代码时会引入错误,但通过不断学习和运用调试技巧,能够提高处理这些错误的能力,确保代码在重构后能够更好地工作。
我们是否预期大多数实体都会发生碰撞?如果是这样,是否可以将标志设置为"非碰撞"而不是?
在讨论碰撞实体时,提出了一个问题,即是否应该设置一个标志,指示实体是否正在碰撞。考虑到大部分实体会发生碰撞,有人建议是否可以将这个标志设置为非碰撞标志,这样可能更具意义。虽然这不是一个坏建议,但在考虑所有实体类型时,尚未明确所有需要的实体类型,因此这个问题还需要进一步思考。
总体来说,讨论的重点是是否应该将标志设置为非碰撞实体,这个决定需要根据实际情况来决定。
那为什么要设置为空位置?
设置实体位置为空的位置的原因,但问题没有明确说明设置空位置的具体情况。询问是否是在同一区域内设置空位置。
使用标记联合体来引用实体,是否能解决我们在 LoadEntityReference 中遇到的问题?
讨论了使用标记的联合(tagged union)来解决与实体引用加载相关的问题,但没有明确指出具体遇到的问题。
你在每个阶段是否都清楚整个游戏架构的宏观图,还是需要不时检查自己做的事情?
讨论了关于保持整体游戏架构的思路,通常可以记住完整的大局,但有时需要检查之前做的工作。由于每天只一小时,可能难以完全记住所有细节,特别是在离开项目一段时间后再回到时。提到即使在重新整理代码时也能遇到问题,并需要不时检查。此外,提到了一些方法,例如通过断言和调试检查来帮助验证假设并捕捉潜在的问题,而不是依赖单元测试,因为在游戏开发中编写全面的单元测试比较困难。