
在使用 gremlin 进行图数据操作时,我们有时需要删除一个特定顶点及其通过不同关系连接的相关顶点。union() 步骤是一个强大的工具,可以合并来自不同遍历路径的元素流。例如,为了删除一个 identity 顶点及其关联的 subscription 和 channel 顶点,一个直观的 gremlin 查询如下:
g.V().hasLabel('Identity').has('phones', '+11234567890').union(
identity(),
__.out('Receives').hasLabel('Subscription'),
__.out('MemberOf').hasLabel('Channel')
)当此查询单独运行时,或在其后添加 elementMap() 步骤以查看属性时,它会正确地识别并返回所有预期的三个顶点(一个 Identity 顶点,一个 Subscription 顶点,一个 Channel 顶点)。
gremlin> g.V().hasLabel('Identity').has('phones', '+11234567890').union(
identity(),
__.out('Receives').hasLabel('Subscription'),
__.out('MemberOf').hasLabel('Channel')
).elementMap()
==> // 打印出所有3个顶点的属性,证明union()正常工作然而,当尝试直接将 drop() 步骤应用于这个 union() 后的遍历时,我们观察到一个意料之外的行为:只有 union() 产生的第一个顶点(在本例中通常是 Identity 顶点,因为它来自 identity() 步骤)被删除,而后续的 Subscription 和 Channel 顶点则保留在图中。
gremlin> g.V().hasLabel('Identity').has('phones', '+11234567890').union(
identity(),
__.out('Receives').hasLabel('Subscription'),
__.out('MemberOf').hasLabel('Channel')
).drop()
// 实际结果:仅Identity顶点被删除,Subscription和Channel顶点保留这与我们通常对 drop() 行为的理解不符,例如,g.V().hasLabel('SomeLabel').drop() 会删除所有匹配 SomeLabel 的顶点。这种差异表明 union() 步骤与 drop() 结合时可能存在特定的流处理机制。
这种现象表明,在某些Gremlin版本或特定图数据库实现(如Neptune 1.1.1.0)中,union() 步骤产生的遍历器流在遇到 drop() 时,可能没有完全“展开”或“物化”所有元素。drop() 步骤可能在处理完从 union() 接收到的第一个遍历器后,就结束了对该流的处理。这可能是Gremlin内部流处理模型的一个微妙之处,甚至可能是一个已知的或未解决的TinkerPop bug。
为了解决 union().drop() 无法删除所有指定顶点的问题,一种有效的解决方案是在 drop() 步骤之前插入 fold().unfold() 序列。
通过这种方式,fold().unfold() 强制 Gremlin 在 drop() 之前将 union() 产生的所有顶点“物化”到一个列表中,然后再将它们重新发射为独立的遍历器。这样,当 drop() 步骤执行时,它接收到的是一个完整的、经过明确定义的顶点集合流,从而能够对所有期望的顶点执行删除操作。
以下是应用 fold().unfold() 解决方案后的 Gremlin 查询:
g.V().hasLabel('Identity').has('phones', '+11234567890').union(
identity(),
__.out('Receives').hasLabel('Subscription'),
__.out('MemberOf').hasLabel('Channel')
).fold().unfold().drop()
// 预期结果:所有3个顶点(Identity、Subscription、Channel)均被删除使用此修改后的查询,Identity 顶点及其关联的 Subscription 和 Channel 顶点都将被成功删除。
尽管 Gremlin 的 union().drop() 组合在特定场景下可能表现出意外的行为,即无法删除 union() 产生的所有目标顶点,但通过在 drop() 之前巧妙地插入 fold().unfold() 步骤,可以有效地解决这一问题。这种方法强制 Gremlin 在执行删除操作前将所有目标顶点物化,确保 drop() 作用于完整的顶点集合。然而,在使用 fold() 时,务必注意其潜在的内存消耗,并根据数据规模评估其适用性。理解 Gremlin 的流处理模型对于编写高效且正确的图遍历至关重要。
以上就是解决 Gremlin union().drop() 无法删除所有指定顶点的问题的详细内容,更多请关注php中文网其它相关文章!
每个人都需要一台速度更快、更稳定的 PC。随着时间的推移,垃圾文件、旧注册表数据和不必要的后台进程会占用资源并降低性能。幸运的是,许多工具可以让 Windows 保持平稳运行。
Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号