如何将 Matplotlib 图形转换为 PIL Image 对象(不保存图像)

Posted

技术标签:

【中文标题】如何将 Matplotlib 图形转换为 PIL Image 对象(不保存图像)【英文标题】:How to convert Matplotlib figure to PIL Image object (without saving image) 【发布时间】:2019-12-10 11:48:32 【问题描述】:

正如标题所述,我正在尝试将 fig 转换为 PIL.Image。我目前能够通过首先将fig 保存到磁盘然后使用Image.open() 打开该文件来做到这一点,但是该过程花费的时间比预期的要长,我希望通过跳过本地保存步骤会更快一些.

这是我目前所拥有的:

# build fig
figsize, dpi = self._calc_fig_size_res(img_height)
fig = plt.Figure(figsize=figsize)
canvas = FigureCanvas(fig)
ax = fig.add_subplot(111)
ax.imshow(torch.from_numpy(S).flip(0), cmap = cmap)
fig.subplots_adjust(left = 0, right = 1, bottom = 0, top = 1)
ax.axis('tight'); ax.axis('off')

# export
fig.savefig(export_path, dpi = dpi)

# open image as PIL object
img = Image.open(export_path)

我在构建无花果后尝试过这样做(它会在导出阶段之前):

pil_img = Image.frombytes('RGB', canvas.get_width_height(), canvas.tostring_rgb())

但它没有显示整个图像。它看起来像是左上角的一部分,但它可能只是数据的一种奇怪表示——我正在使用频谱图,所以图像相当抽象。

【问题讨论】:

【参考方案1】:

编辑#2

PIL.Image.frombytes('RGB', 
fig.canvas.get_width_height(),fig.canvas.tostring_rgb())

与下面的 35/40 毫秒相比,大约需要 2 毫秒。

这是迄今为止我能找到的最快的方法。


我今天也在看这个。

在 matplotlib 文档中 savefig 函数有这个。

pil_kwargsdict,可选传递的附加关键字参数 保存图形时到 PIL.Image.save。仅适用于格式 使用 Pillow 保存的,即 JPEG、TIFF 和(如果关键字是 设置为非无值)PNG。

这一定意味着它在保存之前已经是一个pil图像但我看不到它。

你可以关注这个

Matplotlib: save plot to numpy array

将其放入一个 numpy 数组,然后执行

PIL.Image.fromarray(array)

您可能需要使用数组 [:, :, ::-1] 将通道从 BGR 反转为 RGB

编辑:

到目前为止,我已经测试了每种方法。

import io
    
def save_plot_and_get():
    fig.savefig("test.jpg")
    img = cv2.imread("test.jpg")
    return PIL.Image.fromarray(img)
    
def buffer_plot_and_get():
    buf = io.BytesIO()
    fig.savefig(buf)
    buf.seek(0)
    return PIL.Image.open(buf)
    
def from_canvas():
    lst = list(fig.canvas.get_width_height())
    lst.append(3)
    return PIL.Image.fromarray(np.fromstring(fig.canvas.tostring_rgb(),dtype=np.uint8).reshape(lst))

结果

%timeit save_plot_and_get()

每个循环 35.5 ms ± 148 µs(7 次运行的平均值 ± 标准偏差,每次 10 个循环)

%timeit save_plot_and_get()

每个循环 35.5 ms ± 142 µs(平均值 ± 标准偏差,7 次运行,每次 10 个循环)

%timeit buffer_plot_and_get()

每个循环 40.4 ms ± 152 µs(7 次运行的平均值 ± 标准偏差,每次 10 个循环)

【讨论】:

请注意,这可能依赖于 matplotlib 后端 - 某些后端(例如 QTAgg)需要通过 fig.canvas.draw() 绘制画布以在使用 tostring_rgb() 之前初始化渲染器。见***.com/a/35407794/10342097【参考方案2】:

我使用以下函数:

def fig2img(fig):
    """Convert a Matplotlib figure to a PIL Image and return it"""
    import io
    buf = io.BytesIO()
    fig.savefig(buf)
    buf.seek(0)
    img = Image.open(buf)
    return img

示例用法:

import numpy as np
import matplotlib.pyplot as plt
from PIL import Image

x = np.arange(-3,3)
plt.plot(x)
fig = plt.gcf()

img = fig2img(fig)

img.show()

【讨论】:

这行得通并且非常高效,但是为什么“img.show()”函数会在预览中而不是在 jupyter notebook 中打开图像?【参考方案3】:

我将其标记为重复,然后将其关闭,因为我使用了错误的链接。

反正答案可能就在这里:

how to save a pylab figure into in-memory file which can be read into PIL image?

【讨论】:

谢谢!这绝对是一个改进,因为它是一个Image 对象,它显示了正确的图像,我没有将它保存到磁盘。但是,由于某种原因,图像的分辨率比我将其保存到磁盘然后重新加载时要低得多,这是我在执行此过程时也遇到的问题:icare.univ-lille1.fr/tutorials/convert_a_matplotlib_figure |我将用示例编辑我的帖子 我收回了这一点——我只是忘了包含dpi 参数,这就是为什么分辨率较低的原因。所以谢谢!!我不会将其标记为重复,因为问题略有不同,但如果你觉得是,那么你可以标记。无论哪种方式,都解决了它:)【参考方案4】:

不幸的是,这并没有导致任何速度提高,但我仍会在下面发布我的具体解决方案,以防有人遇到类似问题:

# build fig
figsize, dpi = self._calc_fig_size_res(img_height)
fig = plt.Figure(figsize = figsize)
canvas = FigureCanvas(fig)
ax = fig.add_subplot(111)
ax.imshow(torch.from_numpy(S).flip(0), cmap = camp)
fig.subplots_adjust(left = 0, right = 1, bottom = 0, top = 1)
ax.axis('tight'); ax.axis('off')

# convert to PIL Image object
buf = io.BytesIO()
fig.savefig(buf, format='png', dpi = dpi)
buf.seek(0)
pil_img = deepcopy(Image.open(buf))
buf.close()

【讨论】:

“深拷贝”在哪里?

以上是关于如何将 Matplotlib 图形转换为 PIL Image 对象(不保存图像)的主要内容,如果未能解决你的问题,请参考以下文章

将轮廓(MatplotLib 或 OpenCV)转换为与原始大小相同的图像

Matplotlib 图形图像转 base64

如何将 PIL 图像转换为 numpy 数组?

opencv-PIL-matplotlib-Skimage-Pytorch图片读取区别与联系

使用 matplotlib 将图像显示为灰度

如何将 pygame Surface 转换为 PIL 图像?