本文介绍如何使用 subprocess.popen 在独立命令行窗口中运行另一个 python 脚本,同时确保其继承当前虚拟环境(venv)的解释器和已安装包,避免“模块未找到”错误。核心方案是显式指定 sys.executable 并合理使用 creationflags。
在 Python 开发中,常需从主脚本(如 PythonScript1.py)启动另一个脚本(如 script.py),且要求子进程严格运行于同一虚拟环境中——否则会因路径隔离导致 ImportError。关键在于:子进程不会自动继承父进程的虚拟环境激活状态,shell=True 仅复用系统 shell 环境,而非 venv 的 PATH 或 PYTHONPATH。
sys.executable 返回当前正在执行 Python 脚本的解释器绝对路径(例如 C:\project\.venv\Scripts\python.exe 或 /home/user/project/.venv/bin/python),天然指向激活的虚拟环境。这是最可靠、跨平台的方式:
import subprocess
import sys
# 启动 script.py,复用当前 venv 解释器,在新窗口中运行
process = subprocess.Popen(
[sys.executable, "path/to/script.py"],
creationflags=subprocess.CREATE_NEW_CONSOLE # Windows:强制新控制台
)? 提示:subprocess.CREATE_NEW_CONSOLE 仅适用于 Windows。若需跨平台支持新终端,可改用 os.system(不推荐)或调用平台特定命令(如 gnome-terminal -- python script.py),但会牺牲可移植性与进程控制能力。
import subprocess
import sys
import os
def launch_in_new_console(script_path):
if not os.path.exists(script_path):
raise FileNotFoundError(f"Script not found: {script_path}")
try:
# 使用当前解释器 + 新控制台(Windows)
flags =
subprocess.CREATE_NEW_CONSOLE if os.name == 'nt' else 0
proc = subprocess.Popen(
[sys.executable, script_path],
creationflags=flags,
stdout=subprocess.PIPE,
stderr=subprocess.PIPE,
text=True
)
print(f"Launched {script_path} in new console (PID: {proc.pid})")
return proc
except Exception as e:
print(f"Failed to launch: {e}")
raise
# 调用示例
if __name__ == "__main__":
launch_in_new_console("another_script.py")要让子脚本在新窗口中正确使用虚拟环境:
此方法简洁、健壮、无需手动激活 venv,是生产环境推荐的最佳实践。