数据驱动设计(DOD)以数据布局和访问模式为核心,通过SoA或AoS2结构提升缓存友好性与批量处理效率,常与ECS协同落地。
C++ 数据驱动设计(Data-Oriented Design,DOD)不是“用数据控制逻辑”,而是把“数据布局和访问模式”放在设计首位,让代码围绕高效内存访问来组织。它不反对面向对象,但明确拒绝为抽象而抽象——如果一个 std::vector 里每个 Player 都含 std::string name、std::vector 和虚函数表,那它大概率不适合 DOD 场景。
DOD 的出发点很实际:现代 CPU 远快于内存,瓶颈常在“等数据从 RAM 加载到 L1 缓存”。一次随机读取可能耗时 300+ 个周期,而连续读取同一缓存行(64 字节)里的多个字段,几乎无额外开销。
position.x 放一块内存,position.y 放另一块,而非每个玩家结构体里混着位置、生命值、朝向等字段player->weapon->damage),改用索引或直接数组偏移传统面向对象常用 AoS(Array of Structs):struct Player { vec3 pos; flo —— 每个元素是完整对象,但不同字段分散在内存中。
at hp; int id; } players[1000];
DOD 常用 SoA(Structure of Arrays) 或其变种 AoS2(Array of Small Structs):
vec3* positions; float* healths; int* ids; —— 同类数据连续,遍历时 cache line 利用率高struct PlayerChunk { vec3 pos[64]; float hp[64]; int id[64]; }; —— 折中方案,兼顾局部性和向量化,也便于分块管理(如 ECS 中的 archetype)std::vector(连续内存)+ 索引映射(如 EntityID → index),避免 std::map 或裸指针不必推翻现有代码。从性能敏感模块切入,例如渲染器的顶点处理、游戏逻辑的物理更新、AI 的感知计算。
movss、vmovaps 或 cache-miss 高,DOD 就值得尝试std::span 或自定义 view 类封装,保持接口清晰std::find_if、dynamic_cast、跨 chunk 随机索引;用预排序、位掩码(如 active_mask)、或 ECS 组件存在性查询替代entt(轻量 ECS 库)、boost::hana(编译期元编程组织字段)、或手写代码生成器(根据 JSON schema 自动生成 SoA 结构体)ECS(Entity-Component-System)本身不是 DOD,但它提供了极佳的 DOD 落地容器:组件(Component)天然对应“数据字段”,系统(System)对应“批量操作”,实体(Entity)只是 ID。这种分离让 SoA / AoS2 实现变得自然。
PositionComp 全部连续,VelocityComp 全部连续),系统遍历时可轻松对齐、向量化PlayerState 拆成 Health、Stamina、StatusEffects 更利于独立更新与缓存复用Health 数组,“死亡系统”下一帧读取并清理;必要时用 ring buffer 或 job queue 解耦时序