python异步回调转为同步并实现超时
问题描述
场景:一个服务端A,一个客户端B,存在一个socket连接。现在写的是客户端B部分,服务端不可控。原来是 B先发送一个包,等待A返回指定内容,B再发送下一个包
def do(): s.send(...) yield 1 s.send(...) yield 2# 接收到数据后的回调def callback(): global f next(f) f=do()next(f)
现在想实现一个timeout,并且实现阻塞。B发送数据后阻塞,直到A返回数据(或5秒内未接受到来自A的返回raise一个错误),请教如何实现?
问题解答
回答1:用 Tornado 的话,写不了几行代码吧。
先作个简单的 Server ,以方便演示:
# -*- coding: utf-8 -*-from tornado.ioloop import IOLoopfrom tornado.tcpserver import TCPServerfrom tornado import genclass Server(TCPServer): @gen.coroutine def handle_stream(self, stream, address):while 1: data = yield stream.read_until(’n’) if data.strip() == ’exit’:stream.close()break if data.strip() == ’5’:IOLoop.current().call_at(IOLoop.current().time() + 5, lambda: stream.write(’ok 5n’)) else:stream.write(’okn’)if __name__ == ’__main__’: Server().listen(8000) IOLoop.current().start()
然后,来实现 Client ,基本逻辑是,超时就关闭连接,然后再重新建立连接:
# -*- coding: utf-8 -*-import functoolsfrom tornado.ioloop import IOLoopfrom tornado.tcpclient import TCPClientfrom tornado import gendef when_error(stream): print ’ERROR’ stream.close() main()@gen.coroutinedef main(): client = TCPClient() stream = yield client.connect(’localhost’, 8000) count = 0 IL = IOLoop.current() while 1:count += 1stream.write(str(count) + ’n’)print count, ’...’timer = IL.call_at(IL.time() + 4, functools.partial(when_error, stream))try: data = yield stream.read_until(’n’)except: breakIL.remove_timeout(timer)print datayield gen.Task(IL.add_timeout, IOLoop.current().time() + 1)if __name__ == ’__main__’: main() IOLoop.current().start()
相关文章:
1. docker gitlab 如何git clone?2. docker-compose 为何找不到配置文件?3. docker images显示的镜像过多,狗眼被亮瞎了,怎么办?4. docker不显示端口映射呢?5. dockerfile - [docker build image失败- npm install]6. docker内创建jenkins访问另一个容器下的服务器问题7. angular.js - 如何从AngularJS迁移到Vue?8. docker 17.03 怎么配置 registry mirror ?9. docker 下面创建的IMAGE 他们的 ID 一样?这个是怎么回事????10. javascript - 如何申请blog.xxx.com 这种二级域名?
