0

我使用 NURBS-python 发现了一个有趣的问题,但不确定这是故意这样做还是只是一个错误。我想用 2 个代码来介绍这个。

第一个应该与第二个输出相同,而不是。第一个成功更新了NURBS曲线的控制点并绘制了一条新曲线,

更改控制点列表的方式会影响结果。

import math
from geomdl import BSpline
from geomdl import NURBS
from geomdl import fitting
from geomdl import convert
from geomdl.visualization import VisMPL

P1=[[0, 0], [0, 1], [1, 1], [1, 0], [1, -1], [0, -1], [0, 0]]
Degree=3
CPN=5

P2=[[0, 0], [0, 1], [2, 1], [2, 0], [2, -1], [0, -1], [0, 0]]
CP=[[0, 0], [1, 2], [2, 0], [1, -2], [0, 0]]

c1=NURBS.Curve()
c1.degree=Degree
c1.ctrlpts=P1
c1.weights=[1, 1, 1, 1, 1, 1, 1]
c1.knotvector=[0, 0, 0, 0, 0.25, 0.5, 0.75, 1, 1, 1, 1]

c1.vis=VisMPL.VisCurve2D()
c1.render()

c1.ctrlpts=P2
c1.vis=VisMPL.VisCurve2D()
c1.render()


c2=convert.bspline_to_nurbs(fitting.approximate_curve(P1, Degree, ctrlpts_size=CPN))

c2.vis=VisMPL.VisCurve2D()
c2.render()

c2.ctrlpts=CP

c2.vis=VisMPL.VisCurve2D()
c2.render()

而第二个只是简单地更新控制点,曲线本身并没有改变。

import math
from geomdl import BSpline
from geomdl import NURBS
from geomdl import fitting
from geomdl import convert
from geomdl.visualization import VisMPL

P1=[[0, 0], [0, 1], [1, 1], [1, 0], [1, -1], [0, -1], [0, 0]]
Degree=3
CPN=5

P2=[[0, 0], [0, 1], [2, 1], [2, 0], [2, -1], [0, -1], [0, 0]]
CP=[[0, 0], [1, 2], [2, 0], [1, -2], [0, 0]]

c1=NURBS.Curve()
c1.degree=Degree
c1.ctrlpts=P1
c1.weights=[1, 1, 1, 1, 1, 1, 1]
c1.knotvector=[0, 0, 0, 0, 0.25, 0.5, 0.75, 1, 1, 1, 1]

c1.vis=VisMPL.VisCurve2D()
c1.render()


for i in range(len(P2)):
    c1.ctrlpts[i]=P2[i]
c1.vis=VisMPL.VisCurve2D()
c1.render()

c2=convert.bspline_to_nurbs(fitting.approximate_curve(P1, Degree, ctrlpts_size=CPN))

c2.vis=VisMPL.VisCurve2D()
c2.render()

for i in range(len(CP)):
    c2.ctrlpts[i]=CP[i]

c2.vis=VisMPL.VisCurve2D()
c2.render()

请帮忙,谢谢。

4

1 回答 1

0

Curve.ctrlpts是一个属性。当您访问 getter 时,它会返回控制点列表。访问 setter 会做更多事情,主要是一致性检查和必要的清理。

cpts = c1.ctrlpts调用时,调用__get__属性的方法,返回存储实例内部控制点的列表对象的引用Curve并分配给cpts变量:

>>> cpts = c1.ctrlpts
>>> type(cpts)
<class 'list'>

当你遍历 listcpts时,实际上是遍历了它在Curve类实例中的引用。由于您直接与列表对象交互,因此您无需访问ctrlptssetter,它实际上通过调用Curve.reset()方法进行清理。

不幸的是,无法覆盖内置类(在 API 级别上定义的类)的方法。

>>> list.__getitem__ = my_getter_method
TypeError: can't set attributes of built-in/extension type 'list'

结果,Curve.reset()当底层list对象发生变化时触发该方法变得有点复杂(但并非不可能)。

设置控制点最好和最安全的方法是使用属性的 setter 方法:

my_ctrlpts = [[1, 2, 3], [4, 5, 6], [7, 8, 9]]
c1.ctrlpts = my_ctrlpts

如果你想使用 for 循环,那么你需要手动调用 reset 方法:

for i in range(len(P2)):
    c1.ctrlpts[i]=P2[i]
c1.reset(evalpts=True)

注:我是 NURBS-Python (geomdl) 的作者

于 2019-05-16T18:43:18.340 回答