【3D-AICG 系列-15】Trellis 2 的 O-voxel Shape: Flexible Dual Grid 代码与论文对应

系列文章目录


文章目录

  • 系列文章目录
  • [Mesh -> O-voxel](#Mesh -> O-voxel)
    • 总目标
    • [关键输入(只有 4 个需要关心的)](#关键输入(只有 4 个需要关心的))
    • [关键输出(只有 3 个)](#关键输出(只有 3 个))
    • [关键算法:只有 **2 步**](#关键算法:只有 2 步)
      • [第 1 步:收集约束("哪些体素被穿过、穿过时的几何信息是什么")](#第 1 步:收集约束("哪些体素被穿过、穿过时的几何信息是什么"))
      • [第 2 步:求解("在每个体素内找最佳代表点")](#第 2 步:求解("在每个体素内找最佳代表点"))
    • 一张图串起来

  • 前文有介绍 Trellis 2 的 Flexible Dual Grid 整体流程,本文则聚焦于将这部分的 C++ 代码细节和论文对应起来。

  • 代码来自 /repo/TRELLIS.2/o-voxel/src/convert/flexible_dual_grid.cpp

    /repo/TRELLIS.2/data_toolkit/dual_grid.py 中的 o_voxel.convert.mesh_to_flexible_dual_grid 函数调用

Mesh -> O-voxel

跳出细节,看大局。整个 flexible_dual_grid.cpp 文件只干一件事


总目标

输入一个三角网格(mesh),输出一组稀疏体素,每个体素内有一个"最佳代表点"(dual vertex)。


关键输入(只有 4 个需要关心的)

输入 含义
vertices (N,3) 网格顶点坐标
faces (M,3) 三角面的顶点索引
voxel_size (3,) 每个体素在 x/y/z 方向的尺寸
grid_range (2,3) 体素索引的范围 [min, max)

关键输出(只有 3 个)

输出 含义
voxel_indices (K,3) int K 个活跃体素的整数坐标
dual_vertices (K,3) float 每个体素内的最佳代表点坐标
intersected (K,3) bool 每个体素在 x/y/z 三个方向上是否被表面穿过
  • 活跃体素 = Mesh 表面与体素网格有交的体素
  • voxel_indices (i, j, k):表示"第几格",只能是整数。
  • dual_vertices (x, y, z):表示"这一格里的点在世界/局部坐标下的具体位置",可以是格子内的任意一点。并且 QEF 求出来的解是连续坐标,例如 (2.3, 1.7, 0.8),表示在局部坐标系下 x=2.3、y=1.7、z=0.8 的点,所以必须用 float 才能存这些小数。

关键算法:只有 2 步

第 1 步:收集约束("哪些体素被穿过、穿过时的几何信息是什么")

由 3 个函数完成,每个函数只做一件事------往 QEF (quadratic error function) 矩阵里加一项约束

函数 做什么 一句话
intersect_qef 扫描线找"三角面穿过了哪些体素边",记录交点和法平面 发现活跃体素 + 加"交点平面"约束
face_qef 遍历三角面的包围盒,找与三角面重叠的体素 加"面片平面"约束(让代表点靠近面)
boundry_qef 找只被一个三角形引用的边(开边界),DDA 遍历经过的体素 加"边界线"约束(让代表点靠近边界)

三个函数的共同点:都在往同一个 qefs[i](4×4 矩阵)里 累加,本质是在说"这个体素的代表点应该尽量满足这些几何约束"。

第 2 步:求解("在每个体素内找最佳代表点")

就是 562--765 行的循环:

复制代码
对每个活跃体素:
    Q = 累加好的 QEF 矩阵(+ 可选正则项)
    先无约束解 Ax=b
    如果解在体素内 → 用它
    如果解跑到体素外 → 依次试"固定1个面/2个面/3个面(角点)",取误差最小的
    → 得到 dual_vertex

一句话 :在体素的小盒子里,找一个点,使它离所有穿过该体素的三角面尽量近


一张图串起来

复制代码
mesh (vertices + faces)
        │
        ▼
  ┌─────────────────┐
  │  intersect_qef  │  扫描线 → 发现活跃体素 + 累加 QEF
  └────────┬────────┘
           ▼
  ┌─────────────────┐
  │    face_qef     │  面片重叠检测 → 累加 QEF(可选)
  └────────┬────────┘
           ▼
  ┌─────────────────┐
  │  boundry_qef    │  边界 DDA → 累加 QEF(可选)
  └────────┬────────┘
           ▼
  ┌─────────────────┐
  │   QEF 求解循环   │  每体素解 min ||Ax-b||² → dual_vertex
  └────────┬────────┘
           ▼
  (voxel_indices, dual_vertices, intersected)

所以你只需要记住 :前面 3 个函数是"收集约束、往 QEF 矩阵里加东西",最后一个循环是"解方程、得代表点"。其它所有代码(扫描线怎么扫、投影怎么测、DDA 怎么走、约束求解的分支枚举)都是这两步的实现细节

相关推荐
China_Yanhy13 小时前
动手学大模型第一篇学习总结
人工智能
空间机器人13 小时前
自动驾驶 ADAS 器件选型:算力只是门票,系统才是生死线
人工智能·机器学习·自动驾驶
C+++Python13 小时前
提示词、Agent、MCP、Skill 到底是什么?
人工智能
小松要进步13 小时前
机器学习1
人工智能·机器学习
Mr_Xuhhh13 小时前
Java泛型进阶:从基础到高级特性完全指南
开发语言·windows·python
汀、人工智能13 小时前
[特殊字符] 第40课:二叉树最大深度
数据结构·算法·数据库架构·图论·bfs·二叉树最大深度
沉鱼.4413 小时前
第十二届题目
java·前端·算法
泰恒13 小时前
openclaw近期怎么样了?
人工智能·深度学习·机器学习
KaneLogger13 小时前
从传统笔记到 LLM 驱动的结构化 Wiki
人工智能·程序员·架构
烟锁池塘柳014 小时前
一文读懂UGC、PGC、PUGC、OGC、MGC、BGC与AIGC
aigc