我有一个循环接收套接字数据并打印它的线程:
def post(self):
while True:
try:
data = pickle.loads(self.sock.recv(1024))
print data[0] % tuple(data[1])
except (socket.error, EOFError):
break
然后我有一个GUI将stdout重定向到textctrl,如下所示:
import wx
import sys
import threading
class Redirect:
def __init__(self, ctrl):
self.out = ctrl
def write(self, string):
wx.CallAfter(self.out.AppendText,string)
class GUI(wx.Frame):
def __init__(self, parent):
self.monitor = wx.TextCtrl(self, wx.ID_ANY, \
style = wx.TE_MULTILINE | wx.TE_READONLY)
redir = Redirect(self.monitor)
sys.stdout = redir
self.sizer = BoxSizer(wx.VERTICAL)
self.sizer.Add(self.monitor, 1, wx.GROW | wx.ALL)
self.SetSizer(self.sizer)
self.Show(True)
if __name__ == "__main__":
app = wx.App(False)
frame = GUI(None)
app.MainLoop()
我还有第二个接受输入的textctrl(为简单起见而省略)。问题是有时CallAfter(AppendText,string)不会打印出整个字符串。这是非常罕见的,但有时打印将在字符串的中间突然停止,此时打印下一个字符串(并且应用程序继续打印字符串,因为它们被收到,好像什么也没发生)。
我不知道是什么导致了这种行为,我试图通过键入第二个textctrl来判断是否导致它,但是即使我什么都不做,这些“部分打印”时不时出现。发生了什么事?
答案 0 :(得分:1)
这不是一个真正的答案,只是一种如何复制问题的方法:
import wx
import sys
import threading
def post():
a = 0
for stop in range(100):
a = (a + 1) % 10
data = str(a) * 1000 + " <END>"
print(data)
for i, line in enumerate(frame.monitor.Value.split("\n")[:-1]):
if not line.endswith(" <END>"):
print("Invalid line %d" % (i+1))
class Redirect:
def __init__(self, ctrl):
self.out = ctrl
def write(self, string):
wx.CallAfter(self.out.AppendText, string)
class GUI(wx.Frame):
def __init__(self, parent):
wx.Frame.__init__(self, parent)
self.monitor = wx.TextCtrl(self, wx.ID_ANY, style = wx.TE_MULTILINE | wx.TE_READONLY)
redir = Redirect(self.monitor)
sys.stdout = redir
self.sizer = wx.BoxSizer(wx.VERTICAL)
self.sizer.Add(self.monitor, 1, wx.GROW | wx.ALL)
self.SetSizer(self.sizer)
self.Show(True)
if __name__ == "__main__":
app = wx.App(False)
frame = GUI(None)
t = threading.Thread(target=post)
t.start()
app.MainLoop()
在我的设置上运行时,我得到(通常,并非总是):
Invalid line 30
Invalid line 63
我试过玩延迟和互斥锁/锁,但似乎我没有找到为什么会发生这种情况。