如何在 Python 动画函数中 return 未知数量的对象

How to return an unknown number of objects in Python animation function

我目前正在尝试制作一系列图像的动画,其中为每个图像绘制了最初未知数量的椭圆。到目前为止,我已经尝试了很多事情,但还没有找到解决方案,尽管我想我已经接近了。这是我的代码:

import matplotlib.pyplot as plt
from matplotlib.patches import Ellipse

def plot_images(img1, img2, objects, ax):

    im1 = ax.imshow(img1)
    im2 = ax.imshow(img2 with transparency as an overlay)

    # plotting an ellipse for each object
    e = [None]*len(objects)
    for j in range(len(objects)):
        e[j] = Ellipse(xy=(objects['x'][j], objects['y'][j]),
                            width=6 * objects['a'][j],
                            height=6 * objects['b'][j],
                            angle=objects['theta'][j] * 180. / np.pi)
        e[j].set_facecolor('none')
        e[j].set_edgecolor('red')
        ax.add_artist(e[j])

    return im1, im2, e


def animate(j):

    # extracting objects
    im1, im2, objects = object_finder_function()

    imm1, imm2, e = plot_images(im1, im2, objects, axs)


    return imm1, imm2, e

fig, axs = plt.subplots()
ani = animation.FuncAnimation(fig, animate, frames=image_number, interval=50, blit=True)
plt.show()

现在,当我尝试此代码时,收到以下错误消息:

AttributeError: 'list' object has no attribute 'get_zorder'

所以我尝试了不同的东西,但最终,我发现,当我进行测试时,我输入了 plot_images 函数

return im1, im2, e[0], e[1], e[2]

并相应地更改动画功能,即

imm1, imm2, e0, e1, e2 = plot_images(im1, im2, objects, axs)

return imm1, imm2, e0, e1, e2

我没有收到错误消息,椭圆实际上按照我的预期绘制在相应的框架中。现在的问题是,对于一个人来说,我想绘制每个图像的数百个椭圆,所以我必须手动将它们全部写下来(即 e[0]、e[1]、e[2] - - e[k],动画函数也是如此),这似乎不是正确的方法。另一件事是,正如我已经说过的,每个图像的椭圆数都会发生变化,并且以前不知道,所以我不可能相应地调整函数。

我怎样才能 return 这个省略号列表,以便动画读取它,就好像我会像在工作示例中那样将它们全部写下来一样?

听起来你想压扁 e

您可以使用已经平坦的变量创建一个列表并使用 e:

扩展它
    return tuple([im1, im2] + e)

unpacke任何你想使用它的地方。

您的代码有点不符合 Python 风格,所以为了清晰起见,我稍微清理了一下。您的 AttributeErrorget_zorder 函数有关,该函数在 matplotlib 中用于确定如何对图进行分层。有了你尝试过的东西,我可以告诉你只需要在最后解压你的 list_of_ellipses。

def plot_images(img1, img2, objects, ax):

    im1 = ax.imshow(img1)
    im2 = ax.imshow(img2 with transparency as an overlay)

    list_of_ellipses = []
    for j in range(len(objects)):
        my_ellipse = Ellipse(xy=(objects['x'][j], objects['y'][j]),
                        width=6 * objects['a'][j],
                        height=6 * objects['b'][j],
                        angle=objects['theta'][j] * 180. / np.pi)

        my_ellipse.set_facecolor('none')
        my_ellipse.set_edgecolor('red')
        ax.add_artist(my_ellipse)
        list_of_ellipses.append(my_ellipse)
    return im1, im2, list_of_ellipses


def animate():
    im1, im2, objects = object_finder_function()
    imm1, imm2, list_of_ellipses = plot_images(im1, im2, objects, axs)
    return (imm1, imm2)+tuple(list_of_ellipses)

fig, axs = plt.subplots()
ani = animation.FuncAnimation(fig, animate, frames=image_number, interval=50, blit=True)
plt.show()

假设您正在使用三个实例中的 matplotlib.animation, animate should be returning an iterable and you are returning one that contains three objects. return imm1, imm2, e is returning a tuple。最后一个是列表。通过将 animate 函数更改为:

,您应该能够 return 一个 list 而不是 tuple
def animate(j):
    im1, im2, objects = object_finder_function()
    imm1, imm2, e = plot_images(im1, im2, objects, axs)

    return [imm1, imm2] + e

但是,我会将 plot_images 更改为 return 和 list。可能类似于以下内容:

def create_ellipse(objects, object_idx, artists):
    ellipse = Ellipse(
        xy=(objects['x'][object_idx], objects['y'][object_idx]),
        width=(6 * objects['a'][object_idx]),
        height=(6 * objects['b'][object_idx]),
        angle=(objects['theta'][object_idx] * 180.0 / np.pi))
    ellipse.set_facecolor('none')
    ellipse.set_edgecolor('red')
    artists.add_artists(ellipse)
    return ellipse

def plot_images(img1, img2, objects, ax):
    renderables = [ax.imshow(img1),
                   ax.imshow(img2 with transparency as an overlay)]
    renderables.extend(create_ellipse(objects, idx, ax)
                       for idx in range(len(objects)))
    return renderables