Python并发程序优雅退出需主动监听信号、及时清理资源、避免强制终止;用signal捕获SIGINT设退出标志,以threading.Event或asyncio任务管理生命周期,统一关闭文件、连接、日志等资源,并规避sys.exit()、os._exit()等危险操作。
Python 并发程序优雅退出的核心是:主动监听退出信号、及时清理资源、避免强制终止导致状态不一致或数据丢失。
对于运行在终端的脚本,Ctrl+C 会发送 SIGINT,需捕获并触发有序关闭流程:
signal.signal(signal.SIGINT, handler),在 handler 中设置退出标志(如 event.set() 或修改全局布尔变量)threading.Event 控制工作线程,主线程捕获信号后调用 event.set(),各工作线程循环中检查 event.is_set() 并安全退出不同并发模型退出方式略有差异,但都应避免 threading.Thread._stop()(已废弃且不安全)或 asyncio.Task.cancel() 后不等待:

threading.Event 作为“退出开关”,工作线程定期检查;退出前调用 thread.join(timeout=3) 等待自然结束,超时则记录警告Task 对象;收到退出信号后调用 task.cancel(),再用 await asyncio.gather(*tasks, return_exceptions=True) 等待全部完成executor.shutdown(wait=True, cancel_futures=True) —— Python 3.9+ 支持 cancel_futures,可尝试取消未开始的任务退出前必须完成资源清理,否则可能引发泄漏或脏数据:
with 语句或显式 .close()
logging.shutdown() 或 connection.commit())一些看似合理但实际危险的做法:
sys.exit() —— 可能跳过 finally 和上下文管理器的 __exit__
os._exit() 强制终止 —— 绕过所有 Python 清理逻辑,极易丢失缓冲区数据subprocess.Popen 启动了外部程序,退出前应调用 proc.terminate() + proc.wait()
async def cleanup(): ... 忘记 await cleanup(),导致异步资源未释放