我正在尝试使用Windows DLL的ctypes包装器来调试问题,并且当我通过交互式shell(python或ipython)运行测试时以及当我以非交互方式运行脚本时注意到了差异。
我想知道我在这里看到的差异是否有任何解释?
具体来说,当我以交互方式运行一个简单的测试时,DLL调用将挂起并且永远不会返回,因为与脚本完全相同的代码运行不会出现此问题。
为了更明确我的意思,想象一下你有以下代码
from foobar import bar, foo
bar(foo(1,2,3))
当放入文件时,说“myfoo.py”,并通过“python myfoo.py”进行处理,上面的代码按预期执行。但是,如果你输入上面的python / ipython shell,代码的行为会有所不同(在我的例子中,在调用ctypes.WinDLL函数时会挂起)
其他一些细节:
我在两种情况下都使用相同的解释器和相同的PYTHONPATH。 被封装的DLL是Canon EDSDKv2.9,一个用于远程控制相机的SDK。 它始终挂在DLL中,而不是在python代码中。
初始化时,我的EDSDK包装器启动一个运行方法如下所示的线程:
def run(self):
sys.coinit_flags = 0 #use multithreaded mode
from pythoncom import PumpWaitingMessages
#^^ done here so this thread is correctly initialised
error(EDSDK.EdsInitializeSDK())
self.EDSDK_initialised = True
while self.active:
PumpWaitingMessages()
sleep(self.msg_sleep_time)
error(EDSDK.EdsTerminateSDK())
这个线程的目的基本上是初始化SDK,泵消息,并允许其他线程调用包装方法。
注意:在以前的EDSDK版本中,这在交互式和非交互式方面都有效。我目前的问题只发生在最新版本的EDSDK中。
我怀疑它可能与线程有关(因此是片段),但无法在线找到任何信息来支持我的怀疑。
那么,在以交互方式和非交互方式运行python时,是否有人意识到存在任何差异?可能与Windows线程相关?任何帮助,甚至是疯狂的猜测,都会受到赞赏,因为我完全难过! :)
答案 0 :(得分:1)
Python交互式解释器不是线程安全的。因此,如果您尝试发送阻止命令,整个解释器将挂起。
请参阅this article至于为什么会发生这种情况(tl; dr是IDLE和线程不混合)。至于如何解决这个问题,请使用控制台而不是IDLE GUI。或者,您可以使用脚本。
答案 1 :(得分:1)
在异步或并发操作(例如,多线程,多处理)的上下文中要记住这一点。对于所有意图和目的,当您以交互方式运行Python时,您没有主循环。即使__name__
设置为__main__
,也可能无法在此处分配或执行任何操作。在多处理和某些情况下多线程中尤其如此,其中对象的状态可能不在进程/线程之间共享。
这种行为最多可能令人困惑,最糟糕的情况也很危险。在调试时进入的一个好习惯是始终为线程命名,并使用Python的logging
模块对其进行配对,以跟踪幕后发生的事情。