繁体   English   中英

函数将 NdOverlay 返回给 DynamicMap 时的奇怪行为

[英]Strange behaviour when function returns NdOverlay to DynamicMap

当有一个函数生成一个 NdOverlay of Points 到 DynamicMap 时,我遇到了一些非常奇怪的事情,其中​​该函数与面板小部件相关联(我认为面板小部件并不重要)。

下面的代码是一个产生预期行为的工作示例。 每当您更改小部件值时,都会生成一个新图,其中覆盖了两组点,具有不同的颜色和各自的图例条目。 代码如下图所示。

a_widget = pn.widgets.Select(name='A', options=[1,2,3,4])
b_widget = pn.widgets.IntSlider(name='B', start=10, end=20, value=10)
widget_box = pn.WidgetBox(a_widget, b_widget, align='center')

@pn.depends(a=a_widget.param.value, b=b_widget.param.value)
def get_points(a, b):
    return hv.NdOverlay({x: hv.Points(np.random.rand(10,10)) for x in range(1,3)})

points = hv.DynamicMap(get_points)

pn.Row(widget_box, points)

图片

下面显示的第二个示例旨在证明在某些情况下您可能只想简单地返回一个空图,并且我在本示例中完成的方式与本示例中的方式相同: http:/ /holoviews.org/gallery/demos/bokeh/box_draw_roi_editor.html#bokeh-gallery-box-draw-roi-editora == 1时,此代码的结果是一个空图,但当a值不是1 ,结果很奇怪,如下图所示。

  1. 点都具有相同的颜色
  2. 例如,当更改滑块时,某些点被冻结并且永远不会改变,而在上面的工作示例中并非如此。
a_widget = pn.widgets.Select(name='A', options=[1,2,3,4])
b_widget = pn.widgets.IntSlider(name='B', start=10, end=20, value=10)
widget_box = pn.WidgetBox(a_widget, b_widget, align='center')

@pn.depends(a=a_widget.param.value, b=b_widget.param.value)
def get_points(a, b):
    if a == 1:
        return hv.NdOverlay({None: hv.Points([])})
    else:
        return hv.NdOverlay({x: hv.Points(np.random.rand(10,10)) for x in range(1,3)})

points = hv.DynamicMap(get_points)

pn.Row(widget_box, points)

图片

虽然我无法帮助NdOverlay观察到的问题,但可以在Overlay的帮助下创建有或没有内容的图。

由于b_widget从未在您的代码中使用过,为了简单起见,我将其删除。

a_widget = pn.widgets.Select(name='A', options=[1,2,3,4])
widget_box = pn.WidgetBox(a_widget, align='center')

@pn.depends(a=a_widget.param.value)
def get_points(a):
    images = []
    if a == 3:
        images.append(hv.Points(np.random.rand(10,10), label='None'))
    else:
        for x in range(1,3):
            images.append(hv.Points(np.random.rand(10,10), label=str(x)))
    return hv.Overlay(images)

points = hv.DynamicMap(get_points)

pn.Row(widget_box, points)

NdOverlay 文档中描述的如何使用 NdOverlay 的方式与您的方法不同,这可能是观察到的问题的原因。


无论如何,为了缩小代码的哪一部分负责观察到的问题,我删除了所有不需要重现它的代码。

为了清楚起见,我改名的值a ,而我也确信,对于起始值a提供。

结果在测试代码时, if - else - 语句并不重要,所以我也删除了它。

为了确保变量的行为符合预期,我添加了一些print语句。

这给出了以下最小的可重现示例:

a_widget = pn.widgets.Select(name='A', value='Test', options=['Test','Test1', 'Test2'])

@pn.depends(a=a_widget.param.value)
def get_points(a):
    dict_ = {}
    dict_[str(a)] = hv.Points(np.random.rand(10,10))

    print(dict_)
    overlay = hv.NdOverlay(dict_)
    print(overlay)

    return overlay

points = hv.DynamicMap(get_points)

# using the server approach here to see the outpout of the
# print-statements
app = pn.Row(a_widget, points)
app.app() 

运行此代码并在选择小部件中选择不同的选项时,一旦选择了选项Test1Test3之一,就会发现选项Test不会更新。

当我们像这样更改第一行的默认值时

a_widget = pn.widgets.Select(name='A', value='Test2', options=['Test','Test1', 'Test2'])

现在Test2没有正确更新。

所以看起来这是使用NdOverlayDynamicMap的问题。 所以我建议你向开发人员报告这个问题(如果还没有完成),要么等待新版本发布,要么使用不同的方法(例如如上所示)。

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM