【编程实践】Windows + PySide6 + Matplotlib 绘图时 WinError 32 的完整排查与解决方案

1. 问题背景

在基于 PySide6(Qt)+ Matplotlib 开发桌面 GUI 程序时,需要在界面中动态生成 Matplotlib 图像,并显示在 QLabel 中。

程序功能表现为:

  • Matplotlib 图像可以正常绘制

  • 图像可以正常显示在界面中

  • 但在生成图像后,程序弹出错误提示:

    // 提示信息
    WinError 32: 另一个程序正在使用此文件,进程无法访问
    C:\Users\xxx\AppData\Local\Temp\tmpxxxx.png

该错误并不会影响图像显示结果,但会中断后续逻辑,严重影响程序稳定性。

2. 初步误判与无效尝试

在排查过程中,最先怀疑的方向包括:

  • Matplotlib 后端问题(Qt5Agg / 默认后端)
  • plt.close(fig) 是否调用正确
  • 是否需要切换为 Agg 后端
  • 是否应避免使用系统临时目录

尝试过的方案包括:

  • 显式调用 plt.close(fig)
  • 移除 matplotlib.use('Qt5Agg')
  • 改用 fig.savefig()
  • 改变临时文件保存路径
    这些修改均无法彻底解决问题,说明问题不在 Matplotlib 后端本身。

3. 问题本质定位

最终定位发现,该问题是Windows 文件锁机制 + Qt 文件加载行为共同导致的结果。

3.1 关键代码模式(问题写法)

javascript 复制代码
// python
temp_file = tempfile.NamedTemporaryFile(suffix=".png", delete=False)
plt.savefig(temp_file.name)

pixmap = QPixmap(temp_file.name)
os.unlink(temp_file.name)

(1)NamedTemporaryFile 文件句柄未释放

在 Windows 系统中:

  • NamedTemporaryFile 在未关闭前
  • 文件始终处于 被占用状态
  • 即使 delete=False,依然无法被其他进程安全删除

(2)QPixmap(file_path) 会锁定文件

QPixmap(path) 在 Windows 下会:

  • 直接占用并锁定该文件
  • 且不会在短时间内释放句柄
  • 导致后续 os.unlink() 必然失败

两者叠加后,100% 触发 WinError 32。

4. 正确的解决方案

核心思想

Qt 显示图像应从内存加载,而不是直接从文件路径加载。

即:

  • 文件只作为中转
  • Qt 只接触内存数据
  • 文件可在加载后立即删除

4.1 推荐实现方式(稳定方案)

javascript 复制代码
// python
import tempfile
import os
from PySide6.QtGui import QPixmap

# 1. 生成临时文件路径(不保持打开状态)
fd, tmp_path = tempfile.mkstemp(suffix=".png")
os.close(fd)
# 2. 使用 fig.savefig 写入文件
fig.savefig(tmp_path, dpi=150, bbox_inches="tight")
# 3. 读入内存
with open(tmp_path, "rb") as f:
    img_bytes = f.read()
# 4. 从内存加载到 QPixmap
pixmap = QPixmap()
pixmap.loadFromData(img_bytes)
# 5. 显示
label.setPixmap(pixmap)
# 6. 安全删除文件
os.remove(tmp_path)

5. 为什么这种方式是"稳妥解方案"

项目 结果
Windows 文件锁 完全规避
Qt 文件句柄 不再占用磁盘
多次刷新 稳定
后端依赖
GUI 响应 正常

这是 PySide / PyQt 在 Windows 下加载临时图像的标准工程写法

6. 常见错误总结(避坑清单)

× 不推荐写法

QPixmap("xxx.png")

  • NamedTemporaryFile(delete=False) 不关闭

  • plt.savefig() 混用 pyplot 状态

  • 绘图后立即 os.unlink()

√ 推荐写法

  • fig.savefig()

  • mkstemp + os.close(fd)

  • QPixmap.loadFromData(bytes)

  • 文件仅作为中转

7. 总结

本次问题并非 Matplotlib 绘图错误,而是:

Windows 文件锁机制 + Qt 图像加载方式 + 临时文件管理不当共同导致的典型 GUI 工程级问题。

通过将 Qt 图像加载从"文件路径"切换为"内存数据",问题得以彻底解决。

该经验适用于:

  • PySide6 / PyQt5 / PyQt6
  • Matplotlib / Pillow / OpenCV 生成的临时图像
  • 所有 Windows 桌面 GUI 项目
相关推荐
starrycode8885 小时前
【每日一个知识点】Kotlin开发基础知识
ui·kotlin
czhc11400756636 小时前
c# 1216
windows·microsoft·c#
JPX-NO6 小时前
windows下编程IDE使用docker搭建的rust开发环境(Linux)
ide·windows·docker·rust
大猫和小黄8 小时前
Windows环境下使用Nacos搭建若依(RuoYi)微服务版完整指南
windows·微服务·架构
网硕互联的小客服8 小时前
CC攻击对服务器正常运行会有什么影响?如何预防和解决CC攻击?
运维·服务器·网络·windows·安全
【上下求索】8 小时前
学习笔记096——Windows postgreSQL-18.1[压缩包版本]
windows·笔记·学习·postgresql
企微自动化9 小时前
企业微信客户端 UI 自动化定位技术的稳定性和局限性
ui·自动化·企业微信
jayson.h9 小时前
word宏批量插入.csv格式的附件
windows·word
yoona10209 小时前
Flutter 声明式 UI:为什么 build 会被反复调用?
flutter·ui·区块链·dex