不能直接用 itchat 或 wechaty 发定时消息,因其阻塞式轮询或不成熟架构会导致主线程卡死、收不到消息;需用 threading + queue 将微信长连接与定时调度分离,确保 itchat 实例仅在单线程调用,并通过队列安全传递发送任务。

为什么不能直接用 itchat 或 wechaty 发定时消息
因为 itchat 是阻塞式轮询,wechaty(Python 版)目前不成熟且依赖 Puppeteer,启动慢、易掉线。一旦你用 time.sleep() 或 schedule 在主线程里调微信发送逻辑,整个机器人就卡住,收不到新消息、响应延迟、甚至会断连。
真正能跑稳的方案,是把微信通信和定时调度彻底拆开:一个线程跑微信长连接(保持登录、收发消息),另一个线程只管“到点触发”,然后通过线程安全的队列把任务塞过去。
怎么用 threading + queue 实现非阻塞定时发送
核心不是“定时器多高级”,而是让定时逻辑不碰微信实例本身。微信对象(比如 itchat 实例)必须只在它自己的线程里被调用,其他线程一律不能直接调 itchat.send()。
- 用
threading.Thread(target=itchat.run)启动微信主循环,别加daemon=True,否则可能提前退出 - 另起一个线程跑
schedule或APScheduler,但所有“要发什么”都塞进queue.Queue() - 微信线程里加个简单轮询:
if not msg_queue.empty(): itchat.send(**msg_queue.get()) - 注意
itchat.send()不支持直接发本地路径的图片/视频,得先用open(path, 'rb')读成二进制,再传给itchat.send_image()等专用函数
多媒体消息怎么发才不报错
itchat 对文件类型和路径非常敏感,尤其定时场景下容易因工作目录变化或文件被删导致 FileNotFoundError 或静默失败。
立即学习“Python免费学习笔记(深入)”;
- 所有媒体文件路径必须用绝对路径,推荐用
os.path.abspath('media/photo.jpg') - 图片优先用
itchat.send_image(),不要用send()+isPicture=True,后者不识别扩展名,常发成文件 - 视频必须小于 25MB,且得是 MP4 封装、H.264 编码;用
itchat.send_video(),别用send_file() - 发前加一层存在性检查:
if os.path.exists(file_path) and os.path.getsize(file_path) > 0:
定时器选 schedule 还是 APScheduler
如果只是每天固定时间发几条,schedule 足够轻量;但只要涉及“每 15 分钟发一次”或“跳过节假日”,就必须换 APScheduler——schedule 的 every().minutes.do() 在长时间运行时会因系统休眠、时钟漂移累积误差,实际间隔可能变成 17 分钟甚至更久。
-
schedule没有持久化,程序重启定时任务全丢;APScheduler可配SQLAlchemyJobStore存数据库 - 用
APScheduler时,务必设coalesce=True和max_instances=1,防止重复触发 - 别在定时任务函数里初始化
itchat,它只能在一个线程里 login 一次









