KeyboardInterrupt在多线程python中不起作用

时间:2013-07-27 01:45:32

标签: python

我正在尝试多线程来检查网络连接。我的代码是:

exitFlag = 0
lst_doxygen=[]
lst_sphinx=[]
class myThread (threading.Thread):
    def __init__(self, counter):
        threading.Thread.__init__(self)
    self.counter=counter
    def run(self):
    print "Starting thread"
        link_urls(self.counter)

def link_urls(delay):
    global lst_doxygen
    global lst_sphinx
    global exitFlag

    while exitFlag==0:
        try:
            if network_connection() is True:
                try:
                    links = lxml.html.parse(gr.prefs().get_string('grc', 'doxygen_base_uri', '').split(',')[1]+"annotated.html").xpath("//a/@href")
                    for url in links: 
                        lst_doxygen.append(url)
                    links = lxml.html.parse(gr.prefs().get_string('grc', 'sphinx_base_uri', '').split(',')[1]+"genindex.html").xpath("//a/@href")
                    for url in links: 
                        lst_sphinx.append(url)
                    exitFlag=1
                except IOError, AttributeError:
                    pass
            time.sleep(delay)
            print "my"
        except KeyboardInterrupt:
            exitFlag=1


def network_connection():

    network=False
    try:
        response = urllib2.urlopen("http://google.com", None, 2.5)
        network=True

    except urllib2.URLError, e:
           pass
    return network

我设置了一个标志来在循环中停止线程。我还想按Ctrl-C退出该线程。所以我使用了try-except但是线程仍在工作并且不会退出。如果我尝试使用

if KeyboardInterrupt:
    exitFlag=1

而不是try-except,线程只适用于第一次执行while循环然后存在。 附: 我在另一个模块中创建了myThread类的实例。

3 个答案:

答案 0 :(得分:0)

您只能在主线程上获得信号或KeyboardInterrupt。有多种方法可以处理它,但也许你可以使exitFlag成为一个全局的,并将异常处理程序移动到你的主线程。

答案 1 :(得分:0)

以下是我如何抓住CTRL-C的一般方法。

import time
import signal
import sys

stop = False

def run():
    while not stop:
        print 'I am alive'
        time.sleep(3)

def signal_handler(signal, frame):
    global stop
    print 'You pressed Ctrl+C!'
    stop = True

t1 = threading.Thread(target=run)
t1.start()

signal.signal(signal.SIGINT, signal_handler)
print 'Press Ctrl+C'
signal.pause()

输出:

python threads.py
Press Ctrl+C
I am alive
I am alive
^CYou pressed Ctrl+C!

答案 2 :(得分:0)

最后,我得到了我的问题的答案。我需要将我的线程标记为守护进程。因此,当我将myThread类创建实例时,我将再添加一行:

thread1.myThread(2)
thread1.setDaemon(True)
thread1.start()