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. debian - docker依赖的aufs-tools源码哪里可以找到啊?2. 为什么我ping不通我的docker容器呢???3. 对html实现监测 发现不对4. dockerfile - 为什么docker容器启动不了?5. docker内创建jenkins访问另一个容器下的服务器问题6. angular.js - angular内容过长展开收起效果7. docker - 各位电脑上有多少个容器啊?容器一多,自己都搞混了,咋办呢?8. javascript - js中向下取整9. 在mac下出现了两个docker环境10. angular.js使用$resource服务把数据存入mongodb的问题。

网公网安备