3

我有一个与此类似的问题:Similar Question。我有一个 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.datas 和规范文件中的 runtime_hooks=['first.py'] ......但我无法让它工作。有任何想法吗?我不确定它是否给了我这个错误,因为它是一个子进程。

4

1 回答 1

2

假设您无法重组您的应用程序因此没有必要(例如,通过使用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:我们为每个脚本复制并粘贴了相同的三行代码,我们必须为每个脚本名称键入 3 次。您可以使用类似的东西__import__(sys.argv[1][2:]).run(sys.argv[2:])并进行适当的错误处理。
  • 在第一个参数中使用argparse这个 hacky 的特殊大小写。如果您已经向脚本发送了重要的参数,那么您可能已经在使用argparse或替代。
  • if __name__ == '__main__':为每个调用的脚本添加一个块run(sys.argv[1:]),以便在开发过程中您仍然可以直接运行脚本来测试它们。

我没有做任何这些,因为它们会掩盖这个微不足道的例子的想法。


1 如果您已经完成了该文档作为复习,那么该文档非常棒,但作为教程和解释性原理,并没有那么多。并尝试编写出色的 PyPA 人员多年来无法提出的教程……这可能超出了 SO 答案的范围。

于 2018-04-13T17:35:59.287 回答