gevent和它的monkey补丁还是挺好用的,但是有时候它也有坑。gevent monkey补丁作用是将阻塞改成非阻塞以提升运行速度,一般都是
from gevent import monkey;monkey.patchall()
但当我做多进程加多线程加多协程实验的时候出现了无限递归的问题,仔细查看monkey.patchall()如下
def patch_all(socket=True, dns=True, time=True, select=True, thread=True, os=True, ssl=True,
httplib=False, # Deprecated, to be removed.
subprocess=True, sys=False, aggressive=True, Event=True,
builtins=True, signal=True,
queue=True,
**kwargs):
"""
Do all of the default monkey patching (calls every other applicable
function in this module).
:return: A true value if patching all modules wasn't cancelled, a false
value if it was.
在我的实验场景下把thread=True改成thread=False就能解决问题,即:
from gevent import monkey;monkey.patchall(thread=False)
总结一下:应该就是monkey补丁将thread从阻塞改成非阻塞而导致的问题,把thread方面的补丁取消即可
如果你也遇到了类似的问题,可以考虑修改相关的参数,可能是因为monkey补丁将某些应该阻塞的地方改成了非阻塞
附gevent的一些缺陷:
- 阻塞(真正的阻塞,在内核级别):在程序中的某个地方停止了所有的东西.这很像C代码中monkey patch没有生效
- 保持CPU处于繁忙状态:greenlet不是抢占式的,这可能导致其他greenlet不会被调度
- 在greenlet之间存在死锁的可能