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

How to compile multiple subprocess python files into single .exe file using pyinstaller

我有一个类似的问题:。 我有一个 GUI,用户可以在其中输入信息,其他脚本使用其中的一些信息 run.I 每个按钮有 4 个不同的脚本。我 运行 它们作为一个子进程,这样主图形用户界面就不会运行或说它没有响应。这是我所拥有的示例,因为我使用 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。 ...ETC)。我已经尝试将它添加到规范文件中的 a.datas 和 runtime_hooks=['first.py'] 中……但我无法让它工作。有任何想法吗?我不确定它是否给我这个错误,因为它是一个子进程。

假设您无法重构您的应用程序,因此这不是必需的(例如,通过使用 multiprocessing 而不是 subprocess),有三种解决方案:

  • 确保 .exe 包含作为(可执行)zip 文件的脚本——或者只使用 pkg_resources——并将脚本复制到一个临时目录,以便你可以从那里 运行 它。
  • 编写一个多入口点包装器脚本,可以 运行 作为你的主程序,也可以 运行 作为每个脚本——因为,虽然你不能 运行 从打包的 exe 中提取脚本,你可以从中导入模块
  • 再次使用 pkg_resources,编写一个 运行 脚本的包装器,将其作为字符串加载,然后 运行 用 exec 代替它。

第二个可能是最干净的,但有点工作。而且,虽然我们可以依靠 setuptools entrypoints 完成某些工作,但试图解释如何做到这一点比解释如何手动完成要困难得多,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 答案的范围。