为什么解释的顺序与我的期望有所不同?

时间:2010-03-13 02:12:38

标签: ironpython winforms delayed-execution

我有一个我以前没有遇到过的问题:

似乎我的程序中的解释顺序与我的预期有所不同。我写了一个小的Twitter客户端。单击“GO”按钮后,我的程序实际发布推文需要几秒钟(也可以通过按键盘上的ENTER激活)。我不想在这段时间内多次点击,因为我第一次没有点击它。因此,当单击该按钮时,我希望标签文本显示一些信息,告诉我该按钮已被点击。

我通过在发送推文之前更改标签文本来实现此消息。但是,由于某种原因,在尝试推文之前不会显示该消息。但由于我在推文后收到了确认信息,因此我从未看到此消息,原来的问题仍未得到解决。

我真的很感激任何帮助。以下是相关代码:

class SimpleTextBoxForm(Form):

  def __init__(self):

    # set window properties

    self.Text = "Tweeter"
    self.Width = 235
    self.Height = 250

    #tweet away

    self.label = Label()
    self.label.Text = "Tweet Away..."
    self.label.Location = Point(10, 10)
    self.label.Height = 25
    self.label.Width = 200

    #get the tweet

    self.tweetBox = TextBox()
    self.tweetBox.Location = Point(10, 45)
    self.tweetBox.Width = 200
    self.tweetBox.Height = 60
    self.tweetBox.Multiline = True
    self.tweetBox.WordWrap = True
    self.tweetBox.MaxLength = 140;

    #ask for the login ID

    self.askLogin = Label()
    self.askLogin.Text = "Login:"
    self.askLogin.Location = Point(10, 120)
    self.askLogin.Height = 20
    self.askLogin.Width = 60

    self.login = TextBox()
    self.login.Text= ""
    self.login.Location = Point(80, 120)
    self.login.Height = 40
    self.login.Width = 100

    #ask for the password

    self.askPass = Label()
    self.askPass.Text = "Password:"
    self.askPass.Location = Point(10, 150)
    self.askPass.Height = 20
    self.askPass.Width = 60

    # display password box with character hiding

    self.password = TextBox()
    self.password.Location = Point(80, 150)
    self.password.PasswordChar = "x"
    self.password.Height = 40
    self.password.Width = 100

    #submit button

    self.button1 = Button()
    self.button1.Text = 'Tweet'
    self.button1.Location = Point(10, 180)
    self.button1.Click += self.update

    self.AcceptButton = self.button1

    #pack all the elements of the form

    self.Controls.Add(self.label)
    self.Controls.Add(self.tweetBox)
    self.Controls.Add(self.askLogin)
    self.Controls.Add(self.login)
    self.Controls.Add(self.askPass)
    self.Controls.Add(self.password)
    self.Controls.Add(self.button1)

  def update(self, sender, event):

    if not self.password.Text:
        self.label.Text = "You forgot to enter your password..."
    else:
        self.tweet(self.tweetBox.Text, self.login.Text, self.password.Text)

  def tweet(self, msg, login, password):

    self.label.Text = "Attempting Tweet..."  # this should be executed before sending the tweet is attempted. But this seems to be executed only after the try block

    try:
        success = 'Tweet successfully completed... yay!\n' + 'At: ' + time.asctime().split()[3]

        ServicePointManager.Expect100Continue = False
        Twitter().UpdateAsXML(login, password, msg)

    except:
        error = 'Unhandled Exception. Tweet unsuccessful'
        self.label.Text = error

    else:
        self.label.Text = success
        self.tweetBox.Text = ""

1 个答案:

答案 0 :(得分:2)

解释/执行顺序是正确的,但正在发生的事情是,在进入阻止Web请求之前,您没有给Windows Forms时间来更改标签文本。基本上设置label.Text将消息发布到表单请求文本被更改,但在从事件处理程序返回之前它将不会被操作(并且表单再次开始传送消息)。目前,您没有从事件处理程序返回,直到Web请求完成,此时(a)消息无关紧要,(b)label.Text立即被下一个排队更新覆盖

解决方案是在后台线程中执行Web服务请求,例如使用BackgroundWorker组件。这允许表单在后台线程运行时立即再次开始传送消息,因此标签将立即更新。但请注意,这也意味着表单的其余部分将是响应式的,这意味着用户可以再次按下按钮,因此您可能希望禁用某些内容以及仅更新状态。