我有一个类似的问题:类似的问题。 我有一个GUI,用户可以在其中输入信息,其他脚本使用其中一些信息来运行。每个按钮都有 4 个不同的脚本。我将它们作为子进程运行,以便主 gui 不会采取行动或说它没有响应。这是我拥有的一个例子,因为自从我使用 PAGE 生成 gui 以来,代码真的很长。
###Main.py#####
import subprocess
def resource_path(relative_path):
#I got this from another post to include images but I'm also using it to include the scripts"
try:
# PyInstaller creates a temp folder and stores path in _MEIPASS
base_path = sys._MEIPASS
except Exception:
base_path = os.path.abspath(".")
return os.path.join(base_path, relative_path)
Class aclass:
def get_info(self):
global ModelNumber, Serial,SpecFile,dateprint,Oper,outputfolder
ModelNumber=self.Model.get()
Serial=self.SerialNumber.get()
outputfolder=self.TEntry2.get()
SpecFile= self.Spec_File.get()
return ModelNumber,Serial,SpecFile,outputfolder
def First(self):
aclass.get_info(self) #Where I use the resource path function
First_proc = subprocess.Popen([sys.executable, resource_path('first.py'),str(ModelNumber),str(Serial),str(path),str(outputfolder)])
First_proc.wait()
#####First.py#####
import numpy as np
import scipy
from main import aclass
ModelNumber = sys.argv[1]
Serial = sys.argv[2]
path = sys.argv[3]
path_save = sys.argv[4]
我的第二个、第三个和第四个脚本也是如此。
在我的规范文件中,我添加了:
a.datas +=[('first.py','C\path\to\script\first.py','DATA')]
a.datas +=[('main.py','C\path\to\script\main.py','DATA')]
这可以编译并且可以工作,但是当我尝试将其转换为.exe时,它崩溃了,因为它无法正确导入 first.py 和自己的库(numpy,scipy....等)。我尝试将其添加到 a.data 中,并在规范文件中runtime_hooks=['first.py']...我无法让它工作。有什么想法吗?我不确定它是否给了我这个错误,因为它是一个子过程。
假设您无法重组应用程序,因此没有必要(例如,使用multiprocessing
而不是subprocess
),有三种解决方案:
- 确保.exe将脚本作为(可执行)zip文件(或仅使用
pkg_resources
)包含,并将脚本复制到临时目录,以便可以从那里运行它。 - 编写一个多入口点包装器脚本,该脚本可以作为主程序运行,也可以作为每个脚本运行 - 因为,虽然您无法从打包的 exe 中运行脚本,但您可以从中导入模块。
- 再次使用
pkg_resources
,编写一个运行脚本的包装器,方法是将其加载为字符串并使用exec
运行它。
第二个可能是最干净的,但它有点工作。而且,虽然我们可以依靠setuptools
入口点来完成某些工作,但尝试解释如何执行此操作比解释如何手动执行此操作要困难得多,1因此我将执行后者。
假设您的代码如下所示:
# main.py
import subprocess
import sys
spam, eggs = sys.argv[1], sys.argv[2]
subprocess.run([sys.executable, 'vikings.py', spam])
subprocess.run([sys.executable, 'waitress.py', spam, eggs])
# vikings.py
import sys
print(' '.join(['spam'] * int(sys.argv[1])))
# waitress.py
import sys
import time
spam, eggs = int(sys.argv[1]), int(sys.argv[2]))
if eggs > spam:
print("You can't have more eggs than spam!")
sys.exit(2)
print("Frying...")
time.sleep(2)
raise Exception("This sketch is getting too silly!")
因此,您可以像这样运行它:
$ python3 main.py 3 4
spam spam spam
You can't have more eggs than spam!
我们希望重新组织它,以便有一个脚本来查看命令行参数以决定要导入的内容。这是执行此操作的最小更改:
# main.py
import subprocess
import sys
if sys.argv[1][:2] == '--':
script = sys.argv[1][2:]
if script == 'vikings':
import vikings
vikings.run(*sys.argv[2:])
elif script == 'waitress':
import waitress
waitress.run(*sys.argv[2:])
else:
raise Exception(f'Unknown script {script}')
else:
spam, eggs = sys.argv[1], sys.argv[2]
subprocess.run([sys.executable, __file__, '--vikings', spam])
subprocess.run([sys.executable, __file__, '--waitress', spam, eggs])
# vikings.py
def run(spam):
print(' '.join(['spam'] * int(spam)))
# waitress.py
import sys
import time
def run(spam, eggs):
spam, eggs = int(spam), int(eggs)
if eggs > spam:
print("You can't have more eggs than spam!")
sys.exit(2)
print("Frying...")
time.sleep(2)
raise Exception("This sketch is getting too silly!")
现在:
$ python3 main.py 3 4
spam spam spam
You can't have more eggs than spam!
在现实生活中,您可能需要考虑的一些更改:
- DRY:我们为每个脚本复制和粘贴相同的三行代码,并且必须键入每个脚本名称三次。您可以使用类似
__import__(sys.argv[1][2:]).run(sys.argv[2:])
的东西进行适当的错误处理。 - 使用
argparse
而不是这个笨拙的特殊大小写作为第一个参数。如果您已经在向脚本发送非平凡的参数,那么您可能已经在使用argparse
或替代方案。 - 向每个仅调用
run(sys.argv[1:])
的脚本添加一个if __name__ == '__main__':
块,以便在开发过程中您仍然可以直接运行脚本来测试它们。
我没有做这些,因为它们会模糊这个微不足道的例子的想法。
1 如果您已经完成,那么该文档作为复习非常有用,但作为教程和解释性理由,则不是那么多。并试图编写才华横溢的 PyPA 家伙多年来一直无法想出的教程......这可能超出了SO答案的范围。