以不同方式编写的类似 Gremlin 查询具有性能差异
Similar Gremlin Queries Written in Different Ways Have Performance Differences
我有两个 gremlin 遍历查询几乎都在做同样的事情,每个查询都查找一个带有 ID 的顶点并向新节点插入一个带有边的新节点。
- 我先尝试了这个,因为它看起来更干净:
g.V('22b515e0-dbefb359-10f3-71ff13527bb2').sideEffect(__.outE().drop())\
.addE('someedge').to(__.V().hasLabel('SomeLabel').has('name', \
'SomeName').sideEffect(__.properties().drop()).property('name123', \
'SomeName123')).next()
此查询每次执行大约需要 8 秒。
- 作为解决上述问题的措施,我尝试将遍历更改为此,响应时间以毫秒为单位:
g.V().hasLabel('SomeLabel').has('name', \
'SomeName').sideEffect(__.properties().drop()).property('name123', \
'SomeName123').as_('X').V('22b515e0-dbef-b359-10f3-71ff13527bb2')\
.sideEffect(__.outE().drop()).addE('some \
edge').to(__.select('X')).next()
- 但是,当我直接使用 ID 查找时(在方法 - 1 中),即使这样,响应时间也以毫秒为单位。
g.V('22b515e0-dbef-b359-10f3-71ff13527bb2').sideEffect(__.outE().drop())\
.addE('someedge').to(__.V('8b66cab2-3b1c-41c9-9acb-296dda3c9139')\
.sideEffect(__.properties().drop()).property('name123', \
'SomeName123')).next()
这里有一些显示执行时间的 .profile() 指标。
对于 1:
{'dur': 8823.148244,
'metrics': [{'dur': 0.336926,
'counts': {'traverserCount': 1, 'elementCount': 1},
'name': 'NeptuneGraphQueryStep(Vertex)',
'annotations': {'percentDur': 0.003818659628994895},
'id': '6.0.0()'},
{'dur': 1.882282,
'counts': {'traverserCount': 1, 'elementCount': 1},
'name': 'TraversalSideEffectStep([VertexStep(OUT,edge), ProfileStep, DropStep, ProfileStep])',
'annotations': {'percentDur': 0.02133345091736396},
'id': '1.0.0()',
'metrics': [{'dur': 0.275912,
'counts': {'traverserCount': 11, 'elementCount': 11},
'name': 'VertexStep(OUT,edge)',
'id': '0.1.0(1.0.0())'},
{'dur': 1.569401,
'counts': {},
'name': 'DropStep',
'id': '1.1.0(1.0.0())'}]},
{'dur': 0.115423,
'counts': {'traverserCount': 1, 'elementCount': 1},
'name': 'NoOpBarrierStep',
'annotations': {'percentDur': 0.0013081838455847213},
'id': '5.0.0()'},
{'dur': 8820.813613,
'counts': {'traverserCount': 1, 'elementCount': 1},
'name': 'AddEdgeStep({~to=[[GraphStep(vertex,[]), ProfileStep, NeptuneHasStep([~label.eq(SomeLabel), name.eq(SomeName)]), ProfileStep, TraversalSideEffectStep([PropertiesStep(property), ProfileStep, DropStep, ProfileStep]), ProfileStep, NoOpBarrierStep, ProfileStep, AddPropertyStep({value=[SomeName123], key=[name123]}), ProfileStep]], label=[some edge]})',
'annotations': {'percentDur': 99.97353970560806},
'id': '2.0.0()',
'metrics': [{'dur': 1346.058324,
'counts': {'traverserCount': 373732, 'elementCount': 373732},
'name': 'GraphStep(vertex,[])',
'id': '0.1.0(2.0.0())'},
{'dur': 7473.661772,
'counts': {'traverserCount': 1, 'elementCount': 1},
'name': 'NeptuneHasStep([~label.eq(SomeLabel), name.eq(SomeName)])',
'id': '5.1.0(2.0.0())'},
{'dur': 0.283139,
'counts': {'traverserCount': 1, 'elementCount': 1},
'name': 'TraversalSideEffectStep([PropertiesStep(property), ProfileStep, DropStep, ProfileStep])',
'id': '2.1.0(2.0.0())',
'metrics': [{'dur': 0.096149,
'counts': {'traverserCount': 1, 'elementCount': 1},
'name': 'PropertiesStep(property)',
'id': '0.2.0(2.1.0(2.0.0()))'},
{'dur': 0.149241,
'counts': {},
'name': 'DropStep',
'id': '1.2.0(2.1.0(2.0.0()))'}]},
{'dur': 0.046801,
'counts': {'traverserCount': 1, 'elementCount': 1},
'name': 'NoOpBarrierStep',
'id': '4.1.0(2.0.0())'},
{'dur': 0.306436,
'counts': {'traverserCount': 1, 'elementCount': 1},
'name': 'AddPropertyStep({value=[SomeName123], key=[name123]})',
'id': '3.1.0(2.0.0())'}]}]}
对于 2:
{'dur': 5.596631,
'metrics': [{'dur': 0.510122,
'counts': {'traverserCount': 1, 'elementCount': 1},
'name': 'NeptuneGraphQueryStep(Vertex)',
'annotations': {'percentDur': 9.114804960341319},
'id': '11.0.0()'},
{'dur': 0.462721,
'counts': {'traverserCount': 1, 'elementCount': 1},
'name': 'TraversalSideEffectStep([PropertiesStep(property), ProfileStep, DropStep, ProfileStep])',
'annotations': {'percentDur': 8.267848997012667},
'id': '2.0.0()',
'metrics': [{'dur': 0.179074,
'counts': {'traverserCount': 1, 'elementCount': 1},
'name': 'PropertiesStep(property)',
'id': '0.1.0(2.0.0())'},
{'dur': 0.210574,
'counts': {},
'name': 'DropStep',
'id': '1.1.0(2.0.0())'}]},
{'dur': 0.092257,
'counts': {'traverserCount': 1, 'elementCount': 1},
'name': 'NoOpBarrierStep',
'annotations': {'percentDur': 1.6484381407314508},
'id': '9.0.0()'},
{'dur': 1.653211,
'counts': {'traverserCount': 1, 'elementCount': 1},
'name': 'AddPropertyStep({value=[SomeName123], key=[name123]})@[X]',
'annotations': {'percentDur': 29.539396111696483},
'id': '3.0.0()'},
{'dur': 0.116915,
'counts': {'traverserCount': 1, 'elementCount': 1},
'name': 'GraphStep(vertex,[22b515e0-dbef-b359-10f3-71ff13527bb2])',
'annotations': {'percentDur': 2.0890246292814374},
'id': '4.0.0()'},
{'dur': 1.073502,
'counts': {'traverserCount': 1, 'elementCount': 1},
'name': 'TraversalSideEffectStep([VertexStep(OUT,edge), ProfileStep, DropStep, ProfileStep])',
'annotations': {'percentDur': 19.181218129263836},
'id': '5.0.0()',
'metrics': [{'dur': 0.14199,
'counts': {'traverserCount': 11, 'elementCount': 11},
'name': 'VertexStep(OUT,edge)',
'id': '0.1.0(5.0.0())'},
{'dur': 0.879906,
'counts': {},
'name': 'DropStep',
'id': '1.1.0(5.0.0())'}]},
{'dur': 0.042369,
'counts': {'traverserCount': 1, 'elementCount': 1},
'name': 'NoOpBarrierStep',
'annotations': {'percentDur': 0.7570447292308533},
'id': '10.0.0()'},
{'dur': 1.645534,
'counts': {'traverserCount': 1, 'elementCount': 1},
'name': 'AddEdgeStep({~to=[[SelectOneStep(last,X), ProfileStep]], label=[some edge]})',
'annotations': {'percentDur': 29.40222430244195},
'id': '6.0.0()',
'metrics': [{'dur': 0.024292,
'counts': {'traverserCount': 1, 'elementCount': 1},
'name': 'SelectOneStep(last,X)',
'id': '0.1.0(6.0.0())'}]}]}
对于 3:
{'dur': 9.084957,
'metrics': [{'dur': 0.340621,
'counts': {'traverserCount': 1, 'elementCount': 1},
'name': 'NeptuneGraphQueryStep(Vertex)',
'annotations': {'percentDur': 3.7492857698721083},
'id': '6.0.0()'},
{'dur': 1.92865,
'counts': {'traverserCount': 1, 'elementCount': 1},
'name': 'TraversalSideEffectStep([VertexStep(OUT,edge), ProfileStep, DropStep, ProfileStep])',
'annotations': {'percentDur': 21.229049295445208},
'id': '1.0.0()',
'metrics': [{'dur': 0.282423,
'counts': {'traverserCount': 11, 'elementCount': 11},
'name': 'VertexStep(OUT,edge)',
'id': '0.1.0(1.0.0())'},
{'dur': 1.578817,
'counts': {},
'name': 'DropStep',
'id': '1.1.0(1.0.0())'}]},
{'dur': 0.058816,
'counts': {'traverserCount': 1, 'elementCount': 1},
'name': 'NoOpBarrierStep',
'annotations': {'percentDur': 0.647399872118272},
'id': '5.0.0()'},
{'dur': 6.75687,
'counts': {'traverserCount': 1, 'elementCount': 1},
'name': 'AddEdgeStep({~to=[[GraphStep(vertex,[8b66cab2-3b1c-41c9-9acb-296dda3c9139]), ProfileStep, TraversalSideEffectStep([PropertiesStep(property), ProfileStep, DropStep, ProfileStep]), ProfileStep, NoOpBarrierStep, ProfileStep, AddPropertyStep({value=[SomeName123], key=[name123]}), ProfileStep]], label=[some edge]})',
'annotations': {'percentDur': 74.37426506256442},
'id': '2.0.0()',
'metrics': [{'dur': 0.087179,
'counts': {'traverserCount': 1, 'elementCount': 1},
'name': 'GraphStep(vertex,[8b66cab2-3b1c-41c9-9acb-296dda3c9139])',
'id': '0.1.0(2.0.0())'},
{'dur': 0.332149,
'counts': {'traverserCount': 1, 'elementCount': 1},
'name': 'TraversalSideEffectStep([PropertiesStep(property), ProfileStep, DropStep, ProfileStep])',
'id': '1.1.0(2.0.0())',
'metrics': [{'dur': 0.144028,
'counts': {'traverserCount': 1, 'elementCount': 1},
'name': 'PropertiesStep(property)',
'id': '0.2.0(1.1.0(2.0.0()))'},
{'dur': 0.159151,
'counts': {},
'name': 'DropStep',
'id': '1.2.0(1.1.0(2.0.0()))'}]},
{'dur': 0.052224,
'counts': {'traverserCount': 1, 'elementCount': 1},
'name': 'NoOpBarrierStep',
'id': '3.1.0(2.0.0())'},
{'dur': 5.781101,
'counts': {'traverserCount': 1, 'elementCount': 1},
'name': 'AddPropertyStep({value=[SomeName123], key=[name123]})',
'id': '2.1.0(2.0.0())'}]}]}
图形数据库 (AWS Neptune) 中大约有 30 万个顶点和 300 万条边。
我是不是漏掉了什么?我想让它与 method-1 一起使用,因为这是复杂遍历的一部分,可能不适用于 method-2
从三个查询的profile()
来看,Neptune并没有优化to(V().hasLabel('SomeLabel').has('name','SomeName'))
的子遍历。在这种情况下,它正在对 V()
进行全图扫描:
{'dur': 8820.813613,
'counts': {'traverserCount': 1, 'elementCount': 1},
'name': 'AddEdgeStep({~to=[[GraphStep(vertex,[]), ProfileStep, NeptuneHasStep([~label.eq(SomeLabel), name.eq(SomeName)]), ProfileStep, TraversalSideEffectStep([PropertiesStep(property), ProfileStep, DropStep, ProfileStep]), ProfileStep, NoOpBarrierStep, ProfileStep, AddPropertyStep({value=[SomeName123], key=
[name123]}), ProfileStep]], label=[some edge]})',
'annotations': {'percentDur': 99.97353970560806},
'id': '2.0.0()',
'metrics': [{'dur': 1346.058324,
'counts': {'traverserCount': 373732, 'elementCount': 373732},
'name': 'GraphStep(vertex,[])',
'id': '0.1.0(2.0.0())'},
{'dur': 7473.661772,
'counts': {'traverserCount': 1, 'elementCount': 1},
'name': 'NeptuneHasStep([~label.eq(SomeLabel), name.eq(SomeName)])',
'id': '5.1.0(2.0.0())'},
看到 V()
的遍历器计数是 373732,它馈送到 has()
,后者执行 7 秒内存过滤器以找到 1 个顶点。
我有两个 gremlin 遍历查询几乎都在做同样的事情,每个查询都查找一个带有 ID 的顶点并向新节点插入一个带有边的新节点。
- 我先尝试了这个,因为它看起来更干净:
g.V('22b515e0-dbefb359-10f3-71ff13527bb2').sideEffect(__.outE().drop())\
.addE('someedge').to(__.V().hasLabel('SomeLabel').has('name', \
'SomeName').sideEffect(__.properties().drop()).property('name123', \
'SomeName123')).next()
此查询每次执行大约需要 8 秒。
- 作为解决上述问题的措施,我尝试将遍历更改为此,响应时间以毫秒为单位:
g.V().hasLabel('SomeLabel').has('name', \
'SomeName').sideEffect(__.properties().drop()).property('name123', \
'SomeName123').as_('X').V('22b515e0-dbef-b359-10f3-71ff13527bb2')\
.sideEffect(__.outE().drop()).addE('some \
edge').to(__.select('X')).next()
- 但是,当我直接使用 ID 查找时(在方法 - 1 中),即使这样,响应时间也以毫秒为单位。
g.V('22b515e0-dbef-b359-10f3-71ff13527bb2').sideEffect(__.outE().drop())\
.addE('someedge').to(__.V('8b66cab2-3b1c-41c9-9acb-296dda3c9139')\
.sideEffect(__.properties().drop()).property('name123', \
'SomeName123')).next()
这里有一些显示执行时间的 .profile() 指标。 对于 1:
{'dur': 8823.148244,
'metrics': [{'dur': 0.336926,
'counts': {'traverserCount': 1, 'elementCount': 1},
'name': 'NeptuneGraphQueryStep(Vertex)',
'annotations': {'percentDur': 0.003818659628994895},
'id': '6.0.0()'},
{'dur': 1.882282,
'counts': {'traverserCount': 1, 'elementCount': 1},
'name': 'TraversalSideEffectStep([VertexStep(OUT,edge), ProfileStep, DropStep, ProfileStep])',
'annotations': {'percentDur': 0.02133345091736396},
'id': '1.0.0()',
'metrics': [{'dur': 0.275912,
'counts': {'traverserCount': 11, 'elementCount': 11},
'name': 'VertexStep(OUT,edge)',
'id': '0.1.0(1.0.0())'},
{'dur': 1.569401,
'counts': {},
'name': 'DropStep',
'id': '1.1.0(1.0.0())'}]},
{'dur': 0.115423,
'counts': {'traverserCount': 1, 'elementCount': 1},
'name': 'NoOpBarrierStep',
'annotations': {'percentDur': 0.0013081838455847213},
'id': '5.0.0()'},
{'dur': 8820.813613,
'counts': {'traverserCount': 1, 'elementCount': 1},
'name': 'AddEdgeStep({~to=[[GraphStep(vertex,[]), ProfileStep, NeptuneHasStep([~label.eq(SomeLabel), name.eq(SomeName)]), ProfileStep, TraversalSideEffectStep([PropertiesStep(property), ProfileStep, DropStep, ProfileStep]), ProfileStep, NoOpBarrierStep, ProfileStep, AddPropertyStep({value=[SomeName123], key=[name123]}), ProfileStep]], label=[some edge]})',
'annotations': {'percentDur': 99.97353970560806},
'id': '2.0.0()',
'metrics': [{'dur': 1346.058324,
'counts': {'traverserCount': 373732, 'elementCount': 373732},
'name': 'GraphStep(vertex,[])',
'id': '0.1.0(2.0.0())'},
{'dur': 7473.661772,
'counts': {'traverserCount': 1, 'elementCount': 1},
'name': 'NeptuneHasStep([~label.eq(SomeLabel), name.eq(SomeName)])',
'id': '5.1.0(2.0.0())'},
{'dur': 0.283139,
'counts': {'traverserCount': 1, 'elementCount': 1},
'name': 'TraversalSideEffectStep([PropertiesStep(property), ProfileStep, DropStep, ProfileStep])',
'id': '2.1.0(2.0.0())',
'metrics': [{'dur': 0.096149,
'counts': {'traverserCount': 1, 'elementCount': 1},
'name': 'PropertiesStep(property)',
'id': '0.2.0(2.1.0(2.0.0()))'},
{'dur': 0.149241,
'counts': {},
'name': 'DropStep',
'id': '1.2.0(2.1.0(2.0.0()))'}]},
{'dur': 0.046801,
'counts': {'traverserCount': 1, 'elementCount': 1},
'name': 'NoOpBarrierStep',
'id': '4.1.0(2.0.0())'},
{'dur': 0.306436,
'counts': {'traverserCount': 1, 'elementCount': 1},
'name': 'AddPropertyStep({value=[SomeName123], key=[name123]})',
'id': '3.1.0(2.0.0())'}]}]}
对于 2:
{'dur': 5.596631,
'metrics': [{'dur': 0.510122,
'counts': {'traverserCount': 1, 'elementCount': 1},
'name': 'NeptuneGraphQueryStep(Vertex)',
'annotations': {'percentDur': 9.114804960341319},
'id': '11.0.0()'},
{'dur': 0.462721,
'counts': {'traverserCount': 1, 'elementCount': 1},
'name': 'TraversalSideEffectStep([PropertiesStep(property), ProfileStep, DropStep, ProfileStep])',
'annotations': {'percentDur': 8.267848997012667},
'id': '2.0.0()',
'metrics': [{'dur': 0.179074,
'counts': {'traverserCount': 1, 'elementCount': 1},
'name': 'PropertiesStep(property)',
'id': '0.1.0(2.0.0())'},
{'dur': 0.210574,
'counts': {},
'name': 'DropStep',
'id': '1.1.0(2.0.0())'}]},
{'dur': 0.092257,
'counts': {'traverserCount': 1, 'elementCount': 1},
'name': 'NoOpBarrierStep',
'annotations': {'percentDur': 1.6484381407314508},
'id': '9.0.0()'},
{'dur': 1.653211,
'counts': {'traverserCount': 1, 'elementCount': 1},
'name': 'AddPropertyStep({value=[SomeName123], key=[name123]})@[X]',
'annotations': {'percentDur': 29.539396111696483},
'id': '3.0.0()'},
{'dur': 0.116915,
'counts': {'traverserCount': 1, 'elementCount': 1},
'name': 'GraphStep(vertex,[22b515e0-dbef-b359-10f3-71ff13527bb2])',
'annotations': {'percentDur': 2.0890246292814374},
'id': '4.0.0()'},
{'dur': 1.073502,
'counts': {'traverserCount': 1, 'elementCount': 1},
'name': 'TraversalSideEffectStep([VertexStep(OUT,edge), ProfileStep, DropStep, ProfileStep])',
'annotations': {'percentDur': 19.181218129263836},
'id': '5.0.0()',
'metrics': [{'dur': 0.14199,
'counts': {'traverserCount': 11, 'elementCount': 11},
'name': 'VertexStep(OUT,edge)',
'id': '0.1.0(5.0.0())'},
{'dur': 0.879906,
'counts': {},
'name': 'DropStep',
'id': '1.1.0(5.0.0())'}]},
{'dur': 0.042369,
'counts': {'traverserCount': 1, 'elementCount': 1},
'name': 'NoOpBarrierStep',
'annotations': {'percentDur': 0.7570447292308533},
'id': '10.0.0()'},
{'dur': 1.645534,
'counts': {'traverserCount': 1, 'elementCount': 1},
'name': 'AddEdgeStep({~to=[[SelectOneStep(last,X), ProfileStep]], label=[some edge]})',
'annotations': {'percentDur': 29.40222430244195},
'id': '6.0.0()',
'metrics': [{'dur': 0.024292,
'counts': {'traverserCount': 1, 'elementCount': 1},
'name': 'SelectOneStep(last,X)',
'id': '0.1.0(6.0.0())'}]}]}
对于 3:
{'dur': 9.084957,
'metrics': [{'dur': 0.340621,
'counts': {'traverserCount': 1, 'elementCount': 1},
'name': 'NeptuneGraphQueryStep(Vertex)',
'annotations': {'percentDur': 3.7492857698721083},
'id': '6.0.0()'},
{'dur': 1.92865,
'counts': {'traverserCount': 1, 'elementCount': 1},
'name': 'TraversalSideEffectStep([VertexStep(OUT,edge), ProfileStep, DropStep, ProfileStep])',
'annotations': {'percentDur': 21.229049295445208},
'id': '1.0.0()',
'metrics': [{'dur': 0.282423,
'counts': {'traverserCount': 11, 'elementCount': 11},
'name': 'VertexStep(OUT,edge)',
'id': '0.1.0(1.0.0())'},
{'dur': 1.578817,
'counts': {},
'name': 'DropStep',
'id': '1.1.0(1.0.0())'}]},
{'dur': 0.058816,
'counts': {'traverserCount': 1, 'elementCount': 1},
'name': 'NoOpBarrierStep',
'annotations': {'percentDur': 0.647399872118272},
'id': '5.0.0()'},
{'dur': 6.75687,
'counts': {'traverserCount': 1, 'elementCount': 1},
'name': 'AddEdgeStep({~to=[[GraphStep(vertex,[8b66cab2-3b1c-41c9-9acb-296dda3c9139]), ProfileStep, TraversalSideEffectStep([PropertiesStep(property), ProfileStep, DropStep, ProfileStep]), ProfileStep, NoOpBarrierStep, ProfileStep, AddPropertyStep({value=[SomeName123], key=[name123]}), ProfileStep]], label=[some edge]})',
'annotations': {'percentDur': 74.37426506256442},
'id': '2.0.0()',
'metrics': [{'dur': 0.087179,
'counts': {'traverserCount': 1, 'elementCount': 1},
'name': 'GraphStep(vertex,[8b66cab2-3b1c-41c9-9acb-296dda3c9139])',
'id': '0.1.0(2.0.0())'},
{'dur': 0.332149,
'counts': {'traverserCount': 1, 'elementCount': 1},
'name': 'TraversalSideEffectStep([PropertiesStep(property), ProfileStep, DropStep, ProfileStep])',
'id': '1.1.0(2.0.0())',
'metrics': [{'dur': 0.144028,
'counts': {'traverserCount': 1, 'elementCount': 1},
'name': 'PropertiesStep(property)',
'id': '0.2.0(1.1.0(2.0.0()))'},
{'dur': 0.159151,
'counts': {},
'name': 'DropStep',
'id': '1.2.0(1.1.0(2.0.0()))'}]},
{'dur': 0.052224,
'counts': {'traverserCount': 1, 'elementCount': 1},
'name': 'NoOpBarrierStep',
'id': '3.1.0(2.0.0())'},
{'dur': 5.781101,
'counts': {'traverserCount': 1, 'elementCount': 1},
'name': 'AddPropertyStep({value=[SomeName123], key=[name123]})',
'id': '2.1.0(2.0.0())'}]}]}
图形数据库 (AWS Neptune) 中大约有 30 万个顶点和 300 万条边。
我是不是漏掉了什么?我想让它与 method-1 一起使用,因为这是复杂遍历的一部分,可能不适用于 method-2
从三个查询的profile()
来看,Neptune并没有优化to(V().hasLabel('SomeLabel').has('name','SomeName'))
的子遍历。在这种情况下,它正在对 V()
进行全图扫描:
{'dur': 8820.813613,
'counts': {'traverserCount': 1, 'elementCount': 1},
'name': 'AddEdgeStep({~to=[[GraphStep(vertex,[]), ProfileStep, NeptuneHasStep([~label.eq(SomeLabel), name.eq(SomeName)]), ProfileStep, TraversalSideEffectStep([PropertiesStep(property), ProfileStep, DropStep, ProfileStep]), ProfileStep, NoOpBarrierStep, ProfileStep, AddPropertyStep({value=[SomeName123], key=
[name123]}), ProfileStep]], label=[some edge]})',
'annotations': {'percentDur': 99.97353970560806},
'id': '2.0.0()',
'metrics': [{'dur': 1346.058324,
'counts': {'traverserCount': 373732, 'elementCount': 373732},
'name': 'GraphStep(vertex,[])',
'id': '0.1.0(2.0.0())'},
{'dur': 7473.661772,
'counts': {'traverserCount': 1, 'elementCount': 1},
'name': 'NeptuneHasStep([~label.eq(SomeLabel), name.eq(SomeName)])',
'id': '5.1.0(2.0.0())'},
看到 V()
的遍历器计数是 373732,它馈送到 has()
,后者执行 7 秒内存过滤器以找到 1 个顶点。