1

OSX 10.6.8、Python 2.7.2、Tkinter バージョン 8.5 を使用しています。昨日、インストールされた Python、Tkinter、および IDLE をクリーンアップし、Python 3.2 の以前のインストールをクリーンアップしました。私は MIT の Open CourseWare の「RSS フィード フィルター」問題セット (クレジットではありません) に取り組んでおり、それには私が書いていない、理解できないモジュールがいくつか含まれており、そのうちの 1 つ (news_gui の Popup クラス)モジュール) が、デバッグ方法がわからない未処理の例外をスローしています。

# 6.00 Problem Set 5
# RSS Feed Filter


import pdb
#pdb.set_trace()


import feedparser
import string
import time
from project_util import translate_html
from news_gui import Popup

#################################
########## CODE OMITTED #########
#################################

import thread

def main_thread(p):
    # A sample trigger list - you'll replace
    # this with something more configurable in Problem 11
    t1 = SubjectTrigger("Obama")
    t2 = SummaryTrigger("MIT")
    t3 = PhraseTrigger("Supreme Court")
    t4 = OrTrigger(t2, t3)
    triggerlist = [t1, t4]

    # TODO: Problem 11
    # After implementing readTriggerConfig, uncomment this line 
    #triggerlist = readTriggerConfig("triggers.txt")

    guidShown = []

    while True:
        print "Polling..."

        # Get stories from Google's Top Stories RSS news feed
        stories = process("http://news.google.com/?output=rss")
        # Get stories from Yahoo's Top Stories RSS news feed
        stories.extend(process("http://rss.news.yahoo.com/rss/topstories"))

        # Only select stories we're interested in
        stories = filter_stories(stories, triggerlist)

        # Don't print a story if we have already printed it before
        newstories = []
        for story in stories:
            if story.get_guid() not in guidShown:
                newstories.append(story)

        for story in newstories:
            guidShown.append(story.get_guid())
            p.newWindow(story)

        print "Sleeping..."
        time.sleep(SLEEPTIME)

SLEEPTIME = 60 #seconds -- how often we poll
if __name__ == '__main__':
    print '1'
    p = Popup()
    print '2'
    thread.start_new_thread(main_thread, (p,))
    print '3'
    p.start()
    print '4'

p.start() 行 (下から 2 番目) は、news_gui モジュールから Popup を呼び出しています。そのコードは次のとおりです。

import Tkinter
import time

print help(Tkinter.Tk)

#root = Tk()
#root.withdraw()

import pdb


# The Popup class
class Popup:
    def __init__(self):
        print 'Popup init'

        self.root = Tkinter.Tk()
        self.root.withdraw()

        """
    self.root.mainloop()
        def drawALot():
            self.root = Tk()
            self.root.withdraw()
            self.root.mainloop()
        thread.start_new_thread(drawALot, ())
    """
    def start(self):
        print 'Popup start'
        pdb.set_trace()
        self.root.mainloop()

    def _makeTheWindow(self, item):
        """
        Private method that does the actual window drawing
        """
        print 'Popup _makeTheWindow'
        root = Tkinter.Toplevel()
        root.wm_title("News Alert")

        w = root.winfo_screenwidth()/20
        h = root.winfo_screenheight()/4

        title = Tkinter.Text(root, padx=5, pady=5, height=3, wrap=Tkinter.WORD, bg="white")
        title.tag_config("title", foreground="black", font=("helvetica", 12, "bold"))
        title.tag_config("subject", foreground="black", font=("helvetica", 12, "bold"))
        title.insert(Tkinter.INSERT, "Title: %s" % item.get_title(), "title")
        title.insert(Tkinter.INSERT, "\nSubject: ", "subject")
        title.insert(Tkinter.INSERT, item.get_subject().rstrip(), "subject")    

        title.config(state=Tkinter.DISABLED, relief = Tkinter.FLAT)
        title.grid(sticky=Tkinter.W+Tkinter.E)

        print '_makeTheWindow .5'

        summary = Tkinter.Text(root, padx=10, pady=5, height=15, wrap=Tkinter.WORD, bg="white")
        summary.tag_config("text", foreground="black", font=("helvetica", 12))
        summary.insert(Tkinter.INSERT, item.get_summary().lstrip(), "text")

        summary.config(state=Tkinter.DISABLED, relief = Tkinter.FLAT)
        summary.grid(sticky=Tkinter.W+Tkinter.E)

        link = Tkinter.Text(root, padx=5, pady=5, height=4, bg="white")
        link.tag_config("url", foreground="blue", font=("helvetica", 10, "bold"))
        link.insert(Tkinter.INSERT, item.get_link(), "url")

        link.config(state=Tkinter.DISABLED, relief=Tkinter.FLAT)
        link.grid(sticky=Tkinter.W+Tkinter.E)

        print '_makeTheWindow end'

    def newWindow(self, item):
        """
        Displays a popup with the contents of the NewsStory newsitem
        """
        print 'Popup newWindow'
        self.root.after(0, self._makeTheWindow, item)

pdb を有効にして実行すると、出力は次のようになります。

None
1
Popup init
2
3Polling...

Popup start
> /Users/**********/Desktop/Learn/CS Stuffs/6-00sc-spring-2011/6-00sc-spring-2011/contents/unit-2/lecture-12-introduction-to-simulation-and-random-walks/ps5/news_gui.py(31)start()
-> self.root.mainloop()
(Pdb) s
--Call--Unhandled exception in thread started by 
<function main_thread at 0x1026ed848>>
 /Library/Frameworks/Python.framework/Versions/2.7/lib/python2.7/lib-tk/Tkinter.py(1015)mainloop()
-> def mainloop(self, n=0):
(Pdb) s
> /Library/Frameworks/Python.framework/Versions/2.7/lib/python2.7/lib-tk/Tkinter.py(1017)mainloop()
-> self.tk.mainloop(n)
(Pdb) s

そしてハングします。これはスレッド化と Tkinter に関係があると言えますが、次に何をすべきか途方に暮れています。

編集: Windows 7 仮想マシンに Python 2.7.2 をインストールしたところ、そこでも問題が発生します。

4

1 に答える 1

1

全体的な適合性について話すことはできませんが、解決策の何かを見つけました。同様のピクルスで他の誰かを助けるかもしれませんが、より知識のある誰かがそれでチャイムを鳴らすことができれば、それも役に立ちます.

http://tkinter.unpythonic.net/wiki/mtTkinterから mtTkinter をダウンロードしました。「変更されたコードは、スレッド外の Tkinter 呼び出しをインターセプトし、メイン ループで定期的に実行される 'after' イベントによって読み取られるキューを介してそれらをマーシャリングします。」.

正確な方法はわかりませんが、問題は解決しました。

于 2012-04-04T23:31:34.273 に答える