2010-01-26 104 views
8

我設置了一個進程來讀取一個隊列來傳入urls來下載,但是當urllib2打開一個連接時系統掛起。Python進程被urllib2阻止

import urllib2, multiprocessing 
from threading import Thread 
from Queue import Queue 
from multiprocessing import Queue as ProcessQueue, Process 

def download(url): 
    """Download a page from an url. 
    url [str]: url to get. 
    return [unicode]: page downloaded. 
    """ 
    if settings.DEBUG: 
     print u'Downloading %s' % url 
    request = urllib2.Request(url) 
    response = urllib2.urlopen(request) 
    encoding = response.headers['content-type'].split('charset=')[-1] 
    content = unicode(response.read(), encoding) 
    return content 

def downloader(url_queue, page_queue): 
    def _downloader(url_queue, page_queue): 
     while True: 
      try: 
       url = url_queue.get() 
       page_queue.put_nowait({'url': url, 'page': download(url)}) 
      except Exception, err: 
       print u'Error downloading %s' % url 
       raise err 
      finally: 
       url_queue.task_done() 

    ## Init internal workers 
    internal_url_queue = Queue() 
    internal_page_queue = Queue() 
    for num in range(multiprocessing.cpu_count()): 
     worker = Thread(target=_downloader, args=(internal_url_queue, internal_page_queue)) 
     worker.setDaemon(True) 
     worker.start() 

    # Loop waiting closing 
    for url in iter(url_queue.get, 'STOP'): 
     internal_url_queue.put(url) 

    # Wait for closing 
    internal_url_queue.join() 

# Init the queues 
url_queue = ProcessQueue() 
page_queue = ProcessQueue() 

# Init the process 
download_worker = Process(target=downloader, args=(url_queue, page_queue)) 
download_worker.start() 

從另一個模塊我可以添加網址,當我想我可以停止進程並等待進程關閉。

import module 

module.url_queue.put('http://foobar1') 
module.url_queue.put('http://foobar2') 
module.url_queue.put('http://foobar3') 
module.url_queue.put('STOP') 
downloader.download_worker.join() 

問題是,當我使用urlopen(「response = urllib2.urlopen(request)」)時,它仍然全部被阻止。

如果我調用download()函數或僅使用沒有Process的線程時,沒有任何問題。

回答

4

這裏的問題不是urllib2,而是使用多處理模塊。在Windows下使用多處理模塊時,不得使用在導入模塊時立即運行的代碼,而應將主模塊中的東西放入if __name__=='__main__'塊中。請參見「安全導入主模塊」here

爲您的代碼,使這一變化的下載模塊中的以下內容:

#.... 
def start(): 
    global download_worker 
    download_worker = Process(target=downloader, args=(url_queue, page_queue)) 
    download_worker.start() 

和主要模塊:

import module 
if __name__=='__main__': 
    module.start() 
    module.url_queue.put('http://foobar1') 
    #.... 

因爲你沒有做到這一點,每次子是開始它會再次運行主代碼並啓動另一個進程,導致掛起。

+0

我不使用Windows,但建議使用start()函數修復此問題。謝謝! – Davmuz 2010-01-26 16:07:59