background function in Python

前端 未结 3 1072
名媛妹妹
名媛妹妹 2020-11-27 03:15

I\'ve got a Python script that sometimes displays images to the user. The images can, at times, be quite large, and they are reused often. Displaying them is not critical, b

相关标签:
3条回答
  • 2020-11-27 03:58

    Do something like this:

    def function_that_downloads(my_args):
        # do some long download here
    

    then inline, do something like this:

    import threading
    def my_inline_function(some_args):
        # do some stuff
        download_thread = threading.Thread(target=function_that_downloads, name="Downloader", args=some_args)
        download_thread.start()
        # continue doing stuff
    

    You may want to check if the thread has finished before going on to other things by calling download_thread.isAlive()

    0 讨论(0)
  • 2020-11-27 04:04

    Typically the way to do this would be to use a thread pool and queue downloads which would issue a signal, a.k.a an event, when that task has finished processing. You can do this within the scope of the threading module Python provides.

    To perform said actions, I would use event objects and the Queue module.

    However, a quick and dirty demonstration of what you can do using a simple threading.Thread implementation can be seen below:

    import os
    import threading
    import time
    import urllib2
    
    
    class ImageDownloader(threading.Thread):
    
        def __init__(self, function_that_downloads):
            threading.Thread.__init__(self)
            self.runnable = function_that_downloads
            self.daemon = True
    
        def run(self):
            self.runnable()
    
    
    def downloads():
        with open('somefile.html', 'w+') as f:
            try:
                f.write(urllib2.urlopen('http://google.com').read())
            except urllib2.HTTPError:
                f.write('sorry no dice')
    
    
    print 'hi there user'
    print 'how are you today?'
    thread = ImageDownloader(downloads)
    thread.start()
    while not os.path.exists('somefile.html'):
        print 'i am executing but the thread has started to download'
        time.sleep(1)
    
    print 'look ma, thread is not alive: ', thread.is_alive()
    

    It would probably make sense to not poll like I'm doing above. In which case, I would change the code to this:

    import os
    import threading
    import time
    import urllib2
    
    
    class ImageDownloader(threading.Thread):
    
        def __init__(self, function_that_downloads):
            threading.Thread.__init__(self)
            self.runnable = function_that_downloads
    
        def run(self):
            self.runnable()
    
    
    def downloads():
        with open('somefile.html', 'w+') as f:
            try:
                f.write(urllib2.urlopen('http://google.com').read())
            except urllib2.HTTPError:
                f.write('sorry no dice')
    
    
    print 'hi there user'
    print 'how are you today?'
    thread = ImageDownloader(downloads)
    thread.start()
    # show message
    thread.join()
    # display image
    

    Notice that there's no daemon flag set here.

    0 讨论(0)
  • 2020-11-27 04:09

    I prefer to use gevent for this sort of thing:

    import gevent
    from gevent import monkey; monkey.patch_all()
    
    greenlet = gevent.spawn( function_to_download_image )
    display_message()
    # ... perhaps interaction with the user here
    
    # this will wait for the operation to complete (optional)
    greenlet.join()
    # alternatively if the image display is no longer important, this will abort it:
    #greenlet.kill()
    

    Everything runs in one thread, but whenever a kernel operation blocks, gevent switches contexts when there are other "greenlets" running. Worries about locking, etc are much reduced, as there is only one thing running at a time, yet the image will continue to download whenever a blocking operation executes in the "main" context.

    Depending on how much, and what kind of thing you want to do in the background, this can be either better or worse than threading-based solutions; certainly, it is much more scaleable (ie you can do many more things in the background), but that might not be of concern in the current situation.

    0 讨论(0)
提交回复
热议问题