Мой установщик python зависает во время работы. Как мне это исправить? - PullRequest
0 голосов
/ 13 сентября 2018

Я сделал очень простой GUI Installer для игры с wxPython. Хотя он предназначен для игры, он технически может быть использован для загрузки и распаковки любого zip-файла, если у вас есть ссылка. Проблема в том, что когда я запускаю программу, графический интерфейс зависает. Он по-прежнему загружает и извлекает данные, как и предполагалось, но графический интерфейс полностью заморожен, пока это происходит. Я не могу обновить текстовые поля или создать панель загрузки, если я не могу разморозить ее. Я знаю, почему это зависает, я просто не знаю, как это исправить. Может ли кто-нибудь помочь мне?

Вот мой код:

import requests, os, sys, zipfile, shutil, subprocess, wx, urllib

url = "{Put any zip file URL here to test the program}"
r = requests.get(url, stream = True)

class Frame(wx.Frame):
    def __init__(self, parent, id, title):
        wx.Frame.__init__(self, parent, id, title, style=wx.DEFAULT_FRAME_STYLE ^ wx.RESIZE_BORDER)
        myPanel = wx.Panel(self,-1)
        myButton = wx.Button(myPanel, -1, 'Download', size=(300,50), pos=(40,350))
        myButton.Bind(wx.EVT_LEFT_DOWN, self.onClick)
        self.Show(True)
    def onClick(self, e):
        print ('Clicked')
        if os.path.exists("RFMB6_WINDOWS"):
            print('\n\nRemoving old RFMP files...')
            subprocess.check_call(('attrib -R ' + 'RFMB6_WINDOWS' + '\\* /S').split())
            shutil.rmtree('RFMB6_WINDOWS')
            print('\nRemoved old files.')
        else:
            pass

        print('\n\nDownloading:')
        urllib.request.urlretrieve(url, 'RFMP.zip')
        print('\nDownload Complete.')
        print('\n\nExtracting...')
        zip_ref = zipfile.ZipFile("RFMP.zip", 'r')
        zip_ref.extractall("RFMB6_WINDOWS")
        zip_ref.close()
        print('\nExtraction Complete')
        print('\n\nCleaning up...')
        os.remove("RFMP.zip")
        print('\nDone! You have succesfully installed the newest version of the Ravenfield Multiplayer Private Alpha.')

app = wx.App()
frame = Frame(None, wx.ID_ANY, 'Image')
app.MainLoop()

1 Ответ

0 голосов
/ 13 сентября 2018

Хорошо, так что я понял это. Как я уже сказал, мне нужно было многопоточность GUI, но в то время я не мог понять это. Теперь у меня есть. То, что мне нужно было сделать, это многопоточный графический интерфейс и мои «процессы» в отдельных потоках. Это означает, что GUI (то, что вы видите) работает в отдельном потоке, чем тот, который обрабатывает данные (загрузка, извлечение и т. Д.).

Прежде чем использовать этот код, я хочу сообщить, что это не законченная программа. Это просто необходимый минимум для загрузки и извлечения ZIP-файла из ссылки на скачивание без остановки.

Вот код, который я сейчас использую для решения своей проблемы:

# Import libraries
import requests, os, sys, zipfile, shutil, subprocess, wx, urllib, time
from threading import *

# Define variables
url = "Put any zip file URL here to test the program"
r = requests.get(url, stream = True)

# Button definitions
ID_START = wx.NewId()

# Define notification event for thread completion
EVT_RESULT_ID = wx.NewId()

# Check  for old files
def Check():
    if os.path.exists("Folder"):
        print('\n\nRemoving old files...')
        subprocess.check_call(('attrib -R ' + 'Folder' + '\\* /S').split())
        shutil.rmtree('Folder')
        print('\nRemoved old files.')
    else:
        pass

# Download new file
def Download():
    print('\n\nDownloading:')
    urllib.request.urlretrieve(url, 'temp.zip')
    print('\nDownload Complete.')

# Extract new file
def Extract():
    print('\n\nExtracting...')
    zip_ref = zipfile.ZipFile("temp.zip", 'r')
    zip_ref.extractall("Folder")
    zip_ref.close()
    print('\nExtraction Complete')

# Delete the .zip file but leave the folder
def Clean():
    print('\n\nCleaning up...')
    os.remove("temp.zip")
    print('\nDone!')

# Thread class that executes processing
class WorkerThread(Thread):
    """Worker Thread Class."""
    def __init__(self, notify_window):
        """Init Worker Thread Class."""
        Thread.__init__(self)
        self._notify_window = notify_window
        # This starts the thread running on creation.
        self.start()

# This is what runs on a separate thread when you click the download button
    def run(self):
        """Run Worker Thread."""
        # This is the code executing in the new thread. 
        Check()
        Download()
        Extract()
        Clean()

# GUI Frame class that spins off the worker thread
class MainFrame(wx.Frame):
    """Class MainFrame."""
    def __init__(self, parent, id):
        """Create the MainFrame."""
        wx.Frame.__init__(self, parent, id, 'GUInstaller', 
                          style=wx.DEFAULT_FRAME_STYLE ^ wx.RESIZE_BORDER)
        self.SetSize(400, 350)
        wx.Button(self, ID_START, 'Download', size=(300,50), pos=(42,250))
        self.status = wx.StaticText(self, -1, '', pos=(0,200))
        self.Bind(wx.EVT_BUTTON, self.OnStart, id=ID_START)

        # And indicate we don't have a worker thread yet
        self.worker = None

    def OnStart(self, event):
        """Start Computation."""
        # Trigger the worker thread unless it's already busy
        if not self.worker:
            self.status.SetLabel('Downloading...')
            self.worker = WorkerThread(self)

    def OnResult(self, event):
        """Show Result status."""
        self.status.SetLabel('Done!')
        # The worker is done
        self.worker = None

class MainApp(wx.App):
    """Class Main App."""
    def OnInit(self):
        """Init Main App."""
        self.frame = MainFrame(None, -1)
        self.frame.Show(True)
        self.SetTopWindow(self.frame)
        return True

# Main Loop
if __name__ == '__main__':
    app = MainApp(0)
    app.MainLoop()

P.S. Не все это мой код. Я взял пример с другого сайта и изменил его, чтобы он соответствовал моим потребностям.

...