0

关于代码的一些描述:

  1. 这是我的程序的一部分,我只把相关的行放在这里
  2. 我希望这些代码能做的,是注意我的剪贴板。如果我将“ http:xxx ”复制到我的剪贴板,它会显示一个弹出窗口。如果剪贴板的内容没有改变,则窗口不会' t 再次显示
  3. 运行时可以正常弹窗一次,但是当我将另一个以'http: '开头的字符串复制到剪贴板时就不会再弹了
  4. 我在after方法中尝试了一些不同的间隔值,结果相同。

代码:

from tkinter import *
import os
import tkinter.messagebox as messagebox
import threading
import re

def watch_clipboard(tk,pipeout):  
    content = '' 
    last_content = ''
    while True:
        try:
            content = tk.clipboard_get()
        except TclError:
            pass
        result = re.match('http:',content)
        if content != last_content:
            if result:
                last_content = content
                message = 'show'.encode()    
                os.write(pipeout,message)


class GUI:
    def __init__(self):
        self.tk = Tk()
        self.tk.resizable(0, 0)
        self.tk.title('watch clipboard')
        pipein,pipeout = os.pipe()
        threading.Thread(target=watch_clipboard,daemon=True,args=(self.tk,pipeout)).start()
        self.tk.after(5000,lambda:self.clipboard_confirm(pipein))
        self.tk.mainloop()

    def clipboard_confirm(self,pipein):
        message = os.read(pipein,16)
        if message == b'show':
            self.tk.clipboard_clear()
            messagebox.askokcancel('', 'add this in?', default='ok')
            self.tk.after(5000,clipboard_confirm(pipein))   #add this


if __name__ == '__main__':
    gui = GUI()

编辑:A。罗达斯的代码有效。似乎是多线程导致了问题。深层原因仍然未知。

4

3 回答 3

1

我认为多线程不是适合您情况的最佳方法,after可能就足够了。tk.after您可以使用较低的超时时间来代替繁忙的等待循环。然后你只需要将逻辑移动watch_clipboard到你的类中,这样你就不用担心线程之间的通信了。

class GUI:
    def __init__(self):
        self.tk = Tk()
        self.tk.resizable(0, 0)
        self.tk.title('watch clipboard')
        self.last_content = ''
        self.tk.after(100, self.watch_clipboard)
        self.tk.mainloop()
    def watch_clipboard(self):
        try:
            content = self.tk.clipboard_get()
            if content != self.last_content and content.startswith('http:'):
                self.last_content = content
                self.tk.clipboard_clear()
                messagebox.askokcancel('', 'add this in?', default='ok')
        except TclError:
            pass
        self.tk.after(100, self.watch_clipboard)
于 2013-03-23T19:00:35.500 回答
1

@laike9m
您的代码错误是使用了 os.pipe 。
os.read() 是一个阻塞函数,一旦clipboard_confirm运行就会收到 os.read() 阻塞。导致 UI 卡住。这个bug和多线程
无关。after

于 2019-02-16T13:22:11.273 回答
0

这个问题很可能与使用线程有关——从主线程以外的任何地方调用 Tkinter 函数都可能导致问题。

您是否有理由使用线程而不是利用内置无限循环(事件循环)?我建议after每隔一秒左右检查一次键盘。

于 2013-03-23T19:02:20.570 回答