如何使用pyinstaller将多个子进程python文件编译成单个.exe文件



我有一个类似的问题:类似的问题。 我有一个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答案的范围。

相关内容

  • 没有找到相关文章

最新更新