
本教程讲解如何正确实现“列表每项生成并打印一张独立票据”的功能,解决因打印异步导致所有票据内容均为最后一项的常见问题,并提供稳定、可扩展的python打印方案。
在使用 win32api.ShellExecute(..., "print", ...) 打印文本文件时,该调用是非阻塞式的——它仅向Windows打印队列提交任务后立即返回,并不等待实际打印完成。因此,当循环快速连续执行时,后续迭代会立刻覆写 tik.txt 文件,而前一个打印任务仍从已被覆盖的旧内容中读取(或因文件缓存/延迟读取导致全部输出最后一个值),最终所有票据都显示最后一个列表元素。
✅ 正确实现方式:同步控制 + 资源安全
以下代码采用更健壮的写法,关键改进包括:
- 使用 with open(...) 确保文件及时关闭,避免句柄占用或写入延迟;
- 用 for item in mylist: 替代易出错的 range(len(...)) 索引遍历;
- 插入 time.sleep() 提供可控等待窗口,让系统有足够时间完成当前打印作业;
- 移除冗余模块(如未使用的 win32print, os, sys)保持简洁。
import win32api
import time
mylist = ["a", "b", "c", "d"]
tik_path = r"C:\Gestión Caja\tik.txt" # 推荐使用原始字符串避免转义问题
for item in mylist:
# 每次循环生成专属内容:仅写入当前元素(无换行也可,视票据格式需求)
with open(tik_path, "w", encoding="utf-8") as f:
f.write(item + "\n") # 可按需添加标题、分隔线等,例如 f.write(f"【票据】{item}\n——{time.strftime('%H:%M')}")
# 提交打印任务
win32api.ShellExecute(0, "print", tik_path, None, ".", 0)
# 关键:等待打印队列处理完毕(建议 3–10 秒,依打印机响应速度调整)
time.sleep(5)⚠️ 注意事项与优化建议
- 延迟时间需实测调整:time.sleep(5) 是保守值。若打印机较快(如热敏小票机),可尝试 3;若网络打印机或驱动较慢,建议增至 8–10。过度缩短会导致内容错乱,过长则降低吞吐效率。
- 避免中文路径/文件名异常:确保 tik_path 中无非法字符,推荐使用英文路径;若必须含中文,请确认系统区域设置与 Python 默认编码兼容(本例中显式指定 encoding="utf-8" 提升鲁棒性)。
- 增强容错(进阶):生产环境建议增加 try/except 包裹打印逻辑,并检查文件是否成功写入(如 os.path.getsize(tik_path) > 0)。
- 替代方案(更精准控制):如需严格同步,可调用 win32print 直接发送 RAW 数据到打印机端口(绕过文件+ShellExecute),但开发复杂度显著上升,适用于高并发或定制化票据场景。
通过以上方法,即可确保列表中每个元素均生成内容准确、物理隔离的一张票据,彻底解决“全打最后一项”的典型异步陷阱。











